sonamu 0.5.7 → 0.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api/base-frame.js +12 -2
- package/dist/api/caster.js +66 -2
- package/dist/api/code-converters.js +489 -2
- package/dist/api/config.d.ts +76 -0
- package/dist/api/config.d.ts.map +1 -0
- package/dist/api/config.js +32 -0
- package/dist/api/context.d.ts +1 -0
- package/dist/api/context.d.ts.map +1 -1
- package/dist/api/context.js +3 -2
- package/dist/api/decorators.d.ts.map +1 -1
- package/dist/api/decorators.js +142 -2
- package/dist/api/index.js +9 -2
- package/dist/api/sonamu.d.ts +8 -22
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +482 -2
- package/dist/bin/build-config.d.ts +2 -1
- package/dist/bin/build-config.d.ts.map +1 -1
- package/dist/bin/build-config.js +12 -2
- package/dist/bin/cli-wrapper.js +71 -2
- package/dist/bin/cli.js +418 -2
- package/dist/bin/hot-hook-register.d.ts +11 -0
- package/dist/bin/hot-hook-register.d.ts.map +1 -0
- package/dist/bin/hot-hook-register.js +21 -0
- package/dist/database/_batch_update.js +78 -2
- package/dist/database/base-model.js +247 -2
- package/dist/database/code-generator.js +53 -2
- package/dist/database/db.d.ts +2 -16
- package/dist/database/db.d.ts.map +1 -1
- package/dist/database/db.js +132 -2
- package/dist/database/knex-plugins/knex-on-duplicate-update.js +39 -2
- package/dist/database/puri-wrapper.js +109 -2
- package/dist/database/puri.d.ts +23 -16
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +539 -2
- package/dist/database/puri.types.d.ts +8 -3
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +3 -2
- package/dist/database/transaction-context.js +14 -2
- package/dist/database/upsert-builder.js +215 -2
- package/dist/entity/entity-manager.d.ts +3 -1
- package/dist/entity/entity-manager.d.ts.map +1 -1
- package/dist/entity/entity-manager.js +114 -2
- package/dist/entity/entity-utils.js +210 -2
- package/dist/entity/entity.d.ts.map +1 -1
- package/dist/entity/entity.js +651 -2
- package/dist/exceptions/error-handler.js +29 -2
- package/dist/exceptions/so-exceptions.js +85 -2
- package/dist/file-storage/driver.js +79 -2
- package/dist/file-storage/file-storage.js +75 -2
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +28 -2
- package/dist/migration/code-generation.js +558 -2
- package/dist/migration/migration-set.js +364 -2
- package/dist/migration/migrator.d.ts +0 -9
- package/dist/migration/migrator.d.ts.map +1 -1
- package/dist/migration/migrator.js +510 -2
- package/dist/migration/types.js +3 -2
- package/dist/naite/naite.d.ts +12 -0
- package/dist/naite/naite.d.ts.map +1 -0
- package/dist/naite/naite.js +72 -0
- package/dist/stream/index.js +3 -2
- package/dist/stream/sse.js +38 -2
- package/dist/syncer/api-parser.d.ts +20 -0
- package/dist/syncer/api-parser.d.ts.map +1 -0
- package/dist/syncer/api-parser.js +229 -0
- package/dist/syncer/checksum.d.ts +21 -0
- package/dist/syncer/checksum.d.ts.map +1 -0
- package/dist/syncer/checksum.js +98 -0
- package/dist/syncer/code-generator.d.ts +20 -0
- package/dist/syncer/code-generator.d.ts.map +1 -0
- package/dist/syncer/code-generator.js +141 -0
- package/dist/syncer/entity-operations.d.ts +17 -0
- package/dist/syncer/entity-operations.d.ts.map +1 -0
- package/dist/syncer/entity-operations.js +58 -0
- package/dist/syncer/file-patterns.d.ts +29 -0
- package/dist/syncer/file-patterns.d.ts.map +1 -0
- package/dist/syncer/file-patterns.js +38 -0
- package/dist/syncer/index.d.ts +6 -0
- package/dist/syncer/index.d.ts.map +1 -1
- package/dist/syncer/index.js +9 -2
- package/dist/syncer/module-loader.d.ts +35 -0
- package/dist/syncer/module-loader.d.ts.map +1 -0
- package/dist/syncer/module-loader.js +82 -0
- package/dist/syncer/syncer.d.ts +93 -108
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +375 -2
- package/dist/template/entity-converter.d.ts +14 -0
- package/dist/template/entity-converter.d.ts.map +1 -0
- package/dist/template/entity-converter.js +101 -0
- package/dist/template/helpers.d.ts +23 -0
- package/dist/template/helpers.d.ts.map +1 -0
- package/dist/template/helpers.js +64 -0
- package/dist/{templates → template/implementations}/entity.template.d.ts +3 -3
- package/dist/template/implementations/entity.template.d.ts.map +1 -0
- package/dist/template/implementations/entity.template.js +87 -0
- package/dist/{templates → template/implementations}/generated.template.d.ts +3 -3
- package/dist/template/implementations/generated.template.d.ts.map +1 -0
- package/dist/template/implementations/generated.template.js +232 -0
- package/dist/{templates → template/implementations}/generated_http.template.d.ts +3 -3
- package/dist/template/implementations/generated_http.template.d.ts.map +1 -0
- package/dist/template/implementations/generated_http.template.js +131 -0
- package/dist/{templates → template/implementations}/generated_sso.template.d.ts +3 -3
- package/dist/template/implementations/generated_sso.template.d.ts.map +1 -0
- package/dist/template/implementations/generated_sso.template.js +105 -0
- package/dist/{templates → template/implementations}/init_types.template.d.ts +3 -3
- package/dist/template/implementations/init_types.template.d.ts.map +1 -0
- package/dist/template/implementations/init_types.template.js +38 -0
- package/dist/template/implementations/model.template.d.ts +17 -0
- package/dist/template/implementations/model.template.d.ts.map +1 -0
- package/dist/template/implementations/model.template.js +171 -0
- package/dist/{templates → template/implementations}/model_test.template.d.ts +3 -3
- package/dist/template/implementations/model_test.template.d.ts.map +1 -0
- package/dist/template/implementations/model_test.template.js +35 -0
- package/dist/{templates → template/implementations}/service.template.d.ts +6 -6
- package/dist/template/implementations/service.template.d.ts.map +1 -0
- package/dist/template/implementations/service.template.js +193 -0
- package/dist/{templates → template/implementations}/view_enums_buttonset.template.d.ts +3 -3
- package/dist/template/implementations/view_enums_buttonset.template.d.ts.map +1 -0
- package/dist/template/implementations/view_enums_buttonset.template.js +31 -0
- package/dist/{templates → template/implementations}/view_enums_dropdown.template.d.ts +3 -4
- package/dist/template/implementations/view_enums_dropdown.template.d.ts.map +1 -0
- package/dist/template/implementations/view_enums_dropdown.template.js +50 -0
- package/dist/{templates → template/implementations}/view_enums_select.template.d.ts +3 -3
- package/dist/template/implementations/view_enums_select.template.d.ts.map +1 -0
- package/dist/template/implementations/view_enums_select.template.js +55 -0
- package/dist/{templates → template/implementations}/view_form.template.d.ts +5 -5
- package/dist/template/implementations/view_form.template.d.ts.map +1 -0
- package/dist/template/implementations/view_form.template.js +337 -0
- package/dist/{templates → template/implementations}/view_id_all_select.template.d.ts +3 -3
- package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -0
- package/dist/template/implementations/view_id_all_select.template.js +31 -0
- package/dist/{templates → template/implementations}/view_id_async_select.template.d.ts +3 -3
- package/dist/template/implementations/view_id_async_select.template.d.ts.map +1 -0
- package/dist/template/implementations/view_id_async_select.template.js +105 -0
- package/dist/{templates → template/implementations}/view_list.template.d.ts +5 -13
- package/dist/template/implementations/view_list.template.d.ts.map +1 -0
- package/dist/template/implementations/view_list.template.js +465 -0
- package/dist/{templates → template/implementations}/view_list_columns.template.d.ts +3 -3
- package/dist/template/implementations/view_list_columns.template.d.ts.map +1 -0
- package/dist/template/implementations/view_list_columns.template.js +49 -0
- package/dist/{templates → template/implementations}/view_search_input.template.d.ts +3 -3
- package/dist/template/implementations/view_search_input.template.d.ts.map +1 -0
- package/dist/template/implementations/view_search_input.template.js +64 -0
- package/dist/template/index.d.ts +5 -0
- package/dist/template/index.d.ts.map +1 -0
- package/dist/template/index.js +6 -0
- package/dist/template/template.d.ts +39 -0
- package/dist/template/template.d.ts.map +1 -0
- package/dist/template/template.js +47 -0
- package/dist/template/zod-converter.d.ts +18 -0
- package/dist/template/zod-converter.d.ts.map +1 -0
- package/dist/template/zod-converter.js +166 -0
- package/dist/testing/_relation-graph.js +80 -2
- package/dist/testing/fixture-manager.d.ts.map +1 -1
- package/dist/testing/fixture-manager.js +521 -2
- package/dist/types/types.d.ts +39 -40
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +289 -2
- package/dist/typings/knex.d.js +3 -2
- package/dist/utils/async-utils.d.ts +7 -0
- package/dist/utils/async-utils.d.ts.map +1 -1
- package/dist/utils/async-utils.js +57 -2
- package/dist/utils/console-util.d.ts +2 -0
- package/dist/utils/console-util.d.ts.map +1 -0
- package/dist/utils/console-util.js +6 -0
- package/dist/utils/controller.js +26 -2
- package/dist/utils/esm-utils.d.ts +45 -0
- package/dist/utils/esm-utils.d.ts.map +1 -0
- package/dist/utils/esm-utils.js +56 -0
- package/dist/utils/fs-utils.js +17 -2
- package/dist/utils/lodash-able.js +6 -2
- package/dist/utils/model.js +22 -2
- package/dist/utils/path-utils.d.ts +89 -0
- package/dist/utils/path-utils.d.ts.map +1 -0
- package/dist/utils/path-utils.js +60 -0
- package/dist/utils/process-utils.d.ts +13 -0
- package/dist/utils/process-utils.d.ts.map +1 -0
- package/dist/utils/process-utils.js +36 -0
- package/dist/utils/sql-parser.js +35 -2
- package/dist/utils/utils.d.ts +4 -7
- package/dist/utils/utils.d.ts.map +1 -1
- package/dist/utils/utils.js +33 -2
- package/dist/utils/zod-error.d.ts.map +1 -1
- package/dist/utils/zod-error.js +19 -2
- package/package.json +21 -8
- package/src/api/code-converters.ts +2 -2
- package/src/api/config.ts +142 -0
- package/src/api/context.ts +1 -0
- package/src/api/decorators.ts +1 -0
- package/src/api/sonamu.ts +81 -67
- package/src/bin/build-config.ts +2 -1
- package/src/bin/cli-wrapper.ts +10 -3
- package/src/bin/cli.ts +108 -56
- package/src/bin/hot-hook-register.ts +22 -0
- package/src/database/base-model.ts +1 -1
- package/src/database/code-generator.ts +1 -1
- package/src/database/db.ts +10 -52
- package/src/database/puri.ts +78 -53
- package/src/database/puri.types.ts +18 -5
- package/src/database/upsert-builder.ts +1 -1
- package/src/entity/entity-manager.ts +19 -15
- package/src/entity/entity.ts +4 -3
- package/src/index.ts +2 -0
- package/src/migration/code-generation.ts +1 -1
- package/src/migration/migration-set.ts +1 -1
- package/src/migration/migrator.ts +23 -152
- package/src/naite/naite.ts +70 -0
- package/src/syncer/api-parser.ts +299 -0
- package/src/syncer/checksum.ts +152 -0
- package/src/syncer/code-generator.ts +202 -0
- package/src/syncer/entity-operations.ts +68 -0
- package/src/syncer/file-patterns.ts +56 -0
- package/src/syncer/index.ts +6 -0
- package/src/syncer/module-loader.ts +125 -0
- package/src/syncer/syncer.ts +363 -1420
- package/src/template/entity-converter.ts +123 -0
- package/src/template/helpers.ts +84 -0
- package/src/{templates → template/implementations}/entity.template.ts +4 -4
- package/src/{templates → template/implementations}/generated.template.ts +9 -9
- package/src/{templates → template/implementations}/generated_http.template.ts +9 -6
- package/src/{templates → template/implementations}/generated_sso.template.ts +7 -7
- package/src/{templates → template/implementations}/init_types.template.ts +4 -4
- package/src/{templates → template/implementations}/model.template.ts +9 -9
- package/src/{templates → template/implementations}/model_test.template.ts +5 -5
- package/src/{templates → template/implementations}/service.template.ts +19 -11
- package/src/{templates → template/implementations}/view_enums_buttonset.template.ts +3 -3
- package/src/{templates → template/implementations}/view_enums_dropdown.template.ts +5 -21
- package/src/{templates → template/implementations}/view_enums_select.template.ts +4 -4
- package/src/{templates → template/implementations}/view_form.template.ts +11 -13
- package/src/{templates → template/implementations}/view_id_all_select.template.ts +3 -3
- package/src/{templates → template/implementations}/view_id_async_select.template.ts +3 -3
- package/src/{templates → template/implementations}/view_list.template.ts +13 -64
- package/src/{templates → template/implementations}/view_list_columns.template.ts +3 -3
- package/src/{templates → template/implementations}/view_search_input.template.ts +3 -3
- package/src/template/index.ts +4 -0
- package/src/template/template.ts +86 -0
- package/src/template/zod-converter.ts +219 -0
- package/src/testing/fixture-manager.ts +8 -1
- package/src/types/types.ts +38 -61
- package/src/utils/async-utils.ts +17 -0
- package/src/utils/console-util.ts +4 -0
- package/src/utils/esm-utils.ts +69 -0
- package/src/utils/path-utils.ts +102 -0
- package/src/utils/process-utils.ts +46 -0
- package/src/utils/sql-parser.ts +1 -1
- package/src/utils/utils.ts +14 -40
- package/src/utils/zod-error.ts +0 -1
- package/dist/api/base-frame.js.map +0 -1
- package/dist/api/caster.js.map +0 -1
- package/dist/api/code-converters.js.map +0 -1
- package/dist/api/context.js.map +0 -1
- package/dist/api/decorators.js.map +0 -1
- package/dist/api/index.js.map +0 -1
- package/dist/api/sonamu.js.map +0 -1
- package/dist/bin/build-config.js.map +0 -1
- package/dist/bin/cli-wrapper.js.map +0 -1
- package/dist/bin/cli.js.map +0 -1
- package/dist/database/_batch_update.js.map +0 -1
- package/dist/database/base-model.js.map +0 -1
- package/dist/database/code-generator.js.map +0 -1
- package/dist/database/db.js.map +0 -1
- package/dist/database/knex-plugins/knex-on-duplicate-update.js.map +0 -1
- package/dist/database/puri-wrapper.js.map +0 -1
- package/dist/database/puri.js.map +0 -1
- package/dist/database/puri.types.js.map +0 -1
- package/dist/database/transaction-context.js.map +0 -1
- package/dist/database/upsert-builder.js.map +0 -1
- package/dist/entity/entity-manager.js.map +0 -1
- package/dist/entity/entity-utils.js.map +0 -1
- package/dist/entity/entity.js.map +0 -1
- package/dist/exceptions/error-handler.js.map +0 -1
- package/dist/exceptions/so-exceptions.js.map +0 -1
- package/dist/file-storage/driver.js.map +0 -1
- package/dist/file-storage/file-storage.js.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/migration/code-generation.js.map +0 -1
- package/dist/migration/migration-set.js.map +0 -1
- package/dist/migration/migrator.js.map +0 -1
- package/dist/migration/types.js.map +0 -1
- package/dist/stream/index.js.map +0 -1
- package/dist/stream/sse.js.map +0 -1
- package/dist/syncer/index.js.map +0 -1
- package/dist/syncer/syncer.js.map +0 -1
- package/dist/templates/base-template.d.ts +0 -13
- package/dist/templates/base-template.d.ts.map +0 -1
- package/dist/templates/base-template.js +0 -2
- package/dist/templates/base-template.js.map +0 -1
- package/dist/templates/entity.template.d.ts.map +0 -1
- package/dist/templates/entity.template.js +0 -2
- package/dist/templates/entity.template.js.map +0 -1
- package/dist/templates/generated.template.d.ts.map +0 -1
- package/dist/templates/generated.template.js +0 -2
- package/dist/templates/generated.template.js.map +0 -1
- package/dist/templates/generated_http.template.d.ts.map +0 -1
- package/dist/templates/generated_http.template.js +0 -2
- package/dist/templates/generated_http.template.js.map +0 -1
- package/dist/templates/generated_sso.template.d.ts.map +0 -1
- package/dist/templates/generated_sso.template.js +0 -2
- package/dist/templates/generated_sso.template.js.map +0 -1
- package/dist/templates/index.d.ts +0 -2
- package/dist/templates/index.d.ts.map +0 -1
- package/dist/templates/index.js +0 -2
- package/dist/templates/index.js.map +0 -1
- package/dist/templates/init_types.template.d.ts.map +0 -1
- package/dist/templates/init_types.template.js +0 -2
- package/dist/templates/init_types.template.js.map +0 -1
- package/dist/templates/model.template.d.ts +0 -17
- package/dist/templates/model.template.d.ts.map +0 -1
- package/dist/templates/model.template.js +0 -2
- package/dist/templates/model.template.js.map +0 -1
- package/dist/templates/model_test.template.d.ts.map +0 -1
- package/dist/templates/model_test.template.js +0 -2
- package/dist/templates/model_test.template.js.map +0 -1
- package/dist/templates/service.template.d.ts.map +0 -1
- package/dist/templates/service.template.js +0 -2
- package/dist/templates/service.template.js.map +0 -1
- package/dist/templates/view_enums_buttonset.template.d.ts.map +0 -1
- package/dist/templates/view_enums_buttonset.template.js +0 -2
- package/dist/templates/view_enums_buttonset.template.js.map +0 -1
- package/dist/templates/view_enums_dropdown.template.d.ts.map +0 -1
- package/dist/templates/view_enums_dropdown.template.js +0 -2
- package/dist/templates/view_enums_dropdown.template.js.map +0 -1
- package/dist/templates/view_enums_select.template.d.ts.map +0 -1
- package/dist/templates/view_enums_select.template.js +0 -2
- package/dist/templates/view_enums_select.template.js.map +0 -1
- package/dist/templates/view_form.template.d.ts.map +0 -1
- package/dist/templates/view_form.template.js +0 -2
- package/dist/templates/view_form.template.js.map +0 -1
- package/dist/templates/view_id_all_select.template.d.ts.map +0 -1
- package/dist/templates/view_id_all_select.template.js +0 -2
- package/dist/templates/view_id_all_select.template.js.map +0 -1
- package/dist/templates/view_id_async_select.template.d.ts.map +0 -1
- package/dist/templates/view_id_async_select.template.js +0 -2
- package/dist/templates/view_id_async_select.template.js.map +0 -1
- package/dist/templates/view_list.template.d.ts.map +0 -1
- package/dist/templates/view_list.template.js +0 -2
- package/dist/templates/view_list.template.js.map +0 -1
- package/dist/templates/view_list_columns.template.d.ts.map +0 -1
- package/dist/templates/view_list_columns.template.js +0 -2
- package/dist/templates/view_list_columns.template.js.map +0 -1
- package/dist/templates/view_search_input.template.d.ts.map +0 -1
- package/dist/templates/view_search_input.template.js +0 -2
- package/dist/templates/view_search_input.template.js.map +0 -1
- package/dist/testing/_relation-graph.js.map +0 -1
- package/dist/testing/fixture-manager.js.map +0 -1
- package/dist/types/types.js.map +0 -1
- package/dist/typings/knex.d.js.map +0 -1
- package/dist/utils/async-utils.js.map +0 -1
- package/dist/utils/controller.js.map +0 -1
- package/dist/utils/fs-utils.js.map +0 -1
- package/dist/utils/lodash-able.js.map +0 -1
- package/dist/utils/model.js.map +0 -1
- package/dist/utils/sql-parser.js.map +0 -1
- package/dist/utils/utils.js.map +0 -1
- package/dist/utils/zod-error.js.map +0 -1
- package/src/templates/base-template.ts +0 -19
- package/src/templates/index.ts +0 -1
package/dist/entity/entity.js
CHANGED
|
@@ -1,2 +1,651 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,"__esModule",{value:true});Object.defineProperty(exports,"Entity",{enumerable:true,get:function(){return Entity}});var _lodash=/*#__PURE__*/_interop_require_default(require("lodash"));var _entitymanager=require("./entity-manager");var _types=require("../types/types");var _inflection=/*#__PURE__*/_interop_require_default(require("inflection"));var _path=/*#__PURE__*/_interop_require_default(require("path"));var _promises=require("fs/promises");var _zod=require("zod");var _sonamu=require("../api/sonamu");var _prettier=/*#__PURE__*/_interop_require_default(require("prettier"));var _utils=require("../utils/utils");var _fsutils=require("../utils/fs-utils");function _array_like_to_array(arr,len){if(len==null||len>arr.length)len=arr.length;for(var i=0,arr2=new Array(len);i<len;i++)arr2[i]=arr[i];return arr2}function _array_with_holes(arr){if(Array.isArray(arr))return arr}function _array_without_holes(arr){if(Array.isArray(arr))return _array_like_to_array(arr)}function asyncGeneratorStep(gen,resolve,reject,_next,_throw,key,arg){try{var info=gen[key](arg);var value=info.value}catch(error){reject(error);return}if(info.done){resolve(value)}else{Promise.resolve(value).then(_next,_throw)}}function _async_to_generator(fn){return function(){var self=this,args=arguments;return new Promise(function(resolve,reject){var gen=fn.apply(self,args);function _next(value){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"next",value)}function _throw(err){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"throw",err)}_next(undefined)})}}function _class_call_check(instance,Constructor){if(!(instance instanceof Constructor)){throw new TypeError("Cannot call a class as a function")}}function _defineProperties(target,props){for(var i=0;i<props.length;i++){var descriptor=props[i];descriptor.enumerable=descriptor.enumerable||false;descriptor.configurable=true;if("value"in descriptor)descriptor.writable=true;Object.defineProperty(target,descriptor.key,descriptor)}}function _create_class(Constructor,protoProps,staticProps){if(protoProps)_defineProperties(Constructor.prototype,protoProps);if(staticProps)_defineProperties(Constructor,staticProps);return Constructor}function _define_property(obj,key,value){if(key in obj){Object.defineProperty(obj,key,{value:value,enumerable:true,configurable:true,writable:true})}else{obj[key]=value}return obj}function _interop_require_default(obj){return obj&&obj.__esModule?obj:{default:obj}}function _getRequireWildcardCache(nodeInterop){if(typeof WeakMap!=="function")return null;var cacheBabelInterop=new WeakMap;var cacheNodeInterop=new WeakMap;return(_getRequireWildcardCache=function(nodeInterop){return nodeInterop?cacheNodeInterop:cacheBabelInterop})(nodeInterop)}function _interop_require_wildcard(obj,nodeInterop){if(!nodeInterop&&obj&&obj.__esModule){return obj}if(obj===null||typeof obj!=="object"&&typeof obj!=="function"){return{default:obj}}var cache=_getRequireWildcardCache(nodeInterop);if(cache&&cache.has(obj)){return cache.get(obj)}var newObj={__proto__:null};var hasPropertyDescriptor=Object.defineProperty&&Object.getOwnPropertyDescriptor;for(var key in obj){if(key!=="default"&&Object.prototype.hasOwnProperty.call(obj,key)){var desc=hasPropertyDescriptor?Object.getOwnPropertyDescriptor(obj,key):null;if(desc&&(desc.get||desc.set)){Object.defineProperty(newObj,key,desc)}else{newObj[key]=obj[key]}}}newObj.default=obj;if(cache){cache.set(obj,newObj)}return newObj}function _iterable_to_array(iter){if(typeof Symbol!=="undefined"&&iter[Symbol.iterator]!=null||iter["@@iterator"]!=null)return Array.from(iter)}function _iterable_to_array_limit(arr,i){var _i=arr==null?null:typeof Symbol!=="undefined"&&arr[Symbol.iterator]||arr["@@iterator"];if(_i==null)return;var _arr=[];var _n=true;var _d=false;var _s,_e;try{for(_i=_i.call(arr);!(_n=(_s=_i.next()).done);_n=true){_arr.push(_s.value);if(i&&_arr.length===i)break}}catch(err){_d=true;_e=err}finally{try{if(!_n&&_i["return"]!=null)_i["return"]()}finally{if(_d)throw _e}}return _arr}function _non_iterable_rest(){throw new TypeError("Invalid attempt to destructure non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function _non_iterable_spread(){throw new TypeError("Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function _object_spread(target){for(var i=1;i<arguments.length;i++){var source=arguments[i]!=null?arguments[i]:{};var ownKeys=Object.keys(source);if(typeof Object.getOwnPropertySymbols==="function"){ownKeys=ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function(sym){return Object.getOwnPropertyDescriptor(source,sym).enumerable}))}ownKeys.forEach(function(key){_define_property(target,key,source[key])})}return target}function ownKeys(object,enumerableOnly){var keys=Object.keys(object);if(Object.getOwnPropertySymbols){var symbols=Object.getOwnPropertySymbols(object);if(enumerableOnly){symbols=symbols.filter(function(sym){return Object.getOwnPropertyDescriptor(object,sym).enumerable})}keys.push.apply(keys,symbols)}return keys}function _object_spread_props(target,source){source=source!=null?source:{};if(Object.getOwnPropertyDescriptors){Object.defineProperties(target,Object.getOwnPropertyDescriptors(source))}else{ownKeys(Object(source)).forEach(function(key){Object.defineProperty(target,key,Object.getOwnPropertyDescriptor(source,key))})}return target}function _sliced_to_array(arr,i){return _array_with_holes(arr)||_iterable_to_array_limit(arr,i)||_unsupported_iterable_to_array(arr,i)||_non_iterable_rest()}function _to_array(arr){return _array_with_holes(arr)||_iterable_to_array(arr)||_unsupported_iterable_to_array(arr)||_non_iterable_rest()}function _to_consumable_array(arr){return _array_without_holes(arr)||_iterable_to_array(arr)||_unsupported_iterable_to_array(arr)||_non_iterable_spread()}function _unsupported_iterable_to_array(o,minLen){if(!o)return;if(typeof o==="string")return _array_like_to_array(o,minLen);var n=Object.prototype.toString.call(o).slice(8,-1);if(n==="Object"&&o.constructor)n=o.constructor.name;if(n==="Map"||n==="Set")return Array.from(n);if(n==="Arguments"||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))return _array_like_to_array(o,minLen)}function _ts_generator(thisArg,body){var f,y,t,_={label:0,sent:function(){if(t[0]&1)throw t[1];return t[1]},trys:[],ops:[]},g=Object.create((typeof Iterator==="function"?Iterator:Object).prototype);return g.next=verb(0),g["throw"]=verb(1),g["return"]=verb(2),typeof Symbol==="function"&&(g[Symbol.iterator]=function(){return this}),g;function verb(n){return function(v){return step([n,v])}}function step(op){if(f)throw new TypeError("Generator is already executing.");while(g&&(g=0,op[0]&&(_=0)),_)try{if(f=1,y&&(t=op[0]&2?y["return"]:op[0]?y["throw"]||((t=y["return"])&&t.call(y),0):y.next)&&!(t=t.call(y,op[1])).done)return t;if(y=0,t)op=[op[0]&2,t.value];switch(op[0]){case 0:case 1:t=op;break;case 4:_.label++;return{value:op[1],done:false};case 5:_.label++;y=op[1];op=[0];continue;case 7:op=_.ops.pop();_.trys.pop();continue;default:if(!(t=_.trys,t=t.length>0&&t[t.length-1])&&(op[0]===6||op[0]===2)){_=0;continue}if(op[0]===3&&(!t||op[1]>t[0]&&op[1]<t[3])){_.label=op[1];break}if(op[0]===6&&_.label<t[1]){_.label=t[1];t=op;break}if(t&&_.label<t[2]){_.label=t[2];_.ops.push(op);break}if(t[2])_.ops.pop();_.trys.pop();continue}op=body.call(thisArg,_)}catch(e){op=[6,e];y=0}finally{f=t=0}if(op[0]&5)throw op[1];return{value:op[0]?op[1]:void 0,done:true}}}var Entity=/*#__PURE__*/function(){"use strict";function Entity(param){var id=param.id,parentId=param.parentId,table=param.table,title=param.title,props=param.props,indexes=param.indexes,subsets=param.subsets,enums=param.enums;_class_call_check(this,Entity);_define_property(this,"id",void 0);_define_property(this,"parentId",void 0);_define_property(this,"table",void 0);_define_property(this,"title",void 0);_define_property(this,"names",void 0);_define_property(this,"props",void 0);_define_property(this,"propsDict",void 0);_define_property(this,"relations",void 0);_define_property(this,"indexes",void 0);_define_property(this,"subsets",void 0);_define_property(this,"types",{});_define_property(this,"enums",{});_define_property(this,"enumLabels",{});this.id=id;this.parentId=parentId;this.title=title!==null&&title!==void 0?title:this.id;this.table=table!==null&&table!==void 0?table:_inflection.default.underscore(_inflection.default.pluralize(id));if(props){this.props=props.map(function(prop){if((0,_types.isEnumProp)(prop)){if(prop.id.includes("$Model")){prop.id=prop.id.replace("$Model",id)}}return prop});this.propsDict=props.reduce(function(result,prop){return _object_spread_props(_object_spread({},result),_define_property({},prop.name,prop))},{});this.relations=props.filter(function(prop){return(0,_types.isRelationProp)(prop)}).reduce(function(result,prop){return _object_spread_props(_object_spread({},result),_define_property({},prop.name,prop))},{})}else{this.props=[];this.propsDict={};this.relations={}}this.indexes=indexes!==null&&indexes!==void 0?indexes:[];this.subsets=subsets!==null&&subsets!==void 0?subsets:{};this.enumLabels=enums!==null&&enums!==void 0?enums:{};this.enums=Object.fromEntries(Object.entries(this.enumLabels).map(function(param){var _param=_sliced_to_array(param,2),key=_param[0],enumLabel=_param[1];return[key,_zod.z.enum(Object.keys(enumLabel))]}));this.names={parentFs:_inflection.default.dasherize(_inflection.default.underscore(parentId!==null&&parentId!==void 0?parentId:id)).toLowerCase(),fs:_inflection.default.dasherize(_inflection.default.underscore(id)).toLowerCase(),module:id}}_create_class(Entity,[{key:"getSubsetQuery",value:function getSubsetQuery(subsetKey){var subset=this.subsets[subsetKey];var result=this.resolveSubsetQuery("",subset);return result}},{key:"resolveSubsetQuery",value:function resolveSubsetQuery(prefix,fields){var _this=this;var isAlreadyOuterJoined=arguments.length>2&&arguments[2]!==void 0?arguments[2]:false;prefix=prefix.replace(/\./g,"__");var subsetGroup=_lodash.default.groupBy(fields,function(field){if(field.includes(".")){var _field_split=_sliced_to_array(field.split("."),1),rel=_field_split[0];return rel}else{return""}});var result=Object.keys(subsetGroup).reduce(function(r,groupKey){var _$fields=subsetGroup[groupKey];if(groupKey===""){var realFields=_$fields.filter(function(field){return!(0,_types.isVirtualProp)(_this.propsDict[field])});var virtualFields=_$fields.filter(function(field){return(0,_types.isVirtualProp)(_this.propsDict[field])});if(prefix===""){r.select=r.select.concat(realFields.map(function(field){return"".concat(_this.table,".").concat(field)}));r.virtual=r.virtual.concat(virtualFields)}else{r.select=r.select.concat(realFields.map(function(field){return"".concat(prefix,".").concat(field," as ").concat(prefix,"__").concat(field)}))}return r}var relation=_this.relations[groupKey];if(relation===undefined){throw new Error("존재하지 않는 relation 참조 ".concat(groupKey))}var relEntity=_entitymanager.EntityManager.get(relation.with);if((0,_types.isOneToOneRelationProp)(relation)||(0,_types.isBelongsToOneRelationProp)(relation)){var relFields=_$fields.map(function(field){return field.split(".").slice(1).join(".")});if(relFields.length===1&&relFields[0]==="id"){if(prefix===""){r.select=r.select.concat("".concat(_this.table,".").concat(groupKey,"_id"))}else{r.select=r.select.concat("".concat(prefix,".").concat(groupKey,"_id as ").concat(prefix,"__").concat(groupKey,"_id"))}return r}var innerOrOuter=function(){if(isAlreadyOuterJoined){return"outer"}if((0,_types.isOneToOneRelationProp)(relation)){var _relation_nullable;if(relation.hasJoinColumn===true&&((_relation_nullable=relation.nullable)!==null&&_relation_nullable!==void 0?_relation_nullable:false)===false){return"inner"}else{return"outer"}}else{if(relation.nullable){return"outer"}else{return"inner"}}}();var relSubsetQuery=relEntity.resolveSubsetQuery("".concat(prefix!==""?prefix+".":"").concat(groupKey),relFields,innerOrOuter==="outer");r.select=r.select.concat(relSubsetQuery.select);r.virtual=r.virtual.concat(relSubsetQuery.virtual);var joinAs=prefix===""?groupKey:prefix+"__"+groupKey;var fromTable=prefix===""?_this.table:prefix;var joinClause;if(relation.customJoinClause){joinClause={custom:relation.customJoinClause}}else{var from,to;if((0,_types.isOneToOneRelationProp)(relation)){if(relation.hasJoinColumn){from="".concat(fromTable,".").concat(relation.name,"_id");to="".concat(joinAs,".id")}else{from="".concat(fromTable,".id");to="".concat(joinAs,".").concat(_inflection.default.underscore(_this.names.fs.replace(/\-/g,"_")),"_id")}}else{from="".concat(fromTable,".").concat(relation.name,"_id");to="".concat(joinAs,".id")}joinClause={from:from,to:to}}r.joins.push(_object_spread({as:joinAs,join:innerOrOuter,table:relEntity.table},joinClause));if(relSubsetQuery.loaders.length>0){var convertedLoaders=relSubsetQuery.loaders.map(function(loader){var newAs=[groupKey,loader.as].join("__");return{as:newAs,table:loader.table,manyJoin:loader.manyJoin,oneJoins:loader.oneJoins,select:loader.select,loaders:loader.loaders}});r.loaders=_to_consumable_array(r.loaders).concat(_to_consumable_array(convertedLoaders))}r.joins=r.joins.concat(relSubsetQuery.joins)}else if((0,_types.isHasManyRelationProp)(relation)||(0,_types.isManyToManyRelationProp)(relation)){var relFields1=_$fields.map(function(field){return field.split(".").slice(1).join(".")});var relSubsetQuery1=relEntity.resolveSubsetQuery("",relFields1);var manyJoin;if((0,_types.isHasManyRelationProp)(relation)){var _relation_fromColumn;var fromCol=(_relation_fromColumn=relation===null||relation===void 0?void 0:relation.fromColumn)!==null&&_relation_fromColumn!==void 0?_relation_fromColumn:"id";manyJoin={fromTable:_this.table,fromCol:fromCol,idField:prefix===""?"".concat(fromCol):"".concat(prefix,"__").concat(fromCol),toTable:relEntity.table,toCol:relation.joinColumn}}else if((0,_types.isManyToManyRelationProp)(relation)){manyJoin={fromTable:_this.table,fromCol:"id",idField:prefix===""?"id":"".concat(prefix,"__id"),through:{table:relation.joinTable,fromCol:"".concat(_inflection.default.singularize(_this.table),"_id"),toCol:"".concat(_inflection.default.singularize(relEntity.table),"_id")},toTable:relEntity.table,toCol:"id"}}else{throw new Error}r.loaders.push({as:groupKey,table:relEntity.table,manyJoin:manyJoin,oneJoins:relSubsetQuery1.joins,select:relSubsetQuery1.select,loaders:relSubsetQuery1.loaders})}return r},{select:[],virtual:[],joins:[],loaders:[]});return result}},{key:"fieldExprsToPropNodes",value:function fieldExprsToPropNodes(fieldExprs){var _this=this;var entity=arguments.length>1&&arguments[1]!==void 0?arguments[1]:this;var groups=fieldExprs.reduce(function(result,fieldExpr){var key,value,elseExpr;if(fieldExpr.includes(".")){var ref;ref=_to_array(fieldExpr.split(".")),key=ref[0],elseExpr=ref.slice(1),ref;value=elseExpr.join(".")}else{key="";value=fieldExpr}var _result_key;result[key]=((_result_key=result[key])!==null&&_result_key!==void 0?_result_key:[]).concat(value);return result},{});return Object.keys(groups).map(function(key){var group=groups[key];if(key===""){return group.map(function(propName){if(propName==="uuid"){return{nodeType:"plain",prop:{type:"string",name:"uuid",length:128},children:[]}}var prop=entity.props.find(function(p){return p.name===propName});if(prop===undefined){console.log({propName:propName,groups:groups});throw new Error("".concat(entity.id," -- 잘못된 FieldExpr ").concat(propName))}return{nodeType:"plain",prop:prop,children:[]}})}var prop=entity.propsDict[key];if(!(0,_types.isRelationProp)(prop)){throw new Error("잘못된 FieldExpr ".concat(key,".").concat(group[0]))}var relEntity=_entitymanager.EntityManager.get(prop.with);if((0,_types.isBelongsToOneRelationProp)(prop)||(0,_types.isOneToOneRelationProp)(prop)){if(group.length==1&&(group[0]==="id"||group[0]=="id?")){var idProp=relEntity.propsDict.id;return{nodeType:"plain",prop:_object_spread_props(_object_spread({},idProp),{name:key+"_id",nullable:prop.nullable}),children:[]}}}var children=_this.fieldExprsToPropNodes(group,relEntity);var nodeType=(0,_types.isBelongsToOneRelationProp)(prop)||(0,_types.isOneToOneRelationProp)(prop)?"object":"array";return{prop:prop,children:children,nodeType:nodeType}}).flat()}},{key:"getFieldExprs",value:function getFieldExprs(){var _this=this;var prefix=arguments.length>0&&arguments[0]!==void 0?arguments[0]:"",maxDepth=arguments.length>1&&arguments[1]!==void 0?arguments[1]:3,froms=arguments.length>2&&arguments[2]!==void 0?arguments[2]:[];return this.props.map(function(prop){var propName=[prefix,prop.name].filter(function(v){return v!==""}).join(".");if(propName===prefix){return null}if((0,_types.isRelationProp)(prop)){if(maxDepth<0){return null}if(froms.includes(prop.with)){return null}var relMd=_entitymanager.EntityManager.get(prop.with);return relMd.getFieldExprs(propName,maxDepth-1,_to_consumable_array(froms).concat([_this.id]))}return propName}).flat().filter(function(f){return f!==null})}},{key:"getTableColumns",value:function getTableColumns(){return this.props.map(function(prop){if(prop.type==="relation"){if(prop.relationType==="BelongsToOne"||prop.relationType==="OneToOne"&&prop.hasJoinColumn===true){return{name:"".concat(prop.name,"_id"),type:"int_unsigned"}}else{return null}}return{name:prop.name,type:prop.type}}).filter(_utils.nonNullable)}},{key:"registerModulePaths",value:function registerModulePaths(){return _async_to_generator(function(){var _this,basePath,typesModulePath,typesFileDistPath,importPath,t;return _ts_generator(this,function(_state){switch(_state.label){case 0:_this=this;basePath="".concat(this.names.parentFs);_entitymanager.EntityManager.setModulePath("".concat(this.id,"BaseSchema"),"sonamu.generated");if(Object.keys(this.subsets).length>0){_entitymanager.EntityManager.setModulePath("".concat(this.id,"SubsetKey"),"sonamu.generated");_entitymanager.EntityManager.setModulePath("".concat(this.id,"SubsetMapping"),"sonamu.generated");Object.keys(this.subsets).map(function(subsetKey){_entitymanager.EntityManager.setModulePath("".concat(_this.id,"Subset").concat(subsetKey.toUpperCase()),"sonamu.generated")})}Object.keys(this.enumLabels).map(function(enumId){_entitymanager.EntityManager.setModulePath(enumId,"sonamu.generated")});typesModulePath="".concat(basePath,"/").concat(this.names.parentFs,".types");typesFileDistPath=_path.default.join(_sonamu.Sonamu.apiRootPath,"dist/application/".concat(typesModulePath,".js"));return[4,(0,_fsutils.exists)(typesFileDistPath)];case 1:if(!_state.sent())return[3,3];importPath=_path.default.relative(__dirname,typesFileDistPath);return[4,Promise.resolve(importPath).then(function(p){return /*#__PURE__*/_interop_require_wildcard(require(p))})];case 2:t=_state.sent();this.types=Object.keys(t).reduce(function(result,key){_entitymanager.EntityManager.setModulePath(key,typesModulePath);return _object_spread_props(_object_spread({},result),_define_property({},key,t[key]))},{});_state.label=3;case 3:return[2]}})}).call(this)}},{key:"registerTableSpecs",value:function registerTableSpecs(){var uniqueIndexes=this.indexes.filter(function(idx){return idx.type==="unique"});_entitymanager.EntityManager.setTableSpec({name:this.table,uniqueIndexes:uniqueIndexes})}},{key:"toJson",value:function toJson(){return{id:this.id,parentId:this.parentId,table:this.table,title:this.title,props:this.props,indexes:this.indexes,subsets:this.subsets,enums:this.enumLabels}}},{key:"save",value:function save(){return _async_to_generator(function(){var _this,subsetRows,jsonPath,json,_tmp;return _ts_generator(this,function(_state){switch(_state.label){case 0:_this=this;subsetRows=this.getSubsetRows();this.subsets=Object.fromEntries(Object.entries(this.subsets).map(function(param){var _param=_sliced_to_array(param,1),subsetKey=_param[0];return[subsetKey,_this.subsetRowsToSubsetFields(subsetRows,subsetKey)]}));jsonPath=_path.default.join(_sonamu.Sonamu.apiRootPath,"src/application/".concat(this.names.parentFs,"/").concat(this.names.fs,".entity.json"));json=this.toJson();_tmp=[jsonPath];return[4,_prettier.default.format(JSON.stringify(json),{parser:"json"})];case 1:return[4,_promises.writeFile.apply(void 0,_tmp.concat([_state.sent()]))];case 2:_state.sent();return[4,_entitymanager.EntityManager.register(json)];case 3:_state.sent();return[2]}})}).call(this)}},{key:"getSubsetRows",value:function getSubsetRows(_subsets){var prefixes=arguments.length>1&&arguments[1]!==void 0?arguments[1]:[];if(prefixes.length>10){return[]}var subsets=_subsets!==null&&_subsets!==void 0?_subsets:this.subsets;var subsetKeys=Object.keys(subsets);var allFields=_lodash.default.uniq(subsetKeys.map(function(key){return subsets[key]}).flat());return this.props.map(function(prop){if(prop.type==="relation"&&allFields.find(function(f){return f.startsWith(_to_consumable_array(prefixes).concat([prop.name]).join(".")+".")})){var relEntity=_entitymanager.EntityManager.get(prop.with);var children=relEntity.getSubsetRows(subsets,_to_consumable_array(prefixes).concat(["".concat(prop.name)]));return{field:prop.name,children:children,relationEntity:prop.with,prefixes:prefixes,isOpen:children.length>0,has:Object.fromEntries(subsetKeys.map(function(subsetKey){return[subsetKey,children.every(function(child){return child.has[subsetKey]===true})]}))}}return{field:prop.name,children:[],relationEntity:prop.type==="relation"?prop.with:undefined,prefixes:prefixes,has:Object.fromEntries(subsetKeys.map(function(subsetKey){var subsetFields=subsets[subsetKey];var has=subsetFields.some(function(f){var field=_to_consumable_array(prefixes).concat([prop.name]).join(".");return f===field||f.startsWith(field+".")});return[subsetKey,has]}))}})}},{key:"subsetRowsToSubsetFields",value:function subsetRowsToSubsetFields(subsetRows,subsetKey){var _this=this;return subsetRows.map(function(subsetRow){if(subsetRow.children.length>0){return _this.subsetRowsToSubsetFields(subsetRow.children,subsetKey)}else if(subsetRow.has[subsetKey]){return subsetRow.prefixes.concat(subsetRow.field).join(".")}else{return null}}).filter(_utils.nonNullable).flat()}},{key:"createProp",value:function createProp(prop,at){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:if(!at){this.props.push(prop)}else{this.props.splice(at,0,prop)}return[4,this.save()];case 1:_state.sent();return[2]}})}).call(this)}},{key:"analyzeSubsetField",value:function analyzeSubsetField(subsetField){var _loop=function(i){var propName=arr[i];result.push({entityId:entityId,propName:propName});var prop=_entitymanager.EntityManager.get(entityId).props.find(function(p){return p.name===propName});if(!prop){throw new Error("".concat(entityId,"의 잘못된 서브셋키 ").concat(subsetField))}if((0,_types.isRelationProp)(prop)){entityId=prop.with}};var arr=subsetField.split(".");var entityId=this.id;var result=[];for(var i=0;i<arr.length;i++)_loop(i);return result}},{key:"modifyProp",value:function modifyProp(newProp,at){return _async_to_generator(function(){var _this,oldName,entities,allEntityIds,_iteratorNormalCompletion,_didIteratorError,_iteratorError,_loop,_iterator,_step;return _ts_generator(this,function(_state){switch(_state.label){case 0:_this=this;oldName=this.props[at].name;entities=[this];if(oldName!==newProp.name){allEntityIds=_entitymanager.EntityManager.getAllIds();_iteratorNormalCompletion=true,_didIteratorError=false,_iteratorError=undefined;try{_loop=function(){var relEntityId=_step.value;var relEntity=_entitymanager.EntityManager.get(relEntityId);var relEntitySubsetKeys=Object.keys(relEntity.subsets);var _iteratorNormalCompletion=true,_didIteratorError=false,_iteratorError=undefined;try{for(var _iterator=relEntitySubsetKeys[Symbol.iterator](),_step1;!(_iteratorNormalCompletion=(_step1=_iterator.next()).done);_iteratorNormalCompletion=true){var subsetKey=_step1.value;var subset=relEntity.subsets[subsetKey];var modifiedSubsetFields=subset.map(function(subsetField){var analyzed=relEntity.analyzeSubsetField(subsetField);var modified=analyzed.map(function(a){return a.propName===oldName&&a.entityId===_this.id?_object_spread_props(_object_spread({},a),{propName:newProp.name}):a});return modified.map(function(a){return a.propName}).join(".")});if(subset.join(",")!==modifiedSubsetFields.join(",")){relEntity.subsets[subsetKey]=modifiedSubsetFields;entities.push(relEntity)}}}catch(err){_didIteratorError=true;_iteratorError=err}finally{try{if(!_iteratorNormalCompletion&&_iterator.return!=null){_iterator.return()}}finally{if(_didIteratorError){throw _iteratorError}}}};for(_iterator=allEntityIds[Symbol.iterator]();!(_iteratorNormalCompletion=(_step=_iterator.next()).done);_iteratorNormalCompletion=true)_loop()}catch(err){_didIteratorError=true;_iteratorError=err}finally{try{if(!_iteratorNormalCompletion&&_iterator.return!=null){_iterator.return()}}finally{if(_didIteratorError){throw _iteratorError}}}}this.props[at]=newProp;return[4,Promise.all(entities.map(function(entity){return _async_to_generator(function(){return _ts_generator(this,function(_state){return[2,entity.save()]})})()}))];case 1:_state.sent();return[2]}})}).call(this)}},{key:"delProp",value:function delProp(at){return _async_to_generator(function(){var _this,oldName,entities,allEntityIds,_iteratorNormalCompletion,_didIteratorError,_iteratorError,_loop,_iterator,_step;return _ts_generator(this,function(_state){switch(_state.label){case 0:_this=this;oldName=this.props[at].name;entities=[this];allEntityIds=_entitymanager.EntityManager.getAllIds();_iteratorNormalCompletion=true,_didIteratorError=false,_iteratorError=undefined;try{_loop=function(){var relEntityId=_step.value;var relEntity=_entitymanager.EntityManager.get(relEntityId);var relEntitySubsetKeys=Object.keys(relEntity.subsets);var _iteratorNormalCompletion=true,_didIteratorError=false,_iteratorError=undefined;try{for(var _iterator=relEntitySubsetKeys[Symbol.iterator](),_step1;!(_iteratorNormalCompletion=(_step1=_iterator.next()).done);_iteratorNormalCompletion=true){var subsetKey=_step1.value;var subset=relEntity.subsets[subsetKey];var modifiedSubsetFields=subset.map(function(subsetField){var analyzed=relEntity.analyzeSubsetField(subsetField);if(analyzed.find(function(a){return a.propName===oldName&&a.entityId===_this.id})){return null}else{return subsetField}}).filter(_utils.nonNullable);if(subset.join(",")!==modifiedSubsetFields.join(",")){relEntity.subsets[subsetKey]=modifiedSubsetFields;entities.push(relEntity)}}}catch(err){_didIteratorError=true;_iteratorError=err}finally{try{if(!_iteratorNormalCompletion&&_iterator.return!=null){_iterator.return()}}finally{if(_didIteratorError){throw _iteratorError}}}};for(_iterator=allEntityIds[Symbol.iterator]();!(_iteratorNormalCompletion=(_step=_iterator.next()).done);_iteratorNormalCompletion=true)_loop()}catch(err){_didIteratorError=true;_iteratorError=err}finally{try{if(!_iteratorNormalCompletion&&_iterator.return!=null){_iterator.return()}}finally{if(_didIteratorError){throw _iteratorError}}}_entitymanager.EntityManager.get(this.id).indexes.map(function(index){index.columns=index.columns.filter(function(col){return col!==oldName})});this.props.splice(at,1);return[4,Promise.all(entities.map(function(entity){return _async_to_generator(function(){return _ts_generator(this,function(_state){return[2,entity.save()]})})()}))];case 1:_state.sent();return[2]}})}).call(this)}},{key:"getEntityIdFromSubsetField",value:function getEntityIdFromSubsetField(subsetField){var _this=this;if(subsetField.includes(".")===false){return this.id}var arr=subsetField.split(".").slice(0,-1);var lastEntityId=arr.reduce(function(entityId,field){var relProp=_entitymanager.EntityManager.get(entityId).props.find(function(p){return p.name===field});if(!relProp||relProp.type!=="relation"){console.debug({arr:arr,thisId:_this.id,entityId:entityId,field:field});throw new Error("잘못된 서브셋키 ".concat(subsetField))}return relProp.with},this.id);return lastEntityId}},{key:"moveProp",value:function moveProp(at,to){return _async_to_generator(function(){var prop,newProps;return _ts_generator(this,function(_state){switch(_state.label){case 0:prop=this.props[at];newProps=_to_consumable_array(this.props);newProps.splice(to,0,prop);newProps.splice(at<to?at:at+1,1);this.props=newProps;return[4,this.save()];case 1:_state.sent();return[2]}})}).call(this)}}]);return Entity}();
|
|
2
|
-
|
|
1
|
+
import * as _ from "lodash-es";
|
|
2
|
+
import { createImportUrl } from "../utils/esm-utils.js";
|
|
3
|
+
import { EntityManager as EntityManager } from "./entity-manager.js";
|
|
4
|
+
import { isRelationProp, isVirtualProp, isBelongsToOneRelationProp, isOneToOneRelationProp, isHasManyRelationProp, isManyToManyRelationProp, isEnumProp } from "../types/types.js";
|
|
5
|
+
import inflection from "inflection";
|
|
6
|
+
import path from "path";
|
|
7
|
+
import { writeFile } from "node:fs/promises";
|
|
8
|
+
import { z } from "zod";
|
|
9
|
+
import { Sonamu } from "../api/sonamu.js";
|
|
10
|
+
import prettier from "prettier";
|
|
11
|
+
import { nonNullable } from "../utils/utils.js";
|
|
12
|
+
import { exists } from "../utils/fs-utils.js";
|
|
13
|
+
export class Entity {
|
|
14
|
+
id;
|
|
15
|
+
parentId;
|
|
16
|
+
table;
|
|
17
|
+
title;
|
|
18
|
+
names;
|
|
19
|
+
props;
|
|
20
|
+
propsDict;
|
|
21
|
+
relations;
|
|
22
|
+
indexes;
|
|
23
|
+
subsets;
|
|
24
|
+
types = {};
|
|
25
|
+
enums = {};
|
|
26
|
+
enumLabels = {};
|
|
27
|
+
constructor({ id, parentId, table, title, props, indexes, subsets, enums }){
|
|
28
|
+
// id
|
|
29
|
+
this.id = id;
|
|
30
|
+
this.parentId = parentId;
|
|
31
|
+
this.title = title ?? this.id;
|
|
32
|
+
this.table = table ?? inflection.underscore(inflection.pluralize(id));
|
|
33
|
+
// props
|
|
34
|
+
if (props) {
|
|
35
|
+
this.props = props.map((prop)=>{
|
|
36
|
+
if (isEnumProp(prop)) {
|
|
37
|
+
if (prop.id.includes("$Model")) {
|
|
38
|
+
prop.id = prop.id.replace("$Model", id);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
return prop;
|
|
42
|
+
});
|
|
43
|
+
this.propsDict = props.reduce((result, prop)=>{
|
|
44
|
+
return {
|
|
45
|
+
...result,
|
|
46
|
+
[prop.name]: prop
|
|
47
|
+
};
|
|
48
|
+
}, {});
|
|
49
|
+
// relations
|
|
50
|
+
this.relations = props.filter((prop)=>isRelationProp(prop)).reduce((result, prop)=>{
|
|
51
|
+
return {
|
|
52
|
+
...result,
|
|
53
|
+
[prop.name]: prop
|
|
54
|
+
};
|
|
55
|
+
}, {});
|
|
56
|
+
} else {
|
|
57
|
+
this.props = [];
|
|
58
|
+
this.propsDict = {};
|
|
59
|
+
this.relations = {};
|
|
60
|
+
}
|
|
61
|
+
// indexes
|
|
62
|
+
this.indexes = indexes ?? [];
|
|
63
|
+
// subsets
|
|
64
|
+
this.subsets = subsets ?? {};
|
|
65
|
+
// enums
|
|
66
|
+
this.enumLabels = enums ?? {};
|
|
67
|
+
this.enums = Object.fromEntries(Object.entries(this.enumLabels).map(([key, enumLabel])=>{
|
|
68
|
+
return [
|
|
69
|
+
key,
|
|
70
|
+
z.enum(Object.keys(enumLabel))
|
|
71
|
+
];
|
|
72
|
+
}));
|
|
73
|
+
// names
|
|
74
|
+
this.names = {
|
|
75
|
+
parentFs: inflection.dasherize(inflection.underscore(parentId ?? id)).toLowerCase(),
|
|
76
|
+
fs: inflection.dasherize(inflection.underscore(id)).toLowerCase(),
|
|
77
|
+
module: id
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
/*
|
|
81
|
+
subset SELECT/JOIN/LOADER 결과 리턴
|
|
82
|
+
*/ getSubsetQuery(subsetKey) {
|
|
83
|
+
const subset = this.subsets[subsetKey];
|
|
84
|
+
const result = this.resolveSubsetQuery("", subset);
|
|
85
|
+
return result;
|
|
86
|
+
}
|
|
87
|
+
/*
|
|
88
|
+
*/ resolveSubsetQuery(prefix, fields, isAlreadyOuterJoined = false) {
|
|
89
|
+
// prefix 치환 (prefix는 ToOneRelation이 복수로 붙은 경우 모두 __로 변경됨)
|
|
90
|
+
prefix = prefix.replace(/\./g, "__");
|
|
91
|
+
// 서브셋을 1뎁스만 분리하여 그룹핑
|
|
92
|
+
const subsetGroup = _.groupBy(fields, (field)=>{
|
|
93
|
+
if (field.includes(".")) {
|
|
94
|
+
const [rel] = field.split(".");
|
|
95
|
+
return rel;
|
|
96
|
+
} else {
|
|
97
|
+
return "";
|
|
98
|
+
}
|
|
99
|
+
});
|
|
100
|
+
const result = Object.keys(subsetGroup).reduce((r, groupKey)=>{
|
|
101
|
+
const fields = subsetGroup[groupKey];
|
|
102
|
+
// 현재 테이블 필드셋은 select, virtual에 추가하고 리턴
|
|
103
|
+
if (groupKey === "") {
|
|
104
|
+
const realFields = fields.filter((field)=>!isVirtualProp(this.propsDict[field]));
|
|
105
|
+
const virtualFields = fields.filter((field)=>isVirtualProp(this.propsDict[field]));
|
|
106
|
+
if (prefix === "") {
|
|
107
|
+
// 현재 테이블인 경우
|
|
108
|
+
r.select = r.select.concat(realFields.map((field)=>`${this.table}.${field}`));
|
|
109
|
+
r.virtual = r.virtual.concat(virtualFields);
|
|
110
|
+
} else {
|
|
111
|
+
// 넘어온 테이블인 경우
|
|
112
|
+
r.select = r.select.concat(realFields.map((field)=>`${prefix}.${field} as ${prefix}__${field}`));
|
|
113
|
+
}
|
|
114
|
+
return r;
|
|
115
|
+
}
|
|
116
|
+
const relation = this.relations[groupKey];
|
|
117
|
+
if (relation === undefined) {
|
|
118
|
+
throw new Error(`존재하지 않는 relation 참조 ${groupKey}`);
|
|
119
|
+
}
|
|
120
|
+
const relEntity = EntityManager.get(relation.with);
|
|
121
|
+
if (isOneToOneRelationProp(relation) || isBelongsToOneRelationProp(relation)) {
|
|
122
|
+
// -One Relation: JOIN 으로 처리
|
|
123
|
+
const relFields = fields.map((field)=>field.split(".").slice(1).join("."));
|
|
124
|
+
// -One Relation에서 id 필드만 참조하는 경우 릴레이션 넘기지 않고 리턴
|
|
125
|
+
if (relFields.length === 1 && relFields[0] === "id") {
|
|
126
|
+
if (prefix === "") {
|
|
127
|
+
r.select = r.select.concat(`${this.table}.${groupKey}_id`);
|
|
128
|
+
} else {
|
|
129
|
+
r.select = r.select.concat(`${prefix}.${groupKey}_id as ${prefix}__${groupKey}_id`);
|
|
130
|
+
}
|
|
131
|
+
return r;
|
|
132
|
+
}
|
|
133
|
+
// innerOrOuter
|
|
134
|
+
const innerOrOuter = (()=>{
|
|
135
|
+
if (isAlreadyOuterJoined) {
|
|
136
|
+
return "outer";
|
|
137
|
+
}
|
|
138
|
+
if (isOneToOneRelationProp(relation)) {
|
|
139
|
+
if (relation.hasJoinColumn === true && (relation.nullable ?? false) === false) {
|
|
140
|
+
return "inner";
|
|
141
|
+
} else {
|
|
142
|
+
return "outer";
|
|
143
|
+
}
|
|
144
|
+
} else {
|
|
145
|
+
if (relation.nullable) {
|
|
146
|
+
return "outer";
|
|
147
|
+
} else {
|
|
148
|
+
return "inner";
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
})();
|
|
152
|
+
const relSubsetQuery = relEntity.resolveSubsetQuery(`${prefix !== "" ? prefix + "." : ""}${groupKey}`, relFields, innerOrOuter === "outer");
|
|
153
|
+
r.select = r.select.concat(relSubsetQuery.select);
|
|
154
|
+
r.virtual = r.virtual.concat(relSubsetQuery.virtual);
|
|
155
|
+
const joinAs = prefix === "" ? groupKey : prefix + "__" + groupKey;
|
|
156
|
+
const fromTable = prefix === "" ? this.table : prefix;
|
|
157
|
+
let joinClause;
|
|
158
|
+
if (relation.customJoinClause) {
|
|
159
|
+
joinClause = {
|
|
160
|
+
custom: relation.customJoinClause
|
|
161
|
+
};
|
|
162
|
+
} else {
|
|
163
|
+
let from, to;
|
|
164
|
+
if (isOneToOneRelationProp(relation)) {
|
|
165
|
+
if (relation.hasJoinColumn) {
|
|
166
|
+
from = `${fromTable}.${relation.name}_id`;
|
|
167
|
+
to = `${joinAs}.id`;
|
|
168
|
+
} else {
|
|
169
|
+
from = `${fromTable}.id`;
|
|
170
|
+
to = `${joinAs}.${inflection.underscore(this.names.fs.replace(/\-/g, "_"))}_id`;
|
|
171
|
+
}
|
|
172
|
+
} else {
|
|
173
|
+
from = `${fromTable}.${relation.name}_id`;
|
|
174
|
+
to = `${joinAs}.id`;
|
|
175
|
+
}
|
|
176
|
+
joinClause = {
|
|
177
|
+
from,
|
|
178
|
+
to
|
|
179
|
+
};
|
|
180
|
+
}
|
|
181
|
+
r.joins.push({
|
|
182
|
+
as: joinAs,
|
|
183
|
+
join: innerOrOuter,
|
|
184
|
+
table: relEntity.table,
|
|
185
|
+
...joinClause
|
|
186
|
+
});
|
|
187
|
+
// BelongsToOne 밑에 HasMany가 붙은 경우
|
|
188
|
+
if (relSubsetQuery.loaders.length > 0) {
|
|
189
|
+
const convertedLoaders = relSubsetQuery.loaders.map((loader)=>{
|
|
190
|
+
const newAs = [
|
|
191
|
+
groupKey,
|
|
192
|
+
loader.as
|
|
193
|
+
].join("__");
|
|
194
|
+
return {
|
|
195
|
+
as: newAs,
|
|
196
|
+
table: loader.table,
|
|
197
|
+
manyJoin: loader.manyJoin,
|
|
198
|
+
oneJoins: loader.oneJoins,
|
|
199
|
+
select: loader.select,
|
|
200
|
+
loaders: loader.loaders
|
|
201
|
+
};
|
|
202
|
+
});
|
|
203
|
+
r.loaders = [
|
|
204
|
+
...r.loaders,
|
|
205
|
+
...convertedLoaders
|
|
206
|
+
];
|
|
207
|
+
}
|
|
208
|
+
r.joins = r.joins.concat(relSubsetQuery.joins);
|
|
209
|
+
} else if (isHasManyRelationProp(relation) || isManyToManyRelationProp(relation)) {
|
|
210
|
+
// -Many Relation: Loader 로 처리
|
|
211
|
+
const relFields = fields.map((field)=>field.split(".").slice(1).join("."));
|
|
212
|
+
const relSubsetQuery = relEntity.resolveSubsetQuery("", relFields);
|
|
213
|
+
let manyJoin;
|
|
214
|
+
if (isHasManyRelationProp(relation)) {
|
|
215
|
+
const fromCol = relation?.fromColumn ?? "id";
|
|
216
|
+
manyJoin = {
|
|
217
|
+
fromTable: this.table,
|
|
218
|
+
fromCol,
|
|
219
|
+
idField: prefix === "" ? `${fromCol}` : `${prefix}__${fromCol}`,
|
|
220
|
+
toTable: relEntity.table,
|
|
221
|
+
toCol: relation.joinColumn
|
|
222
|
+
};
|
|
223
|
+
} else if (isManyToManyRelationProp(relation)) {
|
|
224
|
+
manyJoin = {
|
|
225
|
+
fromTable: this.table,
|
|
226
|
+
fromCol: "id",
|
|
227
|
+
idField: prefix === "" ? `id` : `${prefix}__id`,
|
|
228
|
+
through: {
|
|
229
|
+
table: relation.joinTable,
|
|
230
|
+
fromCol: `${inflection.singularize(this.table)}_id`,
|
|
231
|
+
toCol: `${inflection.singularize(relEntity.table)}_id`
|
|
232
|
+
},
|
|
233
|
+
toTable: relEntity.table,
|
|
234
|
+
toCol: "id"
|
|
235
|
+
};
|
|
236
|
+
} else {
|
|
237
|
+
throw new Error();
|
|
238
|
+
}
|
|
239
|
+
r.loaders.push({
|
|
240
|
+
as: groupKey,
|
|
241
|
+
table: relEntity.table,
|
|
242
|
+
manyJoin,
|
|
243
|
+
oneJoins: relSubsetQuery.joins,
|
|
244
|
+
select: relSubsetQuery.select,
|
|
245
|
+
loaders: relSubsetQuery.loaders
|
|
246
|
+
});
|
|
247
|
+
}
|
|
248
|
+
return r;
|
|
249
|
+
}, {
|
|
250
|
+
select: [],
|
|
251
|
+
virtual: [],
|
|
252
|
+
joins: [],
|
|
253
|
+
loaders: []
|
|
254
|
+
});
|
|
255
|
+
return result;
|
|
256
|
+
}
|
|
257
|
+
/*
|
|
258
|
+
FieldExpr[] 을 EntityPropNode[] 로 변환
|
|
259
|
+
*/ fieldExprsToPropNodes(fieldExprs, entity = this) {
|
|
260
|
+
const groups = fieldExprs.reduce((result, fieldExpr)=>{
|
|
261
|
+
let key, value, elseExpr;
|
|
262
|
+
if (fieldExpr.includes(".")) {
|
|
263
|
+
[key, ...elseExpr] = fieldExpr.split(".");
|
|
264
|
+
value = elseExpr.join(".");
|
|
265
|
+
} else {
|
|
266
|
+
key = "";
|
|
267
|
+
value = fieldExpr;
|
|
268
|
+
}
|
|
269
|
+
result[key] = (result[key] ?? []).concat(value);
|
|
270
|
+
return result;
|
|
271
|
+
}, {});
|
|
272
|
+
return Object.keys(groups).map((key)=>{
|
|
273
|
+
const group = groups[key];
|
|
274
|
+
// 일반 prop 처리
|
|
275
|
+
if (key === "") {
|
|
276
|
+
return group.map((propName)=>{
|
|
277
|
+
// uuid 개별 처리
|
|
278
|
+
if (propName === "uuid") {
|
|
279
|
+
return {
|
|
280
|
+
nodeType: "plain",
|
|
281
|
+
prop: {
|
|
282
|
+
type: "string",
|
|
283
|
+
name: "uuid",
|
|
284
|
+
length: 128
|
|
285
|
+
},
|
|
286
|
+
children: []
|
|
287
|
+
};
|
|
288
|
+
}
|
|
289
|
+
const prop = entity.props.find((p)=>p.name === propName);
|
|
290
|
+
if (prop === undefined) {
|
|
291
|
+
console.log({
|
|
292
|
+
propName,
|
|
293
|
+
groups
|
|
294
|
+
});
|
|
295
|
+
throw new Error(`${entity.id} -- 잘못된 FieldExpr ${propName}`);
|
|
296
|
+
}
|
|
297
|
+
return {
|
|
298
|
+
nodeType: "plain",
|
|
299
|
+
prop,
|
|
300
|
+
children: []
|
|
301
|
+
};
|
|
302
|
+
});
|
|
303
|
+
}
|
|
304
|
+
// relation prop 처리
|
|
305
|
+
const prop = entity.propsDict[key];
|
|
306
|
+
if (!isRelationProp(prop)) {
|
|
307
|
+
throw new Error(`잘못된 FieldExpr ${key}.${group[0]}`);
|
|
308
|
+
}
|
|
309
|
+
const relEntity = EntityManager.get(prop.with);
|
|
310
|
+
// relation -One 에 id 필드 하나인 경우
|
|
311
|
+
if (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop)) {
|
|
312
|
+
if (group.length == 1 && (group[0] === "id" || group[0] == "id?")) {
|
|
313
|
+
// id 하나만 있는지 체크해서, 하나만 있으면 상위 prop으로 id를 리턴
|
|
314
|
+
const idProp = relEntity.propsDict.id;
|
|
315
|
+
return {
|
|
316
|
+
nodeType: "plain",
|
|
317
|
+
prop: {
|
|
318
|
+
...idProp,
|
|
319
|
+
name: key + "_id",
|
|
320
|
+
nullable: prop.nullable
|
|
321
|
+
},
|
|
322
|
+
children: []
|
|
323
|
+
};
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
// -One 그외의 경우 object로 리턴
|
|
327
|
+
// -Many의 경우 array로 리턴
|
|
328
|
+
// Recursive 로 뎁스 처리
|
|
329
|
+
const children = this.fieldExprsToPropNodes(group, relEntity);
|
|
330
|
+
const nodeType = isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop) ? "object" : "array";
|
|
331
|
+
return {
|
|
332
|
+
prop,
|
|
333
|
+
children,
|
|
334
|
+
nodeType
|
|
335
|
+
};
|
|
336
|
+
}).flat();
|
|
337
|
+
}
|
|
338
|
+
getFieldExprs(prefix = "", maxDepth = 3, froms = []) {
|
|
339
|
+
return this.props.map((prop)=>{
|
|
340
|
+
const propName = [
|
|
341
|
+
prefix,
|
|
342
|
+
prop.name
|
|
343
|
+
].filter((v)=>v !== "").join(".");
|
|
344
|
+
if (propName === prefix) {
|
|
345
|
+
return null;
|
|
346
|
+
}
|
|
347
|
+
if (isRelationProp(prop)) {
|
|
348
|
+
if (maxDepth < 0) {
|
|
349
|
+
return null;
|
|
350
|
+
}
|
|
351
|
+
if (froms.includes(prop.with)) {
|
|
352
|
+
// 역방향 relation인 경우 제외
|
|
353
|
+
return null;
|
|
354
|
+
}
|
|
355
|
+
// 정방향 relation인 경우 recursive 콜
|
|
356
|
+
const relMd = EntityManager.get(prop.with);
|
|
357
|
+
return relMd.getFieldExprs(propName, maxDepth - 1, [
|
|
358
|
+
...froms,
|
|
359
|
+
this.id
|
|
360
|
+
]);
|
|
361
|
+
}
|
|
362
|
+
return propName;
|
|
363
|
+
}).flat().filter((f)=>f !== null);
|
|
364
|
+
}
|
|
365
|
+
getTableColumns() {
|
|
366
|
+
return this.props.map((prop)=>{
|
|
367
|
+
if (prop.type === "relation") {
|
|
368
|
+
if (prop.relationType === "BelongsToOne" || prop.relationType === "OneToOne" && prop.hasJoinColumn === true) {
|
|
369
|
+
return {
|
|
370
|
+
name: `${prop.name}_id`,
|
|
371
|
+
type: "int_unsigned"
|
|
372
|
+
};
|
|
373
|
+
} else {
|
|
374
|
+
return null;
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
return {
|
|
378
|
+
name: prop.name,
|
|
379
|
+
type: prop.type
|
|
380
|
+
};
|
|
381
|
+
}).filter(nonNullable);
|
|
382
|
+
}
|
|
383
|
+
async registerModulePaths() {
|
|
384
|
+
const basePath = `${this.names.parentFs}`;
|
|
385
|
+
// base-scheme
|
|
386
|
+
EntityManager.setModulePath(`${this.id}BaseSchema`, `sonamu.generated`);
|
|
387
|
+
// subset
|
|
388
|
+
if (Object.keys(this.subsets).length > 0) {
|
|
389
|
+
EntityManager.setModulePath(`${this.id}SubsetKey`, `sonamu.generated`);
|
|
390
|
+
EntityManager.setModulePath(`${this.id}SubsetMapping`, `sonamu.generated`);
|
|
391
|
+
Object.keys(this.subsets).map((subsetKey)=>{
|
|
392
|
+
EntityManager.setModulePath(`${this.id}Subset${subsetKey.toUpperCase()}`, `sonamu.generated`);
|
|
393
|
+
});
|
|
394
|
+
}
|
|
395
|
+
// enums
|
|
396
|
+
Object.keys(this.enumLabels).map((enumId)=>{
|
|
397
|
+
EntityManager.setModulePath(enumId, `sonamu.generated`);
|
|
398
|
+
});
|
|
399
|
+
// types
|
|
400
|
+
const typesModulePath = `${basePath}/${this.names.parentFs}.types`;
|
|
401
|
+
const typesFileDistPath = path.join(Sonamu.apiRootPath, `dist/application/${typesModulePath}.js`);
|
|
402
|
+
if (await exists(typesFileDistPath)) {
|
|
403
|
+
const importUrl = createImportUrl(typesFileDistPath);
|
|
404
|
+
const t = await import(importUrl);
|
|
405
|
+
this.types = Object.keys(t).reduce((result, key)=>{
|
|
406
|
+
EntityManager.setModulePath(key, typesModulePath);
|
|
407
|
+
return {
|
|
408
|
+
...result,
|
|
409
|
+
[key]: t[key]
|
|
410
|
+
};
|
|
411
|
+
}, {});
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
registerTableSpecs() {
|
|
415
|
+
const uniqueIndexes = this.indexes.filter((idx)=>idx.type === "unique");
|
|
416
|
+
EntityManager.setTableSpec({
|
|
417
|
+
name: this.table,
|
|
418
|
+
uniqueIndexes
|
|
419
|
+
});
|
|
420
|
+
}
|
|
421
|
+
toJson() {
|
|
422
|
+
return {
|
|
423
|
+
id: this.id,
|
|
424
|
+
parentId: this.parentId,
|
|
425
|
+
table: this.table,
|
|
426
|
+
title: this.title,
|
|
427
|
+
props: this.props,
|
|
428
|
+
indexes: this.indexes,
|
|
429
|
+
subsets: this.subsets,
|
|
430
|
+
enums: this.enumLabels
|
|
431
|
+
};
|
|
432
|
+
}
|
|
433
|
+
async save() {
|
|
434
|
+
// sort: subsets
|
|
435
|
+
const subsetRows = this.getSubsetRows();
|
|
436
|
+
this.subsets = Object.fromEntries(Object.entries(this.subsets).map(([subsetKey])=>{
|
|
437
|
+
return [
|
|
438
|
+
subsetKey,
|
|
439
|
+
this.subsetRowsToSubsetFields(subsetRows, subsetKey)
|
|
440
|
+
];
|
|
441
|
+
}));
|
|
442
|
+
// save
|
|
443
|
+
const jsonPath = path.join(Sonamu.apiRootPath, `src/application/${this.names.parentFs}/${this.names.fs}.entity.json`);
|
|
444
|
+
const json = this.toJson();
|
|
445
|
+
await writeFile(jsonPath, await prettier.format(JSON.stringify(json), {
|
|
446
|
+
parser: "json"
|
|
447
|
+
}));
|
|
448
|
+
// reload
|
|
449
|
+
await EntityManager.register(json);
|
|
450
|
+
}
|
|
451
|
+
getSubsetRows(_subsets, prefixes = []) {
|
|
452
|
+
if (prefixes.length > 10) {
|
|
453
|
+
return [];
|
|
454
|
+
}
|
|
455
|
+
const subsets = _subsets ?? this.subsets;
|
|
456
|
+
const subsetKeys = Object.keys(subsets);
|
|
457
|
+
const allFields = _.uniq(subsetKeys.map((key)=>subsets[key]).flat());
|
|
458
|
+
return this.props.map((prop)=>{
|
|
459
|
+
if (prop.type === "relation" && allFields.find((f)=>f.startsWith([
|
|
460
|
+
...prefixes,
|
|
461
|
+
prop.name
|
|
462
|
+
].join(".") + "."))) {
|
|
463
|
+
const relEntity = EntityManager.get(prop.with);
|
|
464
|
+
const children = relEntity.getSubsetRows(subsets, [
|
|
465
|
+
...prefixes,
|
|
466
|
+
`${prop.name}`
|
|
467
|
+
]);
|
|
468
|
+
return {
|
|
469
|
+
field: prop.name,
|
|
470
|
+
children,
|
|
471
|
+
relationEntity: prop.with,
|
|
472
|
+
prefixes,
|
|
473
|
+
isOpen: children.length > 0,
|
|
474
|
+
has: Object.fromEntries(subsetKeys.map((subsetKey)=>{
|
|
475
|
+
return [
|
|
476
|
+
subsetKey,
|
|
477
|
+
children.every((child)=>child.has[subsetKey] === true)
|
|
478
|
+
];
|
|
479
|
+
}))
|
|
480
|
+
};
|
|
481
|
+
}
|
|
482
|
+
return {
|
|
483
|
+
field: prop.name,
|
|
484
|
+
children: [],
|
|
485
|
+
relationEntity: prop.type === "relation" ? prop.with : undefined,
|
|
486
|
+
prefixes,
|
|
487
|
+
has: Object.fromEntries(subsetKeys.map((subsetKey)=>{
|
|
488
|
+
const subsetFields = subsets[subsetKey];
|
|
489
|
+
const has = subsetFields.some((f)=>{
|
|
490
|
+
const field = [
|
|
491
|
+
...prefixes,
|
|
492
|
+
prop.name
|
|
493
|
+
].join(".");
|
|
494
|
+
return f === field || f.startsWith(field + ".");
|
|
495
|
+
});
|
|
496
|
+
return [
|
|
497
|
+
subsetKey,
|
|
498
|
+
has
|
|
499
|
+
];
|
|
500
|
+
}))
|
|
501
|
+
};
|
|
502
|
+
});
|
|
503
|
+
}
|
|
504
|
+
subsetRowsToSubsetFields(subsetRows, subsetKey) {
|
|
505
|
+
return subsetRows.map((subsetRow)=>{
|
|
506
|
+
if (subsetRow.children.length > 0) {
|
|
507
|
+
return this.subsetRowsToSubsetFields(subsetRow.children, subsetKey);
|
|
508
|
+
} else if (subsetRow.has[subsetKey]) {
|
|
509
|
+
return subsetRow.prefixes.concat(subsetRow.field).join(".");
|
|
510
|
+
} else {
|
|
511
|
+
return null;
|
|
512
|
+
}
|
|
513
|
+
}).filter(nonNullable).flat();
|
|
514
|
+
}
|
|
515
|
+
async createProp(prop, at) {
|
|
516
|
+
if (!at) {
|
|
517
|
+
this.props.push(prop);
|
|
518
|
+
} else {
|
|
519
|
+
this.props.splice(at, 0, prop);
|
|
520
|
+
}
|
|
521
|
+
await this.save();
|
|
522
|
+
}
|
|
523
|
+
analyzeSubsetField(subsetField) {
|
|
524
|
+
const arr = subsetField.split(".");
|
|
525
|
+
let entityId = this.id;
|
|
526
|
+
const result = [];
|
|
527
|
+
for(let i = 0; i < arr.length; i++){
|
|
528
|
+
const propName = arr[i];
|
|
529
|
+
result.push({
|
|
530
|
+
entityId,
|
|
531
|
+
propName
|
|
532
|
+
});
|
|
533
|
+
const prop = EntityManager.get(entityId).props.find((p)=>p.name === propName);
|
|
534
|
+
if (!prop) {
|
|
535
|
+
throw new Error(`${entityId}의 잘못된 서브셋키 ${subsetField}`);
|
|
536
|
+
}
|
|
537
|
+
if (isRelationProp(prop)) {
|
|
538
|
+
entityId = prop.with;
|
|
539
|
+
}
|
|
540
|
+
}
|
|
541
|
+
return result;
|
|
542
|
+
}
|
|
543
|
+
async modifyProp(newProp, at) {
|
|
544
|
+
// 이전 프롭 이름 저장
|
|
545
|
+
const oldName = this.props[at].name;
|
|
546
|
+
// 저장할 엔티티
|
|
547
|
+
const entities = [
|
|
548
|
+
this
|
|
549
|
+
];
|
|
550
|
+
// 이름이 바뀐 경우
|
|
551
|
+
if (oldName !== newProp.name) {
|
|
552
|
+
// 전체 엔티티에서 현재 수정된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 수정
|
|
553
|
+
const allEntityIds = EntityManager.getAllIds();
|
|
554
|
+
for (const relEntityId of allEntityIds){
|
|
555
|
+
const relEntity = EntityManager.get(relEntityId);
|
|
556
|
+
const relEntitySubsetKeys = Object.keys(relEntity.subsets);
|
|
557
|
+
for (const subsetKey of relEntitySubsetKeys){
|
|
558
|
+
const subset = relEntity.subsets[subsetKey];
|
|
559
|
+
// 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 수정 처리
|
|
560
|
+
const modifiedSubsetFields = subset.map((subsetField)=>{
|
|
561
|
+
const analyzed = relEntity.analyzeSubsetField(subsetField);
|
|
562
|
+
const modified = analyzed.map((a)=>a.propName === oldName && a.entityId === this.id ? {
|
|
563
|
+
...a,
|
|
564
|
+
propName: newProp.name
|
|
565
|
+
} : a);
|
|
566
|
+
// 분석한 필드를 다시 서브셋 필드로 복구
|
|
567
|
+
return modified.map((a)=>a.propName).join(".");
|
|
568
|
+
});
|
|
569
|
+
if (subset.join(",") !== modifiedSubsetFields.join(",")) {
|
|
570
|
+
relEntity.subsets[subsetKey] = modifiedSubsetFields;
|
|
571
|
+
entities.push(relEntity);
|
|
572
|
+
}
|
|
573
|
+
}
|
|
574
|
+
}
|
|
575
|
+
}
|
|
576
|
+
// 프롭 수정
|
|
577
|
+
this.props[at] = newProp;
|
|
578
|
+
await Promise.all(entities.map(async (entity)=>entity.save()));
|
|
579
|
+
}
|
|
580
|
+
async delProp(at) {
|
|
581
|
+
// 이전 프롭 이름 저장
|
|
582
|
+
const oldName = this.props[at].name;
|
|
583
|
+
// 저장할 엔티티
|
|
584
|
+
const entities = [
|
|
585
|
+
this
|
|
586
|
+
];
|
|
587
|
+
// 전체 엔티티에서 현재 삭제된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 제외
|
|
588
|
+
const allEntityIds = EntityManager.getAllIds();
|
|
589
|
+
for (const relEntityId of allEntityIds){
|
|
590
|
+
const relEntity = EntityManager.get(relEntityId);
|
|
591
|
+
const relEntitySubsetKeys = Object.keys(relEntity.subsets);
|
|
592
|
+
for (const subsetKey of relEntitySubsetKeys){
|
|
593
|
+
const subset = relEntity.subsets[subsetKey];
|
|
594
|
+
// 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 이후의 필드를 제외
|
|
595
|
+
const modifiedSubsetFields = subset.map((subsetField)=>{
|
|
596
|
+
const analyzed = relEntity.analyzeSubsetField(subsetField);
|
|
597
|
+
if (analyzed.find((a)=>a.propName === oldName && a.entityId === this.id)) {
|
|
598
|
+
return null;
|
|
599
|
+
} else {
|
|
600
|
+
return subsetField;
|
|
601
|
+
}
|
|
602
|
+
}).filter(nonNullable);
|
|
603
|
+
if (subset.join(",") !== modifiedSubsetFields.join(",")) {
|
|
604
|
+
relEntity.subsets[subsetKey] = modifiedSubsetFields;
|
|
605
|
+
entities.push(relEntity);
|
|
606
|
+
}
|
|
607
|
+
}
|
|
608
|
+
}
|
|
609
|
+
// 현재 엔티티의 인덱스에서 제외
|
|
610
|
+
EntityManager.get(this.id).indexes.map((index)=>{
|
|
611
|
+
index.columns = index.columns.filter((col)=>col !== oldName);
|
|
612
|
+
});
|
|
613
|
+
// 프롭 삭제
|
|
614
|
+
this.props.splice(at, 1);
|
|
615
|
+
await Promise.all(entities.map(async (entity)=>entity.save()));
|
|
616
|
+
}
|
|
617
|
+
getEntityIdFromSubsetField(subsetField) {
|
|
618
|
+
if (subsetField.includes(".") === false) {
|
|
619
|
+
return this.id;
|
|
620
|
+
}
|
|
621
|
+
// 서브셋 필드의 마지막은 프롭이므로 제외
|
|
622
|
+
const arr = subsetField.split(".").slice(0, -1);
|
|
623
|
+
// 서브셋 필드를 내려가면서 마지막으로 relation된 엔티티를 찾음
|
|
624
|
+
const lastEntityId = arr.reduce((entityId, field)=>{
|
|
625
|
+
const relProp = EntityManager.get(entityId).props.find((p)=>p.name === field);
|
|
626
|
+
if (!relProp || relProp.type !== "relation") {
|
|
627
|
+
console.debug({
|
|
628
|
+
arr,
|
|
629
|
+
thisId: this.id,
|
|
630
|
+
entityId,
|
|
631
|
+
field
|
|
632
|
+
});
|
|
633
|
+
throw new Error(`잘못된 서브셋키 ${subsetField}`);
|
|
634
|
+
}
|
|
635
|
+
return relProp.with;
|
|
636
|
+
}, this.id);
|
|
637
|
+
return lastEntityId;
|
|
638
|
+
}
|
|
639
|
+
async moveProp(at, to) {
|
|
640
|
+
const prop = this.props[at];
|
|
641
|
+
const newProps = [
|
|
642
|
+
...this.props
|
|
643
|
+
];
|
|
644
|
+
newProps.splice(to, 0, prop);
|
|
645
|
+
newProps.splice(at < to ? at : at + 1, 1);
|
|
646
|
+
this.props = newProps;
|
|
647
|
+
await this.save();
|
|
648
|
+
}
|
|
649
|
+
}
|
|
650
|
+
|
|
651
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9lbnRpdHkvZW50aXR5LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIF8gZnJvbSBcImxvZGFzaC1lc1wiO1xuaW1wb3J0IHsgY3JlYXRlSW1wb3J0VXJsIH0gZnJvbSBcIi4uL3V0aWxzL2VzbS11dGlsc1wiO1xuaW1wb3J0IHsgRW50aXR5TWFuYWdlciBhcyBFbnRpdHlNYW5hZ2VyIH0gZnJvbSBcIi4vZW50aXR5LW1hbmFnZXJcIjtcbmltcG9ydCB7XG4gIEVudGl0eVByb3AsXG4gIFJlbGF0aW9uUHJvcCxcbiAgaXNSZWxhdGlvblByb3AsXG4gIFN1YnNldFF1ZXJ5LFxuICBpc1ZpcnR1YWxQcm9wLFxuICBpc0JlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcCxcbiAgaXNPbmVUb09uZVJlbGF0aW9uUHJvcCxcbiAgaXNIYXNNYW55UmVsYXRpb25Qcm9wLFxuICBpc01hbnlUb01hbnlSZWxhdGlvblByb3AsXG4gIEVudGl0eVByb3BOb2RlLFxuICBpc0VudW1Qcm9wLFxuICBTdHJpbmdQcm9wLFxuICBFbnRpdHlJbmRleCxcbiAgRW50aXR5SnNvbixcbiAgRW50aXR5U3Vic2V0Um93LFxufSBmcm9tIFwiLi4vdHlwZXMvdHlwZXNcIjtcbmltcG9ydCBpbmZsZWN0aW9uIGZyb20gXCJpbmZsZWN0aW9uXCI7XG5pbXBvcnQgcGF0aCBmcm9tIFwicGF0aFwiO1xuaW1wb3J0IHsgd3JpdGVGaWxlIH0gZnJvbSBcImZzL3Byb21pc2VzXCI7XG5pbXBvcnQgeyB6IH0gZnJvbSBcInpvZFwiO1xuaW1wb3J0IHsgU29uYW11IH0gZnJvbSBcIi4uL2FwaS9zb25hbXVcIjtcbmltcG9ydCBwcmV0dGllciBmcm9tIFwicHJldHRpZXJcIjtcbmltcG9ydCB7IG5vbk51bGxhYmxlIH0gZnJvbSBcIi4uL3V0aWxzL3V0aWxzXCI7XG5pbXBvcnQgeyBleGlzdHMgfSBmcm9tIFwiLi4vdXRpbHMvZnMtdXRpbHNcIjtcblxuZXhwb3J0IGNsYXNzIEVudGl0eSB7XG4gIGlkOiBzdHJpbmc7XG4gIHBhcmVudElkPzogc3RyaW5nO1xuICB0YWJsZTogc3RyaW5nO1xuICB0aXRsZTogc3RyaW5nO1xuICBuYW1lczoge1xuICAgIHBhcmVudEZzOiBzdHJpbmc7XG4gICAgZnM6IHN0cmluZztcbiAgICBtb2R1bGU6IHN0cmluZztcbiAgfTtcbiAgcHJvcHM6IEVudGl0eVByb3BbXTtcbiAgcHJvcHNEaWN0OiB7XG4gICAgW2tleTogc3RyaW5nXTogRW50aXR5UHJvcDtcbiAgfTtcbiAgcmVsYXRpb25zOiB7XG4gICAgW2tleTogc3RyaW5nXTogUmVsYXRpb25Qcm9wO1xuICB9O1xuICBpbmRleGVzOiBFbnRpdHlJbmRleFtdO1xuICBzdWJzZXRzOiB7XG4gICAgW2tleTogc3RyaW5nXTogc3RyaW5nW107XG4gIH07XG4gIHR5cGVzOiB7XG4gICAgW25hbWU6IHN0cmluZ106IHouWm9kVHlwZUFueTtcbiAgfSA9IHt9O1xuICBlbnVtczoge1xuICAgIFtlbnVtSWQ6IHN0cmluZ106IHouWm9kRW51bTxhbnk+O1xuICB9ID0ge307XG4gIGVudW1MYWJlbHM6IHtcbiAgICBbZW51bUlkOiBzdHJpbmddOiB7XG4gICAgICBba2V5OiBzdHJpbmddOiBzdHJpbmc7XG4gICAgfTtcbiAgfSA9IHt9O1xuXG4gIGNvbnN0cnVjdG9yKHtcbiAgICBpZCxcbiAgICBwYXJlbnRJZCxcbiAgICB0YWJsZSxcbiAgICB0aXRsZSxcbiAgICBwcm9wcyxcbiAgICBpbmRleGVzLFxuICAgIHN1YnNldHMsXG4gICAgZW51bXMsXG4gIH06IEVudGl0eUpzb24pIHtcbiAgICAvLyBpZFxuICAgIHRoaXMuaWQgPSBpZDtcbiAgICB0aGlzLnBhcmVudElkID0gcGFyZW50SWQ7XG4gICAgdGhpcy50aXRsZSA9IHRpdGxlID8/IHRoaXMuaWQ7XG4gICAgdGhpcy50YWJsZSA9IHRhYmxlID8/IGluZmxlY3Rpb24udW5kZXJzY29yZShpbmZsZWN0aW9uLnBsdXJhbGl6ZShpZCkpO1xuXG4gICAgLy8gcHJvcHNcbiAgICBpZiAocHJvcHMpIHtcbiAgICAgIHRoaXMucHJvcHMgPSBwcm9wcy5tYXAoKHByb3ApID0+IHtcbiAgICAgICAgaWYgKGlzRW51bVByb3AocHJvcCkpIHtcbiAgICAgICAgICBpZiAocHJvcC5pZC5pbmNsdWRlcyhcIiRNb2RlbFwiKSkge1xuICAgICAgICAgICAgcHJvcC5pZCA9IHByb3AuaWQucmVwbGFjZShcIiRNb2RlbFwiLCBpZCk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiBwcm9wO1xuICAgICAgfSk7XG4gICAgICB0aGlzLnByb3BzRGljdCA9IHByb3BzLnJlZHVjZSgocmVzdWx0LCBwcm9wKSA9PiB7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgLi4ucmVzdWx0LFxuICAgICAgICAgIFtwcm9wLm5hbWVdOiBwcm9wLFxuICAgICAgICB9O1xuICAgICAgfSwge30pO1xuXG4gICAgICAvLyByZWxhdGlvbnNcbiAgICAgIHRoaXMucmVsYXRpb25zID0gcHJvcHNcbiAgICAgICAgLmZpbHRlcigocHJvcCkgPT4gaXNSZWxhdGlvblByb3AocHJvcCkpXG4gICAgICAgIC5yZWR1Y2UoKHJlc3VsdCwgcHJvcCkgPT4ge1xuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAuLi5yZXN1bHQsXG4gICAgICAgICAgICBbcHJvcC5uYW1lXTogcHJvcCxcbiAgICAgICAgICB9O1xuICAgICAgICB9LCB7fSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMucHJvcHMgPSBbXTtcbiAgICAgIHRoaXMucHJvcHNEaWN0ID0ge307XG4gICAgICB0aGlzLnJlbGF0aW9ucyA9IHt9O1xuICAgIH1cblxuICAgIC8vIGluZGV4ZXNcbiAgICB0aGlzLmluZGV4ZXMgPSBpbmRleGVzID8/IFtdO1xuXG4gICAgLy8gc3Vic2V0c1xuICAgIHRoaXMuc3Vic2V0cyA9IHN1YnNldHMgPz8ge307XG5cbiAgICAvLyBlbnVtc1xuICAgIHRoaXMuZW51bUxhYmVscyA9IGVudW1zID8/IHt9O1xuICAgIHRoaXMuZW51bXMgPSBPYmplY3QuZnJvbUVudHJpZXMoXG4gICAgICBPYmplY3QuZW50cmllcyh0aGlzLmVudW1MYWJlbHMpLm1hcCgoW2tleSwgZW51bUxhYmVsXSkgPT4ge1xuICAgICAgICByZXR1cm4gW1xuICAgICAgICAgIGtleSxcbiAgICAgICAgICB6LmVudW0oXG4gICAgICAgICAgICBPYmplY3Qua2V5cyhlbnVtTGFiZWwpIGFzIHVua25vd24gYXMgcmVhZG9ubHkgW3N0cmluZywgLi4uc3RyaW5nW11dXG4gICAgICAgICAgKSxcbiAgICAgICAgXTtcbiAgICAgIH0pXG4gICAgKTtcblxuICAgIC8vIG5hbWVzXG4gICAgdGhpcy5uYW1lcyA9IHtcbiAgICAgIHBhcmVudEZzOiBpbmZsZWN0aW9uXG4gICAgICAgIC5kYXNoZXJpemUoaW5mbGVjdGlvbi51bmRlcnNjb3JlKHBhcmVudElkID8/IGlkKSlcbiAgICAgICAgLnRvTG93ZXJDYXNlKCksXG4gICAgICBmczogaW5mbGVjdGlvbi5kYXNoZXJpemUoaW5mbGVjdGlvbi51bmRlcnNjb3JlKGlkKSkudG9Mb3dlckNhc2UoKSxcbiAgICAgIG1vZHVsZTogaWQsXG4gICAgfTtcbiAgfVxuXG4gIC8qXG4gICAgc3Vic2V0IFNFTEVDVC9KT0lOL0xPQURFUiDqsrDqs7wg66as7YS0XG4gICovXG4gIGdldFN1YnNldFF1ZXJ5KHN1YnNldEtleTogc3RyaW5nKTogU3Vic2V0UXVlcnkge1xuICAgIGNvbnN0IHN1YnNldCA9IHRoaXMuc3Vic2V0c1tzdWJzZXRLZXldO1xuXG4gICAgY29uc3QgcmVzdWx0OiBTdWJzZXRRdWVyeSA9IHRoaXMucmVzb2x2ZVN1YnNldFF1ZXJ5KFwiXCIsIHN1YnNldCk7XG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gIC8qXG4gICAqL1xuICByZXNvbHZlU3Vic2V0UXVlcnkoXG4gICAgcHJlZml4OiBzdHJpbmcsXG4gICAgZmllbGRzOiBzdHJpbmdbXSxcbiAgICBpc0FscmVhZHlPdXRlckpvaW5lZDogYm9vbGVhbiA9IGZhbHNlXG4gICk6IFN1YnNldFF1ZXJ5IHtcbiAgICAvLyBwcmVmaXgg7LmY7ZmYIChwcmVmaXjripQgVG9PbmVSZWxhdGlvbuydtCDrs7XsiJjroZwg67aZ7J2AIOqyveyasCDrqqjrkZAgX1/roZwg67OA6rK965CoKVxuICAgIHByZWZpeCA9IHByZWZpeC5yZXBsYWNlKC9cXC4vZywgXCJfX1wiKTtcblxuICAgIC8vIOyEnOu4jOyFi+ydhCAx646B7Iqk66eMIOu2hOumrO2VmOyXrCDqt7jro7ntlZFcbiAgICBjb25zdCBzdWJzZXRHcm91cCA9IF8uZ3JvdXBCeShmaWVsZHMsIChmaWVsZCkgPT4ge1xuICAgICAgaWYgKGZpZWxkLmluY2x1ZGVzKFwiLlwiKSkge1xuICAgICAgICBjb25zdCBbcmVsXSA9IGZpZWxkLnNwbGl0KFwiLlwiKTtcbiAgICAgICAgcmV0dXJuIHJlbDtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHJldHVybiBcIlwiO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgY29uc3QgcmVzdWx0ID0gT2JqZWN0LmtleXMoc3Vic2V0R3JvdXApLnJlZHVjZShcbiAgICAgIChyLCBncm91cEtleSkgPT4ge1xuICAgICAgICBjb25zdCBmaWVsZHMgPSBzdWJzZXRHcm91cFtncm91cEtleV07XG4gICAgICAgIC8vIO2YhOyerCDthYzsnbTruJQg7ZWE65Oc7IWL7J2AIHNlbGVjdCwgdmlydHVhbOyXkCDstpTqsIDtlZjqs6Ag66as7YS0XG4gICAgICAgIGlmIChncm91cEtleSA9PT0gXCJcIikge1xuICAgICAgICAgIGNvbnN0IHJlYWxGaWVsZHMgPSBmaWVsZHMuZmlsdGVyKFxuICAgICAgICAgICAgKGZpZWxkKSA9PiAhaXNWaXJ0dWFsUHJvcCh0aGlzLnByb3BzRGljdFtmaWVsZF0pXG4gICAgICAgICAgKTtcbiAgICAgICAgICBjb25zdCB2aXJ0dWFsRmllbGRzID0gZmllbGRzLmZpbHRlcigoZmllbGQpID0+XG4gICAgICAgICAgICBpc1ZpcnR1YWxQcm9wKHRoaXMucHJvcHNEaWN0W2ZpZWxkXSlcbiAgICAgICAgICApO1xuXG4gICAgICAgICAgaWYgKHByZWZpeCA9PT0gXCJcIikge1xuICAgICAgICAgICAgLy8g7ZiE7J6sIO2FjOydtOu4lOyduCDqsr3smrBcbiAgICAgICAgICAgIHIuc2VsZWN0ID0gci5zZWxlY3QuY29uY2F0KFxuICAgICAgICAgICAgICByZWFsRmllbGRzLm1hcCgoZmllbGQpID0+IGAke3RoaXMudGFibGV9LiR7ZmllbGR9YClcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgICByLnZpcnR1YWwgPSByLnZpcnR1YWwuY29uY2F0KHZpcnR1YWxGaWVsZHMpO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAvLyDrhJjslrTsmKgg7YWM7J2067iU7J24IOqyveyasFxuICAgICAgICAgICAgci5zZWxlY3QgPSByLnNlbGVjdC5jb25jYXQoXG4gICAgICAgICAgICAgIHJlYWxGaWVsZHMubWFwKFxuICAgICAgICAgICAgICAgIChmaWVsZCkgPT4gYCR7cHJlZml4fS4ke2ZpZWxkfSBhcyAke3ByZWZpeH1fXyR7ZmllbGR9YFxuICAgICAgICAgICAgICApXG4gICAgICAgICAgICApO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHJldHVybiByO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgcmVsYXRpb24gPSB0aGlzLnJlbGF0aW9uc1tncm91cEtleV07XG4gICAgICAgIGlmIChyZWxhdGlvbiA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDsobTsnqztlZjsp4Ag7JWK64qUIHJlbGF0aW9uIOywuOyhsCAke2dyb3VwS2V5fWApO1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IHJlbEVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KHJlbGF0aW9uLndpdGgpO1xuXG4gICAgICAgIGlmIChcbiAgICAgICAgICBpc09uZVRvT25lUmVsYXRpb25Qcm9wKHJlbGF0aW9uKSB8fFxuICAgICAgICAgIGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wKHJlbGF0aW9uKVxuICAgICAgICApIHtcbiAgICAgICAgICAvLyAtT25lIFJlbGF0aW9uOiBKT0lOIOycvOuhnCDsspjrpqxcbiAgICAgICAgICBjb25zdCByZWxGaWVsZHMgPSBmaWVsZHMubWFwKChmaWVsZCkgPT5cbiAgICAgICAgICAgIGZpZWxkLnNwbGl0KFwiLlwiKS5zbGljZSgxKS5qb2luKFwiLlwiKVxuICAgICAgICAgICk7XG5cbiAgICAgICAgICAvLyAtT25lIFJlbGF0aW9u7JeQ7IScIGlkIO2VhOuTnOunjCDssLjsobDtlZjripQg6rK97JqwIOumtOugiOydtOyFmCDrhJjquLDsp4Ag7JWK6rOgIOumrO2EtFxuICAgICAgICAgIGlmIChyZWxGaWVsZHMubGVuZ3RoID09PSAxICYmIHJlbEZpZWxkc1swXSA9PT0gXCJpZFwiKSB7XG4gICAgICAgICAgICBpZiAocHJlZml4ID09PSBcIlwiKSB7XG4gICAgICAgICAgICAgIHIuc2VsZWN0ID0gci5zZWxlY3QuY29uY2F0KGAke3RoaXMudGFibGV9LiR7Z3JvdXBLZXl9X2lkYCk7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICByLnNlbGVjdCA9IHIuc2VsZWN0LmNvbmNhdChcbiAgICAgICAgICAgICAgICBgJHtwcmVmaXh9LiR7Z3JvdXBLZXl9X2lkIGFzICR7cHJlZml4fV9fJHtncm91cEtleX1faWRgXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gcjtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBpbm5lck9yT3V0ZXJcbiAgICAgICAgICBjb25zdCBpbm5lck9yT3V0ZXIgPSAoKCkgPT4ge1xuICAgICAgICAgICAgaWYgKGlzQWxyZWFkeU91dGVySm9pbmVkKSB7XG4gICAgICAgICAgICAgIHJldHVybiBcIm91dGVyXCI7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmIChpc09uZVRvT25lUmVsYXRpb25Qcm9wKHJlbGF0aW9uKSkge1xuICAgICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgICAgcmVsYXRpb24uaGFzSm9pbkNvbHVtbiA9PT0gdHJ1ZSAmJlxuICAgICAgICAgICAgICAgIChyZWxhdGlvbi5udWxsYWJsZSA/PyBmYWxzZSkgPT09IGZhbHNlXG4gICAgICAgICAgICAgICkge1xuICAgICAgICAgICAgICAgIHJldHVybiBcImlubmVyXCI7XG4gICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIFwib3V0ZXJcIjtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgaWYgKHJlbGF0aW9uLm51bGxhYmxlKSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIFwib3V0ZXJcIjtcbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICByZXR1cm4gXCJpbm5lclwiO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSkoKTtcbiAgICAgICAgICBjb25zdCByZWxTdWJzZXRRdWVyeSA9IHJlbEVudGl0eS5yZXNvbHZlU3Vic2V0UXVlcnkoXG4gICAgICAgICAgICBgJHtwcmVmaXggIT09IFwiXCIgPyBwcmVmaXggKyBcIi5cIiA6IFwiXCJ9JHtncm91cEtleX1gLFxuICAgICAgICAgICAgcmVsRmllbGRzLFxuICAgICAgICAgICAgaW5uZXJPck91dGVyID09PSBcIm91dGVyXCJcbiAgICAgICAgICApO1xuICAgICAgICAgIHIuc2VsZWN0ID0gci5zZWxlY3QuY29uY2F0KHJlbFN1YnNldFF1ZXJ5LnNlbGVjdCk7XG4gICAgICAgICAgci52aXJ0dWFsID0gci52aXJ0dWFsLmNvbmNhdChyZWxTdWJzZXRRdWVyeS52aXJ0dWFsKTtcblxuICAgICAgICAgIGNvbnN0IGpvaW5BcyA9IHByZWZpeCA9PT0gXCJcIiA/IGdyb3VwS2V5IDogcHJlZml4ICsgXCJfX1wiICsgZ3JvdXBLZXk7XG4gICAgICAgICAgY29uc3QgZnJvbVRhYmxlID0gcHJlZml4ID09PSBcIlwiID8gdGhpcy50YWJsZSA6IHByZWZpeDtcblxuICAgICAgICAgIGxldCBqb2luQ2xhdXNlO1xuICAgICAgICAgIGlmIChyZWxhdGlvbi5jdXN0b21Kb2luQ2xhdXNlKSB7XG4gICAgICAgICAgICBqb2luQ2xhdXNlID0ge1xuICAgICAgICAgICAgICBjdXN0b206IHJlbGF0aW9uLmN1c3RvbUpvaW5DbGF1c2UsXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBsZXQgZnJvbSwgdG87XG4gICAgICAgICAgICBpZiAoaXNPbmVUb09uZVJlbGF0aW9uUHJvcChyZWxhdGlvbikpIHtcbiAgICAgICAgICAgICAgaWYgKHJlbGF0aW9uLmhhc0pvaW5Db2x1bW4pIHtcbiAgICAgICAgICAgICAgICBmcm9tID0gYCR7ZnJvbVRhYmxlfS4ke3JlbGF0aW9uLm5hbWV9X2lkYDtcbiAgICAgICAgICAgICAgICB0byA9IGAke2pvaW5Bc30uaWRgO1xuICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIGZyb20gPSBgJHtmcm9tVGFibGV9LmlkYDtcbiAgICAgICAgICAgICAgICB0byA9IGAke2pvaW5Bc30uJHtpbmZsZWN0aW9uLnVuZGVyc2NvcmUoXG4gICAgICAgICAgICAgICAgICB0aGlzLm5hbWVzLmZzLnJlcGxhY2UoL1xcLS9nLCBcIl9cIilcbiAgICAgICAgICAgICAgICApfV9pZGA7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIGZyb20gPSBgJHtmcm9tVGFibGV9LiR7cmVsYXRpb24ubmFtZX1faWRgO1xuICAgICAgICAgICAgICB0byA9IGAke2pvaW5Bc30uaWRgO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgam9pbkNsYXVzZSA9IHtcbiAgICAgICAgICAgICAgZnJvbSxcbiAgICAgICAgICAgICAgdG8sXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHIuam9pbnMucHVzaCh7XG4gICAgICAgICAgICBhczogam9pbkFzLFxuICAgICAgICAgICAgam9pbjogaW5uZXJPck91dGVyLFxuICAgICAgICAgICAgdGFibGU6IHJlbEVudGl0eS50YWJsZSxcbiAgICAgICAgICAgIC4uLmpvaW5DbGF1c2UsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICAvLyBCZWxvbmdzVG9PbmUg67CR7JeQIEhhc01hbnnqsIAg67aZ7J2AIOqyveyasFxuICAgICAgICAgIGlmIChyZWxTdWJzZXRRdWVyeS5sb2FkZXJzLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgIGNvbnN0IGNvbnZlcnRlZExvYWRlcnMgPSByZWxTdWJzZXRRdWVyeS5sb2FkZXJzLm1hcCgobG9hZGVyKSA9PiB7XG4gICAgICAgICAgICAgIGNvbnN0IG5ld0FzID0gW2dyb3VwS2V5LCBsb2FkZXIuYXNdLmpvaW4oXCJfX1wiKTtcbiAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICBhczogbmV3QXMsXG4gICAgICAgICAgICAgICAgdGFibGU6IGxvYWRlci50YWJsZSxcbiAgICAgICAgICAgICAgICBtYW55Sm9pbjogbG9hZGVyLm1hbnlKb2luLFxuICAgICAgICAgICAgICAgIG9uZUpvaW5zOiBsb2FkZXIub25lSm9pbnMsXG4gICAgICAgICAgICAgICAgc2VsZWN0OiBsb2FkZXIuc2VsZWN0LFxuICAgICAgICAgICAgICAgIGxvYWRlcnM6IGxvYWRlci5sb2FkZXJzLFxuICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgIHIubG9hZGVycyA9IFsuLi5yLmxvYWRlcnMsIC4uLmNvbnZlcnRlZExvYWRlcnNdO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHIuam9pbnMgPSByLmpvaW5zLmNvbmNhdChyZWxTdWJzZXRRdWVyeS5qb2lucyk7XG4gICAgICAgIH0gZWxzZSBpZiAoXG4gICAgICAgICAgaXNIYXNNYW55UmVsYXRpb25Qcm9wKHJlbGF0aW9uKSB8fFxuICAgICAgICAgIGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcChyZWxhdGlvbilcbiAgICAgICAgKSB7XG4gICAgICAgICAgLy8gLU1hbnkgUmVsYXRpb246IExvYWRlciDroZwg7LKY66asXG4gICAgICAgICAgY29uc3QgcmVsRmllbGRzID0gZmllbGRzLm1hcCgoZmllbGQpID0+XG4gICAgICAgICAgICBmaWVsZC5zcGxpdChcIi5cIikuc2xpY2UoMSkuam9pbihcIi5cIilcbiAgICAgICAgICApO1xuICAgICAgICAgIGNvbnN0IHJlbFN1YnNldFF1ZXJ5ID0gcmVsRW50aXR5LnJlc29sdmVTdWJzZXRRdWVyeShcIlwiLCByZWxGaWVsZHMpO1xuXG4gICAgICAgICAgbGV0IG1hbnlKb2luOiBTdWJzZXRRdWVyeVtcImxvYWRlcnNcIl1bbnVtYmVyXVtcIm1hbnlKb2luXCJdO1xuICAgICAgICAgIGlmIChpc0hhc01hbnlSZWxhdGlvblByb3AocmVsYXRpb24pKSB7XG4gICAgICAgICAgICBjb25zdCBmcm9tQ29sID0gcmVsYXRpb24/LmZyb21Db2x1bW4gPz8gXCJpZFwiO1xuICAgICAgICAgICAgbWFueUpvaW4gPSB7XG4gICAgICAgICAgICAgIGZyb21UYWJsZTogdGhpcy50YWJsZSxcbiAgICAgICAgICAgICAgZnJvbUNvbCxcbiAgICAgICAgICAgICAgaWRGaWVsZDogcHJlZml4ID09PSBcIlwiID8gYCR7ZnJvbUNvbH1gIDogYCR7cHJlZml4fV9fJHtmcm9tQ29sfWAsXG4gICAgICAgICAgICAgIHRvVGFibGU6IHJlbEVudGl0eS50YWJsZSxcbiAgICAgICAgICAgICAgdG9Db2w6IHJlbGF0aW9uLmpvaW5Db2x1bW4sXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH0gZWxzZSBpZiAoaXNNYW55VG9NYW55UmVsYXRpb25Qcm9wKHJlbGF0aW9uKSkge1xuICAgICAgICAgICAgbWFueUpvaW4gPSB7XG4gICAgICAgICAgICAgIGZyb21UYWJsZTogdGhpcy50YWJsZSxcbiAgICAgICAgICAgICAgZnJvbUNvbDogXCJpZFwiLFxuICAgICAgICAgICAgICBpZEZpZWxkOiBwcmVmaXggPT09IFwiXCIgPyBgaWRgIDogYCR7cHJlZml4fV9faWRgLFxuICAgICAgICAgICAgICB0aHJvdWdoOiB7XG4gICAgICAgICAgICAgICAgdGFibGU6IHJlbGF0aW9uLmpvaW5UYWJsZSxcbiAgICAgICAgICAgICAgICBmcm9tQ29sOiBgJHtpbmZsZWN0aW9uLnNpbmd1bGFyaXplKHRoaXMudGFibGUpfV9pZGAsXG4gICAgICAgICAgICAgICAgdG9Db2w6IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUocmVsRW50aXR5LnRhYmxlKX1faWRgLFxuICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICB0b1RhYmxlOiByZWxFbnRpdHkudGFibGUsXG4gICAgICAgICAgICAgIHRvQ29sOiBcImlkXCIsXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICByLmxvYWRlcnMucHVzaCh7XG4gICAgICAgICAgICBhczogZ3JvdXBLZXksXG4gICAgICAgICAgICB0YWJsZTogcmVsRW50aXR5LnRhYmxlLFxuICAgICAgICAgICAgbWFueUpvaW4sXG4gICAgICAgICAgICBvbmVKb2luczogcmVsU3Vic2V0UXVlcnkuam9pbnMsXG4gICAgICAgICAgICBzZWxlY3Q6IHJlbFN1YnNldFF1ZXJ5LnNlbGVjdCxcbiAgICAgICAgICAgIGxvYWRlcnM6IHJlbFN1YnNldFF1ZXJ5LmxvYWRlcnMsXG4gICAgICAgICAgfSk7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gcjtcbiAgICAgIH0sXG4gICAgICB7XG4gICAgICAgIHNlbGVjdDogW10sXG4gICAgICAgIHZpcnR1YWw6IFtdLFxuICAgICAgICBqb2luczogW10sXG4gICAgICAgIGxvYWRlcnM6IFtdLFxuICAgICAgfSBhcyBTdWJzZXRRdWVyeVxuICAgICk7XG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gIC8qXG4gICAgRmllbGRFeHByW10g7J2EIEVudGl0eVByb3BOb2RlW10g66GcIOuzgO2ZmFxuICAqL1xuICBmaWVsZEV4cHJzVG9Qcm9wTm9kZXMoXG4gICAgZmllbGRFeHByczogc3RyaW5nW10sXG4gICAgZW50aXR5OiBFbnRpdHkgPSB0aGlzXG4gICk6IEVudGl0eVByb3BOb2RlW10ge1xuICAgIGNvbnN0IGdyb3VwcyA9IGZpZWxkRXhwcnMucmVkdWNlKFxuICAgICAgKHJlc3VsdCwgZmllbGRFeHByKSA9PiB7XG4gICAgICAgIGxldCBrZXksIHZhbHVlLCBlbHNlRXhwcjtcbiAgICAgICAgaWYgKGZpZWxkRXhwci5pbmNsdWRlcyhcIi5cIikpIHtcbiAgICAgICAgICBba2V5LCAuLi5lbHNlRXhwcl0gPSBmaWVsZEV4cHIuc3BsaXQoXCIuXCIpO1xuICAgICAgICAgIHZhbHVlID0gZWxzZUV4cHIuam9pbihcIi5cIik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAga2V5ID0gXCJcIjtcbiAgICAgICAgICB2YWx1ZSA9IGZpZWxkRXhwcjtcbiAgICAgICAgfVxuICAgICAgICByZXN1bHRba2V5XSA9IChyZXN1bHRba2V5XSA/PyBbXSkuY29uY2F0KHZhbHVlKTtcblxuICAgICAgICByZXR1cm4gcmVzdWx0O1xuICAgICAgfSxcbiAgICAgIHt9IGFzIHtcbiAgICAgICAgW2s6IHN0cmluZ106IHN0cmluZ1tdO1xuICAgICAgfVxuICAgICk7XG5cbiAgICByZXR1cm4gT2JqZWN0LmtleXMoZ3JvdXBzKVxuICAgICAgLm1hcCgoa2V5KSA9PiB7XG4gICAgICAgIGNvbnN0IGdyb3VwID0gZ3JvdXBzW2tleV07XG5cbiAgICAgICAgLy8g7J2867CYIHByb3Ag7LKY66asXG4gICAgICAgIGlmIChrZXkgPT09IFwiXCIpIHtcbiAgICAgICAgICByZXR1cm4gZ3JvdXAubWFwKChwcm9wTmFtZSkgPT4ge1xuICAgICAgICAgICAgLy8gdXVpZCDqsJzrs4Qg7LKY66asXG4gICAgICAgICAgICBpZiAocHJvcE5hbWUgPT09IFwidXVpZFwiKSB7XG4gICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgbm9kZVR5cGU6IFwicGxhaW5cIiBhcyBjb25zdCxcbiAgICAgICAgICAgICAgICBwcm9wOiB7XG4gICAgICAgICAgICAgICAgICB0eXBlOiBcInN0cmluZ1wiLFxuICAgICAgICAgICAgICAgICAgbmFtZTogXCJ1dWlkXCIsXG4gICAgICAgICAgICAgICAgICBsZW5ndGg6IDEyOCxcbiAgICAgICAgICAgICAgICB9IGFzIFN0cmluZ1Byb3AsXG4gICAgICAgICAgICAgICAgY2hpbGRyZW46IFtdLFxuICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBjb25zdCBwcm9wID0gZW50aXR5LnByb3BzLmZpbmQoKHApID0+IHAubmFtZSA9PT0gcHJvcE5hbWUpO1xuICAgICAgICAgICAgaWYgKHByb3AgPT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICBjb25zb2xlLmxvZyh7IHByb3BOYW1lLCBncm91cHMgfSk7XG4gICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgJHtlbnRpdHkuaWR9IC0tIOyemOuqu+uQnCBGaWVsZEV4cHIgJHtwcm9wTmFtZX1gKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIG5vZGVUeXBlOiBcInBsYWluXCIgYXMgY29uc3QsXG4gICAgICAgICAgICAgIHByb3AsXG4gICAgICAgICAgICAgIGNoaWxkcmVuOiBbXSxcbiAgICAgICAgICAgIH07XG4gICAgICAgICAgfSk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyByZWxhdGlvbiBwcm9wIOyymOumrFxuICAgICAgICBjb25zdCBwcm9wID0gZW50aXR5LnByb3BzRGljdFtrZXldO1xuICAgICAgICBpZiAoIWlzUmVsYXRpb25Qcm9wKHByb3ApKSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDsnpjrqrvrkJwgRmllbGRFeHByICR7a2V5fS4ke2dyb3VwWzBdfWApO1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IHJlbEVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG5cbiAgICAgICAgLy8gcmVsYXRpb24gLU9uZSDsl5AgaWQg7ZWE65OcIO2VmOuCmOyduCDqsr3smrBcbiAgICAgICAgaWYgKGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wKHByb3ApIHx8IGlzT25lVG9PbmVSZWxhdGlvblByb3AocHJvcCkpIHtcbiAgICAgICAgICBpZiAoZ3JvdXAubGVuZ3RoID09IDEgJiYgKGdyb3VwWzBdID09PSBcImlkXCIgfHwgZ3JvdXBbMF0gPT0gXCJpZD9cIikpIHtcbiAgICAgICAgICAgIC8vIGlkIO2VmOuCmOunjCDsnojripTsp4Ag7LK07YGs7ZW07IScLCDtlZjrgpjrp4wg7J6I7Jy866m0IOyDgeychCBwcm9w7Jy866GcIGlk66W8IOumrO2EtFxuICAgICAgICAgICAgY29uc3QgaWRQcm9wID0gcmVsRW50aXR5LnByb3BzRGljdC5pZDtcbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIG5vZGVUeXBlOiBcInBsYWluXCIgYXMgY29uc3QsXG4gICAgICAgICAgICAgIHByb3A6IHtcbiAgICAgICAgICAgICAgICAuLi5pZFByb3AsXG4gICAgICAgICAgICAgICAgbmFtZToga2V5ICsgXCJfaWRcIixcbiAgICAgICAgICAgICAgICBudWxsYWJsZTogcHJvcC5udWxsYWJsZSxcbiAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgY2hpbGRyZW46IFtdLFxuICAgICAgICAgICAgfTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAvLyAtT25lIOq3uOyZuOydmCDqsr3smrAgb2JqZWN066GcIOumrO2EtFxuICAgICAgICAvLyAtTWFueeydmCDqsr3smrAgYXJyYXnroZwg66as7YS0XG4gICAgICAgIC8vIFJlY3Vyc2l2ZSDroZwg646B7IqkIOyymOumrFxuICAgICAgICBjb25zdCBjaGlsZHJlbiA9IHRoaXMuZmllbGRFeHByc1RvUHJvcE5vZGVzKGdyb3VwLCByZWxFbnRpdHkpO1xuICAgICAgICBjb25zdCBub2RlVHlwZSA9XG4gICAgICAgICAgaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocHJvcCkgfHwgaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKVxuICAgICAgICAgICAgPyAoXCJvYmplY3RcIiBhcyBjb25zdClcbiAgICAgICAgICAgIDogKFwiYXJyYXlcIiBhcyBjb25zdCk7XG5cbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBwcm9wLFxuICAgICAgICAgIGNoaWxkcmVuLFxuICAgICAgICAgIG5vZGVUeXBlLFxuICAgICAgICB9O1xuICAgICAgfSlcbiAgICAgIC5mbGF0KCk7XG4gIH1cblxuICBnZXRGaWVsZEV4cHJzKFxuICAgIHByZWZpeCA9IFwiXCIsXG4gICAgbWF4RGVwdGg6IG51bWJlciA9IDMsXG4gICAgZnJvbXM6IHN0cmluZ1tdID0gW11cbiAgKTogc3RyaW5nW10ge1xuICAgIHJldHVybiB0aGlzLnByb3BzXG4gICAgICAubWFwKChwcm9wKSA9PiB7XG4gICAgICAgIGNvbnN0IHByb3BOYW1lID0gW3ByZWZpeCwgcHJvcC5uYW1lXS5maWx0ZXIoKHYpID0+IHYgIT09IFwiXCIpLmpvaW4oXCIuXCIpO1xuICAgICAgICBpZiAocHJvcE5hbWUgPT09IHByZWZpeCkge1xuICAgICAgICAgIHJldHVybiBudWxsO1xuICAgICAgICB9XG4gICAgICAgIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICAgIGlmIChtYXhEZXB0aCA8IDApIHtcbiAgICAgICAgICAgIHJldHVybiBudWxsO1xuICAgICAgICAgIH1cbiAgICAgICAgICBpZiAoZnJvbXMuaW5jbHVkZXMocHJvcC53aXRoKSkge1xuICAgICAgICAgICAgLy8g7Jet67Cp7ZalIHJlbGF0aW9u7J24IOqyveyasCDsoJzsmbhcbiAgICAgICAgICAgIHJldHVybiBudWxsO1xuICAgICAgICAgIH1cbiAgICAgICAgICAvLyDsoJXrsKntlqUgcmVsYXRpb27snbgg6rK97JqwIHJlY3Vyc2l2ZSDsvZxcbiAgICAgICAgICBjb25zdCByZWxNZCA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG4gICAgICAgICAgcmV0dXJuIHJlbE1kLmdldEZpZWxkRXhwcnMocHJvcE5hbWUsIG1heERlcHRoIC0gMSwgW1xuICAgICAgICAgICAgLi4uZnJvbXMsXG4gICAgICAgICAgICB0aGlzLmlkLFxuICAgICAgICAgIF0pO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBwcm9wTmFtZTtcbiAgICAgIH0pXG4gICAgICAuZmxhdCgpXG4gICAgICAuZmlsdGVyKChmKSA9PiBmICE9PSBudWxsKSBhcyBzdHJpbmdbXTtcbiAgfVxuXG4gIGdldFRhYmxlQ29sdW1ucygpOiB7IG5hbWU6IHN0cmluZzsgdHlwZTogc3RyaW5nIH1bXSB7XG4gICAgcmV0dXJuIHRoaXMucHJvcHNcbiAgICAgIC5tYXAoKHByb3ApID0+IHtcbiAgICAgICAgaWYgKHByb3AudHlwZSA9PT0gXCJyZWxhdGlvblwiKSB7XG4gICAgICAgICAgaWYgKFxuICAgICAgICAgICAgcHJvcC5yZWxhdGlvblR5cGUgPT09IFwiQmVsb25nc1RvT25lXCIgfHxcbiAgICAgICAgICAgIChwcm9wLnJlbGF0aW9uVHlwZSA9PT0gXCJPbmVUb09uZVwiICYmIHByb3AuaGFzSm9pbkNvbHVtbiA9PT0gdHJ1ZSlcbiAgICAgICAgICApIHtcbiAgICAgICAgICAgIHJldHVybiB7IG5hbWU6IGAke3Byb3AubmFtZX1faWRgLCB0eXBlOiBcImludF91bnNpZ25lZFwiIH07XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJldHVybiBudWxsO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICByZXR1cm4geyBuYW1lOiBwcm9wLm5hbWUsIHR5cGU6IHByb3AudHlwZSB9O1xuICAgICAgfSlcbiAgICAgIC5maWx0ZXIobm9uTnVsbGFibGUpO1xuICB9XG5cbiAgYXN5bmMgcmVnaXN0ZXJNb2R1bGVQYXRocygpIHtcbiAgICBjb25zdCBiYXNlUGF0aCA9IGAke3RoaXMubmFtZXMucGFyZW50RnN9YDtcblxuICAgIC8vIGJhc2Utc2NoZW1lXG4gICAgRW50aXR5TWFuYWdlci5zZXRNb2R1bGVQYXRoKGAke3RoaXMuaWR9QmFzZVNjaGVtYWAsIGBzb25hbXUuZ2VuZXJhdGVkYCk7XG5cbiAgICAvLyBzdWJzZXRcbiAgICBpZiAoT2JqZWN0LmtleXModGhpcy5zdWJzZXRzKS5sZW5ndGggPiAwKSB7XG4gICAgICBFbnRpdHlNYW5hZ2VyLnNldE1vZHVsZVBhdGgoYCR7dGhpcy5pZH1TdWJzZXRLZXlgLCBgc29uYW11LmdlbmVyYXRlZGApO1xuICAgICAgRW50aXR5TWFuYWdlci5zZXRNb2R1bGVQYXRoKFxuICAgICAgICBgJHt0aGlzLmlkfVN1YnNldE1hcHBpbmdgLFxuICAgICAgICBgc29uYW11LmdlbmVyYXRlZGBcbiAgICAgICk7XG4gICAgICBPYmplY3Qua2V5cyh0aGlzLnN1YnNldHMpLm1hcCgoc3Vic2V0S2V5KSA9PiB7XG4gICAgICAgIEVudGl0eU1hbmFnZXIuc2V0TW9kdWxlUGF0aChcbiAgICAgICAgICBgJHt0aGlzLmlkfVN1YnNldCR7c3Vic2V0S2V5LnRvVXBwZXJDYXNlKCl9YCxcbiAgICAgICAgICBgc29uYW11LmdlbmVyYXRlZGBcbiAgICAgICAgKTtcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIC8vIGVudW1zXG4gICAgT2JqZWN0LmtleXModGhpcy5lbnVtTGFiZWxzKS5tYXAoKGVudW1JZCkgPT4ge1xuICAgICAgRW50aXR5TWFuYWdlci5zZXRNb2R1bGVQYXRoKGVudW1JZCwgYHNvbmFtdS5nZW5lcmF0ZWRgKTtcbiAgICB9KTtcblxuICAgIC8vIHR5cGVzXG4gICAgY29uc3QgdHlwZXNNb2R1bGVQYXRoID0gYCR7YmFzZVBhdGh9LyR7dGhpcy5uYW1lcy5wYXJlbnRGc30udHlwZXNgO1xuICAgIGNvbnN0IHR5cGVzRmlsZURpc3RQYXRoID0gcGF0aC5qb2luKFxuICAgICAgU29uYW11LmFwaVJvb3RQYXRoLFxuICAgICAgYGRpc3QvYXBwbGljYXRpb24vJHt0eXBlc01vZHVsZVBhdGh9LmpzYFxuICAgICk7XG5cbiAgICBpZiAoYXdhaXQgZXhpc3RzKHR5cGVzRmlsZURpc3RQYXRoKSkge1xuICAgICAgY29uc3QgaW1wb3J0VXJsID0gY3JlYXRlSW1wb3J0VXJsKHR5cGVzRmlsZURpc3RQYXRoKTtcbiAgICAgIGNvbnN0IHQgPSBhd2FpdCBpbXBvcnQoaW1wb3J0VXJsKTtcbiAgICAgIHRoaXMudHlwZXMgPSBPYmplY3Qua2V5cyh0KS5yZWR1Y2UoKHJlc3VsdCwga2V5KSA9PiB7XG4gICAgICAgIEVudGl0eU1hbmFnZXIuc2V0TW9kdWxlUGF0aChrZXksIHR5cGVzTW9kdWxlUGF0aCk7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgLi4ucmVzdWx0LFxuICAgICAgICAgIFtrZXldOiB0W2tleV0sXG4gICAgICAgIH07XG4gICAgICB9LCB7fSk7XG4gICAgfVxuICB9XG5cbiAgcmVnaXN0ZXJUYWJsZVNwZWNzKCk6IHZvaWQge1xuICAgIGNvbnN0IHVuaXF1ZUluZGV4ZXMgPSB0aGlzLmluZGV4ZXMuZmlsdGVyKChpZHgpID0+IGlkeC50eXBlID09PSBcInVuaXF1ZVwiKTtcblxuICAgIEVudGl0eU1hbmFnZXIuc2V0VGFibGVTcGVjKHtcbiAgICAgIG5hbWU6IHRoaXMudGFibGUsXG4gICAgICB1bmlxdWVJbmRleGVzLFxuICAgIH0pO1xuICB9XG5cbiAgdG9Kc29uKCk6IEVudGl0eUpzb24ge1xuICAgIHJldHVybiB7XG4gICAgICBpZDogdGhpcy5pZCxcbiAgICAgIHBhcmVudElkOiB0aGlzLnBhcmVudElkLFxuICAgICAgdGFibGU6IHRoaXMudGFibGUsXG4gICAgICB0aXRsZTogdGhpcy50aXRsZSxcbiAgICAgIHByb3BzOiB0aGlzLnByb3BzLFxuICAgICAgaW5kZXhlczogdGhpcy5pbmRleGVzLFxuICAgICAgc3Vic2V0czogdGhpcy5zdWJzZXRzLFxuICAgICAgZW51bXM6IHRoaXMuZW51bUxhYmVscyxcbiAgICB9O1xuICB9XG5cbiAgYXN5bmMgc2F2ZSgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyBzb3J0OiBzdWJzZXRzXG4gICAgY29uc3Qgc3Vic2V0Um93cyA9IHRoaXMuZ2V0U3Vic2V0Um93cygpO1xuICAgIHRoaXMuc3Vic2V0cyA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgIE9iamVjdC5lbnRyaWVzKHRoaXMuc3Vic2V0cykubWFwKChbc3Vic2V0S2V5XSkgPT4ge1xuICAgICAgICByZXR1cm4gW1xuICAgICAgICAgIHN1YnNldEtleSxcbiAgICAgICAgICB0aGlzLnN1YnNldFJvd3NUb1N1YnNldEZpZWxkcyhzdWJzZXRSb3dzLCBzdWJzZXRLZXkpLFxuICAgICAgICBdO1xuICAgICAgfSlcbiAgICApO1xuXG4gICAgLy8gc2F2ZVxuICAgIGNvbnN0IGpzb25QYXRoID0gcGF0aC5qb2luKFxuICAgICAgU29uYW11LmFwaVJvb3RQYXRoLFxuICAgICAgYHNyYy9hcHBsaWNhdGlvbi8ke3RoaXMubmFtZXMucGFyZW50RnN9LyR7dGhpcy5uYW1lcy5mc30uZW50aXR5Lmpzb25gXG4gICAgKTtcbiAgICBjb25zdCBqc29uID0gdGhpcy50b0pzb24oKTtcbiAgICBhd2FpdCB3cml0ZUZpbGUoXG4gICAgICBqc29uUGF0aCxcbiAgICAgIGF3YWl0IHByZXR0aWVyLmZvcm1hdChKU09OLnN0cmluZ2lmeShqc29uKSwge1xuICAgICAgICBwYXJzZXI6IFwianNvblwiLFxuICAgICAgfSlcbiAgICApO1xuXG4gICAgLy8gcmVsb2FkXG4gICAgYXdhaXQgRW50aXR5TWFuYWdlci5yZWdpc3Rlcihqc29uKTtcbiAgfVxuXG4gIGdldFN1YnNldFJvd3MoXG4gICAgX3N1YnNldHM/OiB7IFtrZXk6IHN0cmluZ106IHN0cmluZ1tdIH0sXG4gICAgcHJlZml4ZXM6IHN0cmluZ1tdID0gW11cbiAgKTogRW50aXR5U3Vic2V0Um93W10ge1xuICAgIGlmIChwcmVmaXhlcy5sZW5ndGggPiAxMCkge1xuICAgICAgcmV0dXJuIFtdO1xuICAgIH1cblxuICAgIGNvbnN0IHN1YnNldHMgPSBfc3Vic2V0cyA/PyB0aGlzLnN1YnNldHM7XG4gICAgY29uc3Qgc3Vic2V0S2V5cyA9IE9iamVjdC5rZXlzKHN1YnNldHMpO1xuICAgIGNvbnN0IGFsbEZpZWxkcyA9IF8udW5pcShzdWJzZXRLZXlzLm1hcCgoa2V5KSA9PiBzdWJzZXRzW2tleV0pLmZsYXQoKSk7XG5cbiAgICByZXR1cm4gdGhpcy5wcm9wcy5tYXAoKHByb3ApID0+IHtcbiAgICAgIGlmIChcbiAgICAgICAgcHJvcC50eXBlID09PSBcInJlbGF0aW9uXCIgJiZcbiAgICAgICAgYWxsRmllbGRzLmZpbmQoKGYpID0+XG4gICAgICAgICAgZi5zdGFydHNXaXRoKFsuLi5wcmVmaXhlcywgcHJvcC5uYW1lXS5qb2luKFwiLlwiKSArIFwiLlwiKVxuICAgICAgICApXG4gICAgICApIHtcbiAgICAgICAgY29uc3QgcmVsRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocHJvcC53aXRoKTtcbiAgICAgICAgY29uc3QgY2hpbGRyZW4gPSByZWxFbnRpdHkuZ2V0U3Vic2V0Um93cyhzdWJzZXRzLCBbXG4gICAgICAgICAgLi4ucHJlZml4ZXMsXG4gICAgICAgICAgYCR7cHJvcC5uYW1lfWAsXG4gICAgICAgIF0pO1xuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgZmllbGQ6IHByb3AubmFtZSxcbiAgICAgICAgICBjaGlsZHJlbixcbiAgICAgICAgICByZWxhdGlvbkVudGl0eTogcHJvcC53aXRoLFxuICAgICAgICAgIHByZWZpeGVzLFxuICAgICAgICAgIGlzT3BlbjogY2hpbGRyZW4ubGVuZ3RoID4gMCxcbiAgICAgICAgICBoYXM6IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgICAgICAgIHN1YnNldEtleXMubWFwKChzdWJzZXRLZXkpID0+IHtcbiAgICAgICAgICAgICAgcmV0dXJuIFtcbiAgICAgICAgICAgICAgICBzdWJzZXRLZXksXG4gICAgICAgICAgICAgICAgY2hpbGRyZW4uZXZlcnkoKGNoaWxkKSA9PiBjaGlsZC5oYXNbc3Vic2V0S2V5XSA9PT0gdHJ1ZSksXG4gICAgICAgICAgICAgIF07XG4gICAgICAgICAgICB9KVxuICAgICAgICAgICksXG4gICAgICAgIH07XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB7XG4gICAgICAgIGZpZWxkOiBwcm9wLm5hbWUsXG4gICAgICAgIGNoaWxkcmVuOiBbXSxcbiAgICAgICAgcmVsYXRpb25FbnRpdHk6IHByb3AudHlwZSA9PT0gXCJyZWxhdGlvblwiID8gcHJvcC53aXRoIDogdW5kZWZpbmVkLFxuICAgICAgICBwcmVmaXhlcyxcbiAgICAgICAgaGFzOiBPYmplY3QuZnJvbUVudHJpZXMoXG4gICAgICAgICAgc3Vic2V0S2V5cy5tYXAoKHN1YnNldEtleSkgPT4ge1xuICAgICAgICAgICAgY29uc3Qgc3Vic2V0RmllbGRzID0gc3Vic2V0c1tzdWJzZXRLZXldO1xuICAgICAgICAgICAgY29uc3QgaGFzID0gc3Vic2V0RmllbGRzLnNvbWUoKGYpID0+IHtcbiAgICAgICAgICAgICAgY29uc3QgZmllbGQgPSBbLi4ucHJlZml4ZXMsIHByb3AubmFtZV0uam9pbihcIi5cIik7XG4gICAgICAgICAgICAgIHJldHVybiBmID09PSBmaWVsZCB8fCBmLnN0YXJ0c1dpdGgoZmllbGQgKyBcIi5cIik7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIHJldHVybiBbc3Vic2V0S2V5LCBoYXNdO1xuICAgICAgICAgIH0pXG4gICAgICAgICksXG4gICAgICB9O1xuICAgIH0pO1xuICB9XG5cbiAgc3Vic2V0Um93c1RvU3Vic2V0RmllbGRzKFxuICAgIHN1YnNldFJvd3M6IEVudGl0eVN1YnNldFJvd1tdLFxuICAgIHN1YnNldEtleTogc3RyaW5nXG4gICk6IHN0cmluZ1tdIHtcbiAgICByZXR1cm4gc3Vic2V0Um93c1xuICAgICAgLm1hcCgoc3Vic2V0Um93KSA9PiB7XG4gICAgICAgIGlmIChzdWJzZXRSb3cuY2hpbGRyZW4ubGVuZ3RoID4gMCkge1xuICAgICAgICAgIHJldHVybiB0aGlzLnN1YnNldFJvd3NUb1N1YnNldEZpZWxkcyhzdWJzZXRSb3cuY2hpbGRyZW4sIHN1YnNldEtleSk7XG4gICAgICAgIH0gZWxzZSBpZiAoc3Vic2V0Um93Lmhhc1tzdWJzZXRLZXldKSB7XG4gICAgICAgICAgcmV0dXJuIHN1YnNldFJvdy5wcmVmaXhlcy5jb25jYXQoc3Vic2V0Um93LmZpZWxkKS5qb2luKFwiLlwiKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgfVxuICAgICAgfSlcbiAgICAgIC5maWx0ZXIobm9uTnVsbGFibGUpXG4gICAgICAuZmxhdCgpO1xuICB9XG5cbiAgYXN5bmMgY3JlYXRlUHJvcChwcm9wOiBFbnRpdHlQcm9wLCBhdD86IG51bWJlcik6IFByb21pc2U8dm9pZD4ge1xuICAgIGlmICghYXQpIHtcbiAgICAgIHRoaXMucHJvcHMucHVzaChwcm9wKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5wcm9wcy5zcGxpY2UoYXQsIDAsIHByb3ApO1xuICAgIH1cbiAgICBhd2FpdCB0aGlzLnNhdmUoKTtcbiAgfVxuXG4gIGFuYWx5emVTdWJzZXRGaWVsZChzdWJzZXRGaWVsZDogc3RyaW5nKToge1xuICAgIGVudGl0eUlkOiBzdHJpbmc7XG4gICAgcHJvcE5hbWU6IHN0cmluZztcbiAgfVtdIHtcbiAgICBjb25zdCBhcnIgPSBzdWJzZXRGaWVsZC5zcGxpdChcIi5cIik7XG5cbiAgICBsZXQgZW50aXR5SWQgPSB0aGlzLmlkO1xuICAgIGNvbnN0IHJlc3VsdDoge1xuICAgICAgZW50aXR5SWQ6IHN0cmluZztcbiAgICAgIHByb3BOYW1lOiBzdHJpbmc7XG4gICAgfVtdID0gW107XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBhcnIubGVuZ3RoOyBpKyspIHtcbiAgICAgIGNvbnN0IHByb3BOYW1lID0gYXJyW2ldO1xuICAgICAgcmVzdWx0LnB1c2goe1xuICAgICAgICBlbnRpdHlJZCxcbiAgICAgICAgcHJvcE5hbWUsXG4gICAgICB9KTtcblxuICAgICAgY29uc3QgcHJvcCA9IEVudGl0eU1hbmFnZXIuZ2V0KGVudGl0eUlkKS5wcm9wcy5maW5kKFxuICAgICAgICAocCkgPT4gcC5uYW1lID09PSBwcm9wTmFtZVxuICAgICAgKTtcbiAgICAgIGlmICghcHJvcCkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7ZW50aXR5SWR97J2YIOyemOuqu+uQnCDshJzruIzshYvtgqQgJHtzdWJzZXRGaWVsZH1gKTtcbiAgICAgIH1cbiAgICAgIGlmIChpc1JlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICBlbnRpdHlJZCA9IHByb3Aud2l0aDtcbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gIGFzeW5jIG1vZGlmeVByb3AobmV3UHJvcDogRW50aXR5UHJvcCwgYXQ6IG51bWJlcik6IFByb21pc2U8dm9pZD4ge1xuICAgIC8vIOydtOyghCDtlITroa0g7J2066aEIOyggOyepVxuICAgIGNvbnN0IG9sZE5hbWUgPSB0aGlzLnByb3BzW2F0XS5uYW1lO1xuXG4gICAgLy8g7KCA7J6l7ZWgIOyXlO2LsO2LsFxuICAgIGNvbnN0IGVudGl0aWVzOiBFbnRpdHlbXSA9IFt0aGlzXTtcblxuICAgIC8vIOydtOumhOydtCDrsJTrgJAg6rK97JqwXG4gICAgaWYgKG9sZE5hbWUgIT09IG5ld1Byb3AubmFtZSkge1xuICAgICAgLy8g7KCE7LK0IOyXlO2LsO2LsOyXkOyEnCDtmITsnqwg7IiY7KCV65CcIO2UhOuhreydhCDssLjsobDtlZjqs6Ag7J6I64qUIOuqqOuToCDshJzruIzshYvtlYTrk5wg7LC+7JWE7IScIOyImOyglVxuICAgICAgY29uc3QgYWxsRW50aXR5SWRzID0gRW50aXR5TWFuYWdlci5nZXRBbGxJZHMoKTtcbiAgICAgIGZvciAoY29uc3QgcmVsRW50aXR5SWQgb2YgYWxsRW50aXR5SWRzKSB7XG4gICAgICAgIGNvbnN0IHJlbEVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KHJlbEVudGl0eUlkKTtcbiAgICAgICAgY29uc3QgcmVsRW50aXR5U3Vic2V0S2V5cyA9IE9iamVjdC5rZXlzKHJlbEVudGl0eS5zdWJzZXRzKTtcbiAgICAgICAgZm9yIChjb25zdCBzdWJzZXRLZXkgb2YgcmVsRW50aXR5U3Vic2V0S2V5cykge1xuICAgICAgICAgIGNvbnN0IHN1YnNldCA9IHJlbEVudGl0eS5zdWJzZXRzW3N1YnNldEtleV07XG5cbiAgICAgICAgICAvLyDshJzruIzshYsg7ZWE65Oc66W8IOyInO2ajO2VmOupsCwg7JeU7Yuw7YuwLe2UhOuhrSDri6jsnITroZwg67aE7ISd7ZWcIO2bhCDtmITsnqwg7JeU7Yuw7YuwLe2UhOuhreqzvCDsnbzsuZjtlZjripQg6rK97JqwIOyImOyglSDsspjrpqxcbiAgICAgICAgICBjb25zdCBtb2RpZmllZFN1YnNldEZpZWxkcyA9IHN1YnNldC5tYXAoKHN1YnNldEZpZWxkKSA9PiB7XG4gICAgICAgICAgICBjb25zdCBhbmFseXplZCA9IHJlbEVudGl0eS5hbmFseXplU3Vic2V0RmllbGQoc3Vic2V0RmllbGQpO1xuICAgICAgICAgICAgY29uc3QgbW9kaWZpZWQgPSBhbmFseXplZC5tYXAoKGEpID0+XG4gICAgICAgICAgICAgIGEucHJvcE5hbWUgPT09IG9sZE5hbWUgJiYgYS5lbnRpdHlJZCA9PT0gdGhpcy5pZFxuICAgICAgICAgICAgICAgID8ge1xuICAgICAgICAgICAgICAgICAgICAuLi5hLFxuICAgICAgICAgICAgICAgICAgICBwcm9wTmFtZTogbmV3UHJvcC5uYW1lLFxuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIDogYVxuICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIC8vIOu2hOyEne2VnCDtlYTrk5zrpbwg64uk7IucIOyEnOu4jOyFiyDtlYTrk5zroZwg67O16rWsXG4gICAgICAgICAgICByZXR1cm4gbW9kaWZpZWQubWFwKChhKSA9PiBhLnByb3BOYW1lKS5qb2luKFwiLlwiKTtcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIGlmIChzdWJzZXQuam9pbihcIixcIikgIT09IG1vZGlmaWVkU3Vic2V0RmllbGRzLmpvaW4oXCIsXCIpKSB7XG4gICAgICAgICAgICByZWxFbnRpdHkuc3Vic2V0c1tzdWJzZXRLZXldID0gbW9kaWZpZWRTdWJzZXRGaWVsZHM7XG4gICAgICAgICAgICBlbnRpdGllcy5wdXNoKHJlbEVudGl0eSk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7ZSE66GtIOyImOyglVxuICAgIHRoaXMucHJvcHNbYXRdID0gbmV3UHJvcDtcblxuICAgIGF3YWl0IFByb21pc2UuYWxsKGVudGl0aWVzLm1hcChhc3luYyAoZW50aXR5KSA9PiBlbnRpdHkuc2F2ZSgpKSk7XG4gIH1cblxuICBhc3luYyBkZWxQcm9wKGF0OiBudW1iZXIpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyDsnbTsoIQg7ZSE66GtIOydtOumhCDsoIDsnqVcbiAgICBjb25zdCBvbGROYW1lID0gdGhpcy5wcm9wc1thdF0ubmFtZTtcblxuICAgIC8vIOyggOyepe2VoCDsl5Tti7Dti7BcbiAgICBjb25zdCBlbnRpdGllczogRW50aXR5W10gPSBbdGhpc107XG5cbiAgICAvLyDsoITssrQg7JeU7Yuw7Yuw7JeQ7IScIO2YhOyerCDsgq3soJzrkJwg7ZSE66Gt7J2EIOywuOyhsO2VmOqzoCDsnojripQg66qo65OgIOyEnOu4jOyFi+2VhOuTnCDssL7slYTshJwg7KCc7Jm4XG4gICAgY29uc3QgYWxsRW50aXR5SWRzID0gRW50aXR5TWFuYWdlci5nZXRBbGxJZHMoKTtcbiAgICBmb3IgKGNvbnN0IHJlbEVudGl0eUlkIG9mIGFsbEVudGl0eUlkcykge1xuICAgICAgY29uc3QgcmVsRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocmVsRW50aXR5SWQpO1xuICAgICAgY29uc3QgcmVsRW50aXR5U3Vic2V0S2V5cyA9IE9iamVjdC5rZXlzKHJlbEVudGl0eS5zdWJzZXRzKTtcbiAgICAgIGZvciAoY29uc3Qgc3Vic2V0S2V5IG9mIHJlbEVudGl0eVN1YnNldEtleXMpIHtcbiAgICAgICAgY29uc3Qgc3Vic2V0ID0gcmVsRW50aXR5LnN1YnNldHNbc3Vic2V0S2V5XTtcbiAgICAgICAgLy8g7ISc67iM7IWLIO2VhOuTnOulvCDsiJztmoztlZjrqbAsIOyXlO2LsO2LsC3tlITroa0g64uo7JyE66GcIOu2hOyEne2VnCDtm4Qg7ZiE7J6sIOyXlO2LsO2LsC3tlITroa3qs7wg7J287LmY7ZWY64qUIOqyveyasCDsnbTtm4TsnZgg7ZWE65Oc66W8IOygnOyZuFxuICAgICAgICBjb25zdCBtb2RpZmllZFN1YnNldEZpZWxkcyA9IHN1YnNldFxuICAgICAgICAgIC5tYXAoKHN1YnNldEZpZWxkKSA9PiB7XG4gICAgICAgICAgICBjb25zdCBhbmFseXplZCA9IHJlbEVudGl0eS5hbmFseXplU3Vic2V0RmllbGQoc3Vic2V0RmllbGQpO1xuICAgICAgICAgICAgaWYgKFxuICAgICAgICAgICAgICBhbmFseXplZC5maW5kKFxuICAgICAgICAgICAgICAgIChhKSA9PiBhLnByb3BOYW1lID09PSBvbGROYW1lICYmIGEuZW50aXR5SWQgPT09IHRoaXMuaWRcbiAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgIHJldHVybiBudWxsO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgcmV0dXJuIHN1YnNldEZpZWxkO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0pXG4gICAgICAgICAgLmZpbHRlcihub25OdWxsYWJsZSk7XG5cbiAgICAgICAgaWYgKHN1YnNldC5qb2luKFwiLFwiKSAhPT0gbW9kaWZpZWRTdWJzZXRGaWVsZHMuam9pbihcIixcIikpIHtcbiAgICAgICAgICByZWxFbnRpdHkuc3Vic2V0c1tzdWJzZXRLZXldID0gbW9kaWZpZWRTdWJzZXRGaWVsZHM7XG4gICAgICAgICAgZW50aXRpZXMucHVzaChyZWxFbnRpdHkpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8g7ZiE7J6sIOyXlO2LsO2LsOydmCDsnbjrjbHsiqTsl5DshJwg7KCc7Jm4XG4gICAgRW50aXR5TWFuYWdlci5nZXQodGhpcy5pZCkuaW5kZXhlcy5tYXAoKGluZGV4KSA9PiB7XG4gICAgICBpbmRleC5jb2x1bW5zID0gaW5kZXguY29sdW1ucy5maWx0ZXIoKGNvbCkgPT4gY29sICE9PSBvbGROYW1lKTtcbiAgICB9KTtcblxuICAgIC8vIO2UhOuhrSDsgq3soJxcbiAgICB0aGlzLnByb3BzLnNwbGljZShhdCwgMSk7XG5cbiAgICBhd2FpdCBQcm9taXNlLmFsbChlbnRpdGllcy5tYXAoYXN5bmMgKGVudGl0eSkgPT4gZW50aXR5LnNhdmUoKSkpO1xuICB9XG5cbiAgZ2V0RW50aXR5SWRGcm9tU3Vic2V0RmllbGQoc3Vic2V0RmllbGQ6IHN0cmluZyk6IHN0cmluZyB7XG4gICAgaWYgKHN1YnNldEZpZWxkLmluY2x1ZGVzKFwiLlwiKSA9PT0gZmFsc2UpIHtcbiAgICAgIHJldHVybiB0aGlzLmlkO1xuICAgIH1cblxuICAgIC8vIOyEnOu4jOyFiyDtlYTrk5zsnZgg66eI7KeA66eJ7J2AIO2UhOuhreydtOuvgOuhnCDsoJzsmbhcbiAgICBjb25zdCBhcnIgPSBzdWJzZXRGaWVsZC5zcGxpdChcIi5cIikuc2xpY2UoMCwgLTEpO1xuXG4gICAgLy8g7ISc67iM7IWLIO2VhOuTnOulvCDrgrTroKTqsIDrqbTshJwg66eI7KeA66eJ7Jy866GcIHJlbGF0aW9u65CcIOyXlO2LsO2LsOulvCDssL7snYxcbiAgICBjb25zdCBsYXN0RW50aXR5SWQgPSBhcnIucmVkdWNlKChlbnRpdHlJZCwgZmllbGQpID0+IHtcbiAgICAgIGNvbnN0IHJlbFByb3AgPSBFbnRpdHlNYW5hZ2VyLmdldChlbnRpdHlJZCkucHJvcHMuZmluZChcbiAgICAgICAgKHApID0+IHAubmFtZSA9PT0gZmllbGRcbiAgICAgICk7XG4gICAgICBpZiAoIXJlbFByb3AgfHwgcmVsUHJvcC50eXBlICE9PSBcInJlbGF0aW9uXCIpIHtcbiAgICAgICAgY29uc29sZS5kZWJ1Zyh7IGFyciwgdGhpc0lkOiB0aGlzLmlkLCBlbnRpdHlJZCwgZmllbGQgfSk7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihg7J6Y66q765CcIOyEnOu4jOyFi+2CpCAke3N1YnNldEZpZWxkfWApO1xuICAgICAgfVxuICAgICAgcmV0dXJuIHJlbFByb3Aud2l0aDtcbiAgICB9LCB0aGlzLmlkKTtcbiAgICByZXR1cm4gbGFzdEVudGl0eUlkO1xuICB9XG5cbiAgYXN5bmMgbW92ZVByb3AoYXQ6IG51bWJlciwgdG86IG51bWJlcik6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IHByb3AgPSB0aGlzLnByb3BzW2F0XTtcbiAgICBjb25zdCBuZXdQcm9wcyA9IFsuLi50aGlzLnByb3BzXTtcbiAgICBuZXdQcm9wcy5zcGxpY2UodG8sIDAsIHByb3ApO1xuICAgIG5ld1Byb3BzLnNwbGljZShhdCA8IHRvID8gYXQgOiBhdCArIDEsIDEpO1xuICAgIHRoaXMucHJvcHMgPSBuZXdQcm9wcztcblxuICAgIGF3YWl0IHRoaXMuc2F2ZSgpO1xuICB9XG59XG4iXSwibmFtZXMiOlsiXyIsImNyZWF0ZUltcG9ydFVybCIsIkVudGl0eU1hbmFnZXIiLCJpc1JlbGF0aW9uUHJvcCIsImlzVmlydHVhbFByb3AiLCJpc0JlbG9uZ3NUb09uZVJlbGF0aW9uUHJvcCIsImlzT25lVG9PbmVSZWxhdGlvblByb3AiLCJpc0hhc01hbnlSZWxhdGlvblByb3AiLCJpc01hbnlUb01hbnlSZWxhdGlvblByb3AiLCJpc0VudW1Qcm9wIiwiaW5mbGVjdGlvbiIsInBhdGgiLCJ3cml0ZUZpbGUiLCJ6IiwiU29uYW11IiwicHJldHRpZXIiLCJub25OdWxsYWJsZSIsImV4aXN0cyIsIkVudGl0eSIsImlkIiwicGFyZW50SWQiLCJ0YWJsZSIsInRpdGxlIiwibmFtZXMiLCJwcm9wcyIsInByb3BzRGljdCIsInJlbGF0aW9ucyIsImluZGV4ZXMiLCJzdWJzZXRzIiwidHlwZXMiLCJlbnVtcyIsImVudW1MYWJlbHMiLCJ1bmRlcnNjb3JlIiwicGx1cmFsaXplIiwibWFwIiwicHJvcCIsImluY2x1ZGVzIiwicmVwbGFjZSIsInJlZHVjZSIsInJlc3VsdCIsIm5hbWUiLCJmaWx0ZXIiLCJPYmplY3QiLCJmcm9tRW50cmllcyIsImVudHJpZXMiLCJrZXkiLCJlbnVtTGFiZWwiLCJlbnVtIiwia2V5cyIsInBhcmVudEZzIiwiZGFzaGVyaXplIiwidG9Mb3dlckNhc2UiLCJmcyIsIm1vZHVsZSIsImdldFN1YnNldFF1ZXJ5Iiwic3Vic2V0S2V5Iiwic3Vic2V0IiwicmVzb2x2ZVN1YnNldFF1ZXJ5IiwicHJlZml4IiwiZmllbGRzIiwiaXNBbHJlYWR5T3V0ZXJKb2luZWQiLCJzdWJzZXRHcm91cCIsImdyb3VwQnkiLCJmaWVsZCIsInJlbCIsInNwbGl0IiwiciIsImdyb3VwS2V5IiwicmVhbEZpZWxkcyIsInZpcnR1YWxGaWVsZHMiLCJzZWxlY3QiLCJjb25jYXQiLCJ2aXJ0dWFsIiwicmVsYXRpb24iLCJ1bmRlZmluZWQiLCJFcnJvciIsInJlbEVudGl0eSIsImdldCIsIndpdGgiLCJyZWxGaWVsZHMiLCJzbGljZSIsImpvaW4iLCJsZW5ndGgiLCJpbm5lck9yT3V0ZXIiLCJoYXNKb2luQ29sdW1uIiwibnVsbGFibGUiLCJyZWxTdWJzZXRRdWVyeSIsImpvaW5BcyIsImZyb21UYWJsZSIsImpvaW5DbGF1c2UiLCJjdXN0b21Kb2luQ2xhdXNlIiwiY3VzdG9tIiwiZnJvbSIsInRvIiwiam9pbnMiLCJwdXNoIiwiYXMiLCJsb2FkZXJzIiwiY29udmVydGVkTG9hZGVycyIsImxvYWRlciIsIm5ld0FzIiwibWFueUpvaW4iLCJvbmVKb2lucyIsImZyb21Db2wiLCJmcm9tQ29sdW1uIiwiaWRGaWVsZCIsInRvVGFibGUiLCJ0b0NvbCIsImpvaW5Db2x1bW4iLCJ0aHJvdWdoIiwiam9pblRhYmxlIiwic2luZ3VsYXJpemUiLCJmaWVsZEV4cHJzVG9Qcm9wTm9kZXMiLCJmaWVsZEV4cHJzIiwiZW50aXR5IiwiZ3JvdXBzIiwiZmllbGRFeHByIiwidmFsdWUiLCJlbHNlRXhwciIsImdyb3VwIiwicHJvcE5hbWUiLCJub2RlVHlwZSIsInR5cGUiLCJjaGlsZHJlbiIsImZpbmQiLCJwIiwiY29uc29sZSIsImxvZyIsImlkUHJvcCIsImZsYXQiLCJnZXRGaWVsZEV4cHJzIiwibWF4RGVwdGgiLCJmcm9tcyIsInYiLCJyZWxNZCIsImYiLCJnZXRUYWJsZUNvbHVtbnMiLCJyZWxhdGlvblR5cGUiLCJyZWdpc3Rlck1vZHVsZVBhdGhzIiwiYmFzZVBhdGgiLCJzZXRNb2R1bGVQYXRoIiwidG9VcHBlckNhc2UiLCJlbnVtSWQiLCJ0eXBlc01vZHVsZVBhdGgiLCJ0eXBlc0ZpbGVEaXN0UGF0aCIsImFwaVJvb3RQYXRoIiwiaW1wb3J0VXJsIiwidCIsInJlZ2lzdGVyVGFibGVTcGVjcyIsInVuaXF1ZUluZGV4ZXMiLCJpZHgiLCJzZXRUYWJsZVNwZWMiLCJ0b0pzb24iLCJzYXZlIiwic3Vic2V0Um93cyIsImdldFN1YnNldFJvd3MiLCJzdWJzZXRSb3dzVG9TdWJzZXRGaWVsZHMiLCJqc29uUGF0aCIsImpzb24iLCJmb3JtYXQiLCJKU09OIiwic3RyaW5naWZ5IiwicGFyc2VyIiwicmVnaXN0ZXIiLCJfc3Vic2V0cyIsInByZWZpeGVzIiwic3Vic2V0S2V5cyIsImFsbEZpZWxkcyIsInVuaXEiLCJzdGFydHNXaXRoIiwicmVsYXRpb25FbnRpdHkiLCJpc09wZW4iLCJoYXMiLCJldmVyeSIsImNoaWxkIiwic3Vic2V0RmllbGRzIiwic29tZSIsInN1YnNldFJvdyIsImNyZWF0ZVByb3AiLCJhdCIsInNwbGljZSIsImFuYWx5emVTdWJzZXRGaWVsZCIsInN1YnNldEZpZWxkIiwiYXJyIiwiZW50aXR5SWQiLCJpIiwibW9kaWZ5UHJvcCIsIm5ld1Byb3AiLCJvbGROYW1lIiwiZW50aXRpZXMiLCJhbGxFbnRpdHlJZHMiLCJnZXRBbGxJZHMiLCJyZWxFbnRpdHlJZCIsInJlbEVudGl0eVN1YnNldEtleXMiLCJtb2RpZmllZFN1YnNldEZpZWxkcyIsImFuYWx5emVkIiwibW9kaWZpZWQiLCJhIiwiUHJvbWlzZSIsImFsbCIsImRlbFByb3AiLCJpbmRleCIsImNvbHVtbnMiLCJjb2wiLCJnZXRFbnRpdHlJZEZyb21TdWJzZXRGaWVsZCIsImxhc3RFbnRpdHlJZCIsInJlbFByb3AiLCJkZWJ1ZyIsInRoaXNJZCIsIm1vdmVQcm9wIiwibmV3UHJvcHMiXSwibWFwcGluZ3MiOiJBQUFBLFlBQVlBLE9BQU8sWUFBWTtBQUMvQixTQUFTQyxlQUFlLFFBQVEsd0JBQXFCO0FBQ3JELFNBQVNDLGlCQUFpQkEsYUFBYSxRQUFRLHNCQUFtQjtBQUNsRSxTQUdFQyxjQUFjLEVBRWRDLGFBQWEsRUFDYkMsMEJBQTBCLEVBQzFCQyxzQkFBc0IsRUFDdEJDLHFCQUFxQixFQUNyQkMsd0JBQXdCLEVBRXhCQyxVQUFVLFFBS0wsb0JBQWlCO0FBQ3hCLE9BQU9DLGdCQUFnQixhQUFhO0FBQ3BDLE9BQU9DLFVBQVUsT0FBTztBQUN4QixTQUFTQyxTQUFTLFFBQVEsbUJBQWM7QUFDeEMsU0FBU0MsQ0FBQyxRQUFRLE1BQU07QUFDeEIsU0FBU0MsTUFBTSxRQUFRLG1CQUFnQjtBQUN2QyxPQUFPQyxjQUFjLFdBQVc7QUFDaEMsU0FBU0MsV0FBVyxRQUFRLG9CQUFpQjtBQUM3QyxTQUFTQyxNQUFNLFFBQVEsdUJBQW9CO0FBRTNDLE9BQU8sTUFBTUM7SUFDWEMsR0FBVztJ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
|