sonamu 0.5.7 → 0.7.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/.swcrc.project-default +18 -0
- package/bin/cli.js +24 -0
- package/dist/ai/agents/agent.d.ts +11 -0
- package/dist/ai/agents/agent.d.ts.map +1 -0
- package/dist/ai/agents/agent.js +65 -0
- package/dist/ai/agents/index.d.ts +3 -0
- package/dist/ai/agents/index.d.ts.map +1 -0
- package/dist/ai/agents/index.js +4 -0
- package/dist/ai/agents/types.d.ts +43 -0
- package/dist/ai/agents/types.d.ts.map +1 -0
- package/dist/ai/agents/types.js +3 -0
- package/dist/ai/index.d.ts +2 -0
- package/dist/ai/index.d.ts.map +1 -0
- package/dist/ai/index.js +3 -0
- package/dist/ai/providers/rtzr/api.d.ts +22 -0
- package/dist/ai/providers/rtzr/api.d.ts.map +1 -0
- package/dist/ai/providers/rtzr/api.js +28 -0
- package/dist/ai/providers/rtzr/error.d.ts +18 -0
- package/dist/ai/providers/rtzr/error.d.ts.map +1 -0
- package/dist/ai/providers/rtzr/error.js +29 -0
- package/dist/ai/providers/rtzr/index.d.ts +5 -0
- package/dist/ai/providers/rtzr/index.d.ts.map +1 -0
- package/dist/ai/providers/rtzr/index.js +6 -0
- package/dist/ai/providers/rtzr/model.d.ts +52 -0
- package/dist/ai/providers/rtzr/model.d.ts.map +1 -0
- package/dist/ai/providers/rtzr/model.js +137 -0
- package/dist/ai/providers/rtzr/options.d.ts +7 -0
- package/dist/ai/providers/rtzr/options.d.ts.map +1 -0
- package/dist/ai/providers/rtzr/options.js +47 -0
- package/dist/ai/providers/rtzr/provider.d.ts +18 -0
- package/dist/ai/providers/rtzr/provider.d.ts.map +1 -0
- package/dist/ai/providers/rtzr/provider.js +54 -0
- package/dist/ai/providers/rtzr/utils.d.ts +19 -0
- package/dist/ai/providers/rtzr/utils.d.ts.map +1 -0
- package/dist/ai/providers/rtzr/utils.js +88 -0
- package/dist/api/base-frame.d.ts +2 -2
- package/dist/api/base-frame.d.ts.map +1 -1
- package/dist/api/base-frame.js +13 -2
- package/dist/api/caster.d.ts.map +1 -1
- package/dist/api/caster.js +71 -2
- package/dist/api/code-converters.d.ts +58 -14
- package/dist/api/code-converters.d.ts.map +1 -1
- package/dist/api/code-converters.js +258 -2
- package/dist/api/config.d.ts +90 -0
- package/dist/api/config.d.ts.map +1 -0
- package/dist/api/config.js +25 -0
- package/dist/api/context.d.ts +4 -2
- package/dist/api/context.d.ts.map +1 -1
- package/dist/api/context.js +3 -2
- package/dist/api/decorators.d.ts +20 -6
- package/dist/api/decorators.d.ts.map +1 -1
- package/dist/api/decorators.js +235 -2
- package/dist/api/index.d.ts +2 -2
- package/dist/api/index.d.ts.map +1 -1
- package/dist/api/index.js +9 -2
- package/dist/api/sonamu.d.ts +10 -24
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +514 -2
- package/dist/api/validator.d.ts +6 -0
- package/dist/api/validator.d.ts.map +1 -0
- package/dist/api/validator.js +81 -0
- package/dist/bin/build-config.d.ts +6 -1
- package/dist/bin/build-config.d.ts.map +1 -1
- package/dist/bin/build-config.js +15 -2
- package/dist/bin/cli.js +519 -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/bin/loader-register.d.ts +2 -0
- package/dist/bin/loader-register.d.ts.map +1 -0
- package/dist/bin/loader-register.js +34 -0
- package/dist/database/_batch_update.d.ts +5 -3
- package/dist/database/_batch_update.d.ts.map +1 -1
- package/dist/database/_batch_update.js +95 -2
- package/dist/database/base-model.d.ts +96 -10
- package/dist/database/base-model.d.ts.map +1 -1
- package/dist/database/base-model.js +390 -2
- package/dist/database/base-model.types.d.ts +93 -0
- package/dist/database/base-model.types.d.ts.map +1 -0
- package/dist/database/base-model.types.js +10 -0
- package/dist/database/code-generator.d.ts +1 -1
- package/dist/database/code-generator.d.ts.map +1 -1
- package/dist/database/code-generator.js +54 -2
- package/dist/database/db.d.ts +6 -21
- package/dist/database/db.d.ts.map +1 -1
- package/dist/database/db.js +129 -2
- package/dist/database/puri-subset.test-d.js +81 -0
- package/dist/database/puri-subset.types.d.ts +123 -0
- package/dist/database/puri-subset.types.d.ts.map +1 -0
- package/dist/database/puri-subset.types.js +16 -0
- package/dist/database/puri-wrapper.d.ts +13 -11
- package/dist/database/puri-wrapper.d.ts.map +1 -1
- package/dist/database/puri-wrapper.js +109 -2
- package/dist/database/puri.d.ts +41 -23
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +601 -2
- package/dist/database/puri.types.d.ts +25 -6
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +6 -2
- package/dist/database/transaction-context.d.ts +1 -1
- package/dist/database/transaction-context.d.ts.map +1 -1
- package/dist/database/transaction-context.js +14 -2
- package/dist/database/upsert-builder.d.ts +9 -3
- package/dist/database/upsert-builder.d.ts.map +1 -1
- package/dist/database/upsert-builder.js +365 -2
- package/dist/entity/entity-manager.d.ts +167 -2
- package/dist/entity/entity-manager.d.ts.map +1 -1
- package/dist/entity/entity-manager.js +130 -2
- package/dist/entity/entity.d.ts +5 -3
- package/dist/entity/entity.d.ts.map +1 -1
- package/dist/entity/entity.js +750 -2
- package/dist/exceptions/error-handler.d.ts +1 -1
- package/dist/exceptions/error-handler.d.ts.map +1 -1
- package/dist/exceptions/error-handler.js +29 -2
- package/dist/exceptions/so-exceptions.d.ts +1 -1
- package/dist/exceptions/so-exceptions.d.ts.map +1 -1
- package/dist/exceptions/so-exceptions.js +85 -2
- package/dist/file-storage/driver.d.ts +1 -1
- package/dist/file-storage/driver.d.ts.map +1 -1
- package/dist/file-storage/driver.js +79 -2
- package/dist/file-storage/file-storage.js +75 -2
- package/dist/index.d.ts +18 -9
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +34 -2
- package/dist/migration/code-generation.d.ts +1 -1
- package/dist/migration/code-generation.d.ts.map +1 -1
- package/dist/migration/code-generation.js +614 -2
- package/dist/migration/migration-set.d.ts +2 -10
- package/dist/migration/migration-set.d.ts.map +1 -1
- package/dist/migration/migration-set.js +213 -2
- package/dist/migration/migrator.d.ts +24 -82
- package/dist/migration/migrator.d.ts.map +1 -1
- package/dist/migration/migrator.js +330 -2
- package/dist/migration/postgresql-schema-reader.d.ts +51 -0
- package/dist/migration/postgresql-schema-reader.d.ts.map +1 -0
- package/dist/migration/postgresql-schema-reader.js +245 -0
- package/dist/migration/types.d.ts +6 -38
- package/dist/migration/types.d.ts.map +1 -1
- package/dist/migration/types.js +3 -2
- package/dist/naite/messaging-types.d.ts +43 -0
- package/dist/naite/messaging-types.d.ts.map +1 -0
- package/dist/naite/messaging-types.js +7 -0
- package/dist/naite/naite-reporter.d.ts +41 -0
- package/dist/naite/naite-reporter.d.ts.map +1 -0
- package/dist/naite/naite-reporter.js +102 -0
- package/dist/naite/naite.d.ts +95 -0
- package/dist/naite/naite.d.ts.map +1 -0
- package/dist/naite/naite.js +316 -0
- package/dist/stream/index.js +3 -2
- package/dist/stream/sse.d.ts +2 -2
- package/dist/stream/sse.d.ts.map +1 -1
- package/dist/stream/sse.js +38 -2
- package/dist/syncer/api-parser.d.ts +10 -0
- package/dist/syncer/api-parser.d.ts.map +1 -0
- package/dist/syncer/api-parser.js +240 -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 +161 -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 +59 -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 +87 -0
- package/dist/syncer/syncer.d.ts +98 -106
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +422 -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 +108 -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 +86 -0
- package/dist/{templates → template/implementations}/generated.template.d.ts +3 -4
- package/dist/template/implementations/generated.template.d.ts.map +1 -0
- package/dist/template/implementations/generated.template.js +249 -0
- package/dist/{templates → template/implementations}/generated_http.template.d.ts +3 -4
- 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 +4 -5
- package/dist/template/implementations/generated_sso.template.d.ts.map +1 -0
- package/dist/template/implementations/generated_sso.template.js +134 -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 +181 -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 +201 -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 +475 -0
- package/dist/template/implementations/view_list_columns.template.d.ts +17 -0
- 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 +7 -0
- package/dist/template/index.d.ts.map +1 -0
- package/dist/template/index.js +8 -0
- package/dist/template/template-manager.d.ts +56 -0
- package/dist/template/template-manager.d.ts.map +1 -0
- package/dist/template/template-manager.js +125 -0
- package/dist/template/template-types.d.ts +16 -0
- package/dist/template/template-types.d.ts.map +1 -0
- package/dist/template/template-types.js +7 -0
- package/dist/template/template.d.ts +49 -0
- package/dist/template/template.d.ts.map +1 -0
- package/dist/template/template.js +60 -0
- package/dist/template/zod-converter.d.ts +51 -0
- package/dist/template/zod-converter.d.ts.map +1 -0
- package/dist/template/zod-converter.js +449 -0
- package/dist/testing/_relation-graph.d.ts +1 -1
- package/dist/testing/_relation-graph.d.ts.map +1 -1
- package/dist/testing/_relation-graph.js +89 -2
- package/dist/testing/fixture-manager.d.ts +42 -11
- package/dist/testing/fixture-manager.d.ts.map +1 -1
- package/dist/testing/fixture-manager.js +623 -2
- package/dist/types/types.d.ts +747 -143
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +546 -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.d.ts +1 -0
- package/dist/utils/controller.d.ts.map +1 -1
- package/dist/utils/controller.js +29 -2
- package/dist/utils/esm-utils.d.ts +39 -0
- package/dist/utils/esm-utils.d.ts.map +1 -0
- package/dist/utils/esm-utils.js +49 -0
- package/dist/utils/formatter.d.ts +3 -0
- package/dist/utils/formatter.d.ts.map +1 -0
- package/dist/utils/formatter.js +110 -0
- package/dist/utils/fs-utils.d.ts +1 -1
- package/dist/utils/fs-utils.d.ts.map +1 -1
- package/dist/utils/fs-utils.js +17 -2
- package/dist/utils/lodash-able.d.ts.map +1 -1
- package/dist/utils/lodash-able.js +6 -2
- package/dist/utils/model.js +22 -2
- package/dist/utils/object-utils.d.ts +44 -0
- package/dist/utils/object-utils.d.ts.map +1 -0
- package/dist/utils/object-utils.js +191 -0
- 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.d.ts +5 -1
- package/dist/utils/sql-parser.d.ts.map +1 -1
- package/dist/utils/sql-parser.js +46 -2
- package/dist/utils/type-utils.d.ts +23 -0
- package/dist/utils/type-utils.d.ts.map +1 -0
- package/dist/utils/type-utils.js +45 -0
- package/dist/utils/utils.d.ts +10 -7
- package/dist/utils/utils.d.ts.map +1 -1
- package/dist/utils/utils.js +72 -2
- package/dist/utils/zod-error.d.ts +1 -1
- package/dist/utils/zod-error.d.ts.map +1 -1
- package/dist/utils/zod-error.js +19 -2
- package/package.json +65 -27
- package/src/ai/agents/agent.ts +87 -0
- package/src/ai/agents/index.ts +2 -0
- package/src/ai/agents/types.ts +47 -0
- package/src/ai/index.ts +1 -0
- package/src/ai/providers/rtzr/api.ts +37 -0
- package/src/ai/providers/rtzr/error.ts +34 -0
- package/src/ai/providers/rtzr/index.ts +4 -0
- package/src/ai/providers/rtzr/model.ts +201 -0
- package/src/ai/providers/rtzr/options.ts +49 -0
- package/src/ai/providers/rtzr/provider.ts +91 -0
- package/src/ai/providers/rtzr/utils.ts +127 -0
- package/src/api/base-frame.ts +4 -2
- package/src/api/caster.ts +17 -23
- package/src/api/code-converters.ts +178 -535
- package/src/api/config.ts +125 -0
- package/src/api/context.ts +7 -17
- package/src/api/decorators.ts +176 -46
- package/src/api/index.ts +2 -2
- package/src/api/sonamu.ts +190 -167
- package/src/api/validator.ts +83 -0
- package/src/bin/build-config.ts +8 -1
- package/src/bin/cli.ts +258 -124
- package/src/bin/hot-hook-register.ts +22 -0
- package/src/bin/loader-register.ts +38 -0
- package/src/database/_batch_update.ts +46 -31
- package/src/database/base-model.ts +390 -182
- package/src/database/base-model.types.ts +155 -0
- package/src/database/code-generator.ts +13 -32
- package/src/database/db.ts +40 -96
- package/src/database/puri-subset.test-d.ts +471 -0
- package/src/database/puri-subset.types.ts +195 -0
- package/src/database/puri-wrapper.ts +58 -67
- package/src/database/puri.ts +229 -148
- package/src/database/puri.types.ts +76 -30
- package/src/database/transaction-context.ts +1 -1
- package/src/database/upsert-builder.ts +262 -132
- package/src/entity/entity-manager.ts +48 -36
- package/src/entity/entity.ts +330 -248
- package/src/exceptions/error-handler.ts +3 -3
- package/src/exceptions/so-exceptions.ts +11 -11
- package/src/file-storage/driver.ts +5 -5
- package/src/file-storage/file-storage.ts +2 -2
- package/src/index.ts +18 -10
- package/src/migration/code-generation.ts +185 -172
- package/src/migration/migration-set.ts +80 -293
- package/src/migration/migrator.ts +199 -571
- package/src/migration/mysql-schema-reader.ts.txt +272 -0
- package/src/migration/postgresql-schema-reader.ts +310 -0
- package/src/migration/types.ts +6 -39
- package/src/naite/messaging-types.ts +51 -0
- package/src/naite/naite-reporter.ts +128 -0
- package/src/naite/naite.ts +415 -0
- package/src/shared/web.shared.ts.txt +20 -24
- package/src/stream/sse.ts +5 -5
- package/src/syncer/api-parser.ts +282 -0
- package/src/syncer/checksum.ts +140 -0
- package/src/syncer/code-generator.ts +198 -0
- package/src/syncer/entity-operations.ts +65 -0
- package/src/syncer/file-patterns.ts +56 -0
- package/src/syncer/index.ts +6 -0
- package/src/syncer/module-loader.ts +128 -0
- package/src/syncer/syncer.ts +389 -1453
- package/src/template/entity-converter.ts +114 -0
- package/src/template/helpers.ts +81 -0
- package/src/{templates → template/implementations}/entity.template.ts +7 -7
- package/src/{templates → template/implementations}/generated.template.ts +101 -101
- package/src/{templates → template/implementations}/generated_http.template.ts +27 -57
- package/src/template/implementations/generated_sso.template.ts +151 -0
- package/src/{templates → template/implementations}/init_types.template.ts +5 -7
- package/src/{templates → template/implementations}/model.template.ts +52 -43
- package/src/{templates → template/implementations}/model_test.template.ts +5 -5
- package/src/{templates → template/implementations}/service.template.ts +66 -82
- package/src/{templates → template/implementations}/view_enums_buttonset.template.ts +3 -3
- package/src/{templates → template/implementations}/view_enums_dropdown.template.ts +4 -20
- package/src/{templates → template/implementations}/view_enums_select.template.ts +4 -4
- package/src/{templates → template/implementations}/view_form.template.ts +40 -83
- package/src/{templates → template/implementations}/view_id_all_select.template.ts +3 -3
- package/src/{templates → template/implementations}/view_id_async_select.template.ts +10 -24
- package/src/{templates → template/implementations}/view_list.template.ts +60 -152
- package/src/{templates → template/implementations}/view_list_columns.template.ts +5 -11
- package/src/{templates → template/implementations}/view_search_input.template.ts +3 -3
- package/src/template/index.ts +6 -0
- package/src/template/template-manager.ts +166 -0
- package/src/template/template-types.ts +16 -0
- package/src/template/template.ts +105 -0
- package/src/template/zod-converter.ts +525 -0
- package/src/testing/_relation-graph.ts +18 -11
- package/src/testing/fixture-manager.ts +472 -359
- package/src/types/types.ts +553 -308
- package/src/typings/knex.d.ts +7 -9
- package/src/utils/async-utils.ts +23 -10
- package/src/utils/console-util.ts +4 -0
- package/src/utils/controller.ts +3 -0
- package/src/utils/esm-utils.ts +59 -0
- package/src/utils/formatter.ts +109 -0
- package/src/utils/fs-utils.ts +1 -1
- package/src/utils/lodash-able.ts +1 -4
- package/src/utils/object-utils.ts +217 -0
- package/src/utils/path-utils.ts +99 -0
- package/src/utils/process-utils.ts +46 -0
- package/src/utils/sql-parser.ts +23 -5
- package/src/utils/type-utils.ts +83 -0
- package/src/utils/utils.ts +66 -43
- package/src/utils/zod-error.ts +3 -4
- 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.d.ts +0 -3
- package/dist/bin/cli-wrapper.d.ts.map +0 -1
- package/dist/bin/cli-wrapper.js +0 -3
- 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.d.ts +0 -2
- package/dist/database/knex-plugins/knex-on-duplicate-update.d.ts.map +0 -1
- package/dist/database/knex-plugins/knex-on-duplicate-update.js +0 -2
- 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.d.ts +0 -61
- package/dist/entity/entity-utils.d.ts.map +0 -1
- package/dist/entity/entity-utils.js +0 -2
- 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 +0 -17
- 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/bin/cli-wrapper.ts +0 -75
- package/src/database/knex-plugins/knex-on-duplicate-update.ts +0 -45
- package/src/entity/entity-utils.ts +0 -291
- package/src/templates/base-template.ts +0 -19
- package/src/templates/generated_sso.template.ts +0 -138
- package/src/templates/index.ts +0 -1
package/dist/entity/entity.js
CHANGED
|
@@ -1,2 +1,750 @@
|
|
|
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 assert from "assert";
|
|
2
|
+
import { writeFile } from "node:fs/promises";
|
|
3
|
+
import inflection from "inflection";
|
|
4
|
+
import path from "path";
|
|
5
|
+
import { group, unique } from "radashi";
|
|
6
|
+
import { z } from "zod";
|
|
7
|
+
import { Sonamu } from "../api/sonamu.js";
|
|
8
|
+
import { isBelongsToOneRelationProp, isEnumProp, isHasManyRelationProp, isManyToManyRelationProp, isOneToOneRelationProp, isRelationProp, isVirtualProp } from "../types/types.js";
|
|
9
|
+
import { importMembers } from "../utils/esm-utils.js";
|
|
10
|
+
import { formatCode } from "../utils/formatter.js";
|
|
11
|
+
import { exists } from "../utils/fs-utils.js";
|
|
12
|
+
import { runtimePath } from "../utils/path-utils.js";
|
|
13
|
+
import { assertDefined, nonNullable } from "../utils/utils.js";
|
|
14
|
+
import { EntityManager } from "./entity-manager.js";
|
|
15
|
+
export class Entity {
|
|
16
|
+
id;
|
|
17
|
+
parentId;
|
|
18
|
+
table;
|
|
19
|
+
title;
|
|
20
|
+
names;
|
|
21
|
+
props;
|
|
22
|
+
propsDict;
|
|
23
|
+
relations;
|
|
24
|
+
indexes;
|
|
25
|
+
subsets;
|
|
26
|
+
types = {};
|
|
27
|
+
enums = {};
|
|
28
|
+
enumLabels = {};
|
|
29
|
+
constructor({ id, parentId, table, title, props, indexes, subsets, enums }){
|
|
30
|
+
// id
|
|
31
|
+
this.id = id;
|
|
32
|
+
this.parentId = parentId;
|
|
33
|
+
this.title = title ?? this.id;
|
|
34
|
+
this.table = table ?? inflection.underscore(inflection.pluralize(id));
|
|
35
|
+
// props
|
|
36
|
+
if (props) {
|
|
37
|
+
this.props = props.map((prop)=>{
|
|
38
|
+
if (isEnumProp(prop)) {
|
|
39
|
+
if (prop.id.includes("$Model")) {
|
|
40
|
+
prop.id = prop.id.replace("$Model", id);
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
return prop;
|
|
44
|
+
});
|
|
45
|
+
this.propsDict = Object.fromEntries(props.map((prop)=>{
|
|
46
|
+
return [
|
|
47
|
+
prop.name,
|
|
48
|
+
prop
|
|
49
|
+
];
|
|
50
|
+
}));
|
|
51
|
+
// relations
|
|
52
|
+
this.relations = Object.fromEntries(props.filter((prop)=>isRelationProp(prop)).map((prop)=>[
|
|
53
|
+
prop.name,
|
|
54
|
+
prop
|
|
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을 Puri 코드로 변환
|
|
82
|
+
*/ getPuriSubsetQuery(subsetKey) {
|
|
83
|
+
const subset = this.subsets[subsetKey];
|
|
84
|
+
const subsetQuery = this.resolveSubsetQuery("", subset);
|
|
85
|
+
const lines = [];
|
|
86
|
+
// from
|
|
87
|
+
lines.push(`return qbWrapper`);
|
|
88
|
+
lines.push(`.from("${this.table}")`);
|
|
89
|
+
// join
|
|
90
|
+
for (const join of subsetQuery.joins){
|
|
91
|
+
const joinMethod = join.join === "inner" ? "join" : "leftJoin";
|
|
92
|
+
if ("custom" in join) {
|
|
93
|
+
// custom join clause는 raw 사용
|
|
94
|
+
lines.push(`.${joinMethod}({ ${join.as}: "${join.table}" }, qbWrapper.knex.raw(\`${join.custom}\`))`);
|
|
95
|
+
} else {
|
|
96
|
+
lines.push(`.${joinMethod}({ ${join.as}: "${join.table}" }, "${join.from}", "${join.to}")`);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
// select
|
|
100
|
+
const selectObj = {};
|
|
101
|
+
for (const selectItem of subsetQuery.select){
|
|
102
|
+
// "users.id" 또는 "users.id as user__id" 형태
|
|
103
|
+
const match = selectItem.match(/^(.+?)(?: as (.+))?$/);
|
|
104
|
+
if (match) {
|
|
105
|
+
const [, column, alias] = match;
|
|
106
|
+
const key = alias ?? assertDefined(column.split(".").pop());
|
|
107
|
+
selectObj[key] = `"${column.trim()}"`;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
lines.push(`.select({`);
|
|
111
|
+
Object.entries(selectObj).forEach(([key, value])=>{
|
|
112
|
+
lines.push(`${key}: ${value},`);
|
|
113
|
+
});
|
|
114
|
+
lines.push(`});`);
|
|
115
|
+
return lines.join("\n");
|
|
116
|
+
}
|
|
117
|
+
getPuriLoaderQuery(subsetKey) {
|
|
118
|
+
const subset = this.subsets[subsetKey];
|
|
119
|
+
const { loaders } = this.resolveSubsetQuery("", subset);
|
|
120
|
+
const lines = [
|
|
121
|
+
`[`
|
|
122
|
+
];
|
|
123
|
+
const parseSelect = (select, table)=>{
|
|
124
|
+
const tablePrefix = `${table}.`;
|
|
125
|
+
if (select.startsWith(tablePrefix)) {
|
|
126
|
+
return `${select.replace(tablePrefix, "")}: "${select}"`;
|
|
127
|
+
}
|
|
128
|
+
if (select.includes(" as ")) {
|
|
129
|
+
const [column, alias] = select.split(" as ");
|
|
130
|
+
return `${alias}: "${column}"`;
|
|
131
|
+
}
|
|
132
|
+
return `${select}: "${select}"`;
|
|
133
|
+
};
|
|
134
|
+
// 재귀적으로 loader 생성하는 헬퍼 함수
|
|
135
|
+
const generateLoaderCode = (loaders)=>{
|
|
136
|
+
const loaderLines = [];
|
|
137
|
+
for (const loader of loaders){
|
|
138
|
+
const { toTable, toCol, through } = loader.manyJoin;
|
|
139
|
+
loaderLines.push("{", `as: "${loader.as}",`, `refId: "${loader.manyJoin.idField}",`, `qb: (qbWrapper: PuriWrapper<DatabaseSchemaExtend>, fromIds: number[]) => {`);
|
|
140
|
+
if (through === undefined) {
|
|
141
|
+
// HasMany
|
|
142
|
+
loaderLines.push(//
|
|
143
|
+
"return qbWrapper", `.from("${toTable}")`);
|
|
144
|
+
loader.oneJoins.forEach((join)=>{
|
|
145
|
+
const joinType = join.join === "inner" ? "join" : "leftJoin";
|
|
146
|
+
if ("custom" in join) {
|
|
147
|
+
// FIXME: 검증 필요
|
|
148
|
+
loaderLines.push(`.${joinType}({ ${join.as}: "${join.table}" }, (j) => {`, `j.on(Puri.rawString("${join.custom}"));`, "})");
|
|
149
|
+
} else {
|
|
150
|
+
loaderLines.push(`.${joinType}({ ${join.as}: "${join.table}" }, "${join.from}", "${join.to}")`);
|
|
151
|
+
}
|
|
152
|
+
});
|
|
153
|
+
loaderLines.push(`.whereIn("${toTable}.${toCol}", fromIds)`, `.select({`, `${loader.select.map((select)=>parseSelect(select, toTable)).join(",")},`, `refId: "${toTable}.${toCol}",`, `});`);
|
|
154
|
+
} else {
|
|
155
|
+
// ManyToMany
|
|
156
|
+
loaderLines.push("return qbWrapper", `.from("${through.table}")`, `.join("${toTable}", "${through.table}.${through.toCol}", "${toTable}.${toCol}")`);
|
|
157
|
+
loader.oneJoins.forEach((join)=>{
|
|
158
|
+
const joinType = join.join === "inner" ? "join" : "leftJoin";
|
|
159
|
+
if ("custom" in join) {
|
|
160
|
+
// FIXME: 검증 필요
|
|
161
|
+
loaderLines.push(`.${joinType}({ ${join.as}: "${join.table}" }, (j) => {`, `j.on(Puri.rawString("${join.custom}"));`, "})");
|
|
162
|
+
} else {
|
|
163
|
+
loaderLines.push(`.${joinType}({ ${join.as}: "${join.table}" }, "${join.from}", "${join.to}")`);
|
|
164
|
+
}
|
|
165
|
+
});
|
|
166
|
+
loaderLines.push(`.whereIn("${through.table}.${through.fromCol}", fromIds)`, `.select({`, `${loader.select.map((select)=>parseSelect(select, toTable)).join(",")},`, `refId: "${through.table}.${through.fromCol}",`, `});`);
|
|
167
|
+
}
|
|
168
|
+
loaderLines.push(`},`);
|
|
169
|
+
// 중첩 loaders 처리
|
|
170
|
+
if (loader.loaders && loader.loaders.length > 0) {
|
|
171
|
+
loaderLines.push("loaders: [", ...generateLoaderCode(loader.loaders), "],");
|
|
172
|
+
}
|
|
173
|
+
loaderLines.push("},");
|
|
174
|
+
}
|
|
175
|
+
return loaderLines;
|
|
176
|
+
};
|
|
177
|
+
lines.push(...generateLoaderCode(loaders));
|
|
178
|
+
lines.push(`]`);
|
|
179
|
+
return lines.join("\n");
|
|
180
|
+
}
|
|
181
|
+
/*
|
|
182
|
+
subset SELECT/JOIN/LOADER 결과 리턴
|
|
183
|
+
*/ getSubsetQuery(subsetKey) {
|
|
184
|
+
const subset = this.subsets[subsetKey];
|
|
185
|
+
const result = this.resolveSubsetQuery("", subset);
|
|
186
|
+
return result;
|
|
187
|
+
}
|
|
188
|
+
/*
|
|
189
|
+
*/ resolveSubsetQuery(prefix, fields, isAlreadyOuterJoined = false) {
|
|
190
|
+
// prefix 치환 (prefix는 ToOneRelation이 복수로 붙은 경우 모두 __로 변경됨)
|
|
191
|
+
prefix = prefix.replace(/\./g, "__");
|
|
192
|
+
// 서브셋을 1뎁스만 분리하여 그룹핑
|
|
193
|
+
const subsetGroup = group(fields, (field)=>{
|
|
194
|
+
if (field.includes(".")) {
|
|
195
|
+
const [rel] = field.split(".");
|
|
196
|
+
return rel;
|
|
197
|
+
} else {
|
|
198
|
+
return "";
|
|
199
|
+
}
|
|
200
|
+
});
|
|
201
|
+
const result = Object.keys(subsetGroup).reduce((r, groupKey)=>{
|
|
202
|
+
const fields = subsetGroup[groupKey];
|
|
203
|
+
assert(fields !== undefined, "fields is undefined");
|
|
204
|
+
// 현재 테이블 필드셋은 select, virtual에 추가하고 리턴
|
|
205
|
+
if (groupKey === "") {
|
|
206
|
+
const realFields = fields.filter((field)=>!isVirtualProp(this.propsDict[field]));
|
|
207
|
+
const virtualFields = fields.filter((field)=>isVirtualProp(this.propsDict[field]));
|
|
208
|
+
if (prefix === "") {
|
|
209
|
+
// 현재 테이블인 경우
|
|
210
|
+
r.select = r.select.concat(realFields.map((field)=>`${this.table}.${field}`));
|
|
211
|
+
r.virtual = r.virtual.concat(virtualFields);
|
|
212
|
+
} else {
|
|
213
|
+
// 넘어온 테이블인 경우
|
|
214
|
+
r.select = r.select.concat(realFields.map((field)=>`${prefix}.${field} as ${prefix}__${field}`));
|
|
215
|
+
}
|
|
216
|
+
return r;
|
|
217
|
+
}
|
|
218
|
+
const relation = this.relations[groupKey];
|
|
219
|
+
if (relation === undefined) {
|
|
220
|
+
throw new Error(`존재하지 않는 relation 참조 ${groupKey}`);
|
|
221
|
+
}
|
|
222
|
+
const relEntity = EntityManager.get(relation.with);
|
|
223
|
+
if (isOneToOneRelationProp(relation) || isBelongsToOneRelationProp(relation)) {
|
|
224
|
+
// -One Relation: JOIN 으로 처리
|
|
225
|
+
const relFields = fields.map((field)=>field.split(".").slice(1).join("."));
|
|
226
|
+
// -One Relation에서 id 필드만 참조하는 경우 릴레이션 넘기지 않고 리턴
|
|
227
|
+
if (relFields.length === 1 && relFields[0] === "id") {
|
|
228
|
+
if (prefix === "") {
|
|
229
|
+
r.select = r.select.concat(`${this.table}.${groupKey}_id`);
|
|
230
|
+
} else {
|
|
231
|
+
r.select = r.select.concat(`${prefix}.${groupKey}_id as ${prefix}__${groupKey}_id`);
|
|
232
|
+
}
|
|
233
|
+
return r;
|
|
234
|
+
}
|
|
235
|
+
// innerOrOuter
|
|
236
|
+
const innerOrOuter = (()=>{
|
|
237
|
+
if (isAlreadyOuterJoined) {
|
|
238
|
+
return "outer";
|
|
239
|
+
}
|
|
240
|
+
if (isOneToOneRelationProp(relation)) {
|
|
241
|
+
if (relation.hasJoinColumn === true && (relation.nullable ?? false) === false) {
|
|
242
|
+
return "inner";
|
|
243
|
+
} else {
|
|
244
|
+
return "outer";
|
|
245
|
+
}
|
|
246
|
+
} else {
|
|
247
|
+
if (relation.nullable) {
|
|
248
|
+
return "outer";
|
|
249
|
+
} else {
|
|
250
|
+
return "inner";
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
})();
|
|
254
|
+
const relSubsetQuery = relEntity.resolveSubsetQuery(`${prefix !== "" ? `${prefix}.` : ""}${groupKey}`, relFields, innerOrOuter === "outer");
|
|
255
|
+
r.select = r.select.concat(relSubsetQuery.select);
|
|
256
|
+
r.virtual = r.virtual.concat(relSubsetQuery.virtual);
|
|
257
|
+
const joinAs = prefix === "" ? groupKey : `${prefix}__${groupKey}`;
|
|
258
|
+
const fromTable = prefix === "" ? this.table : prefix;
|
|
259
|
+
let joinClause;
|
|
260
|
+
if (relation.customJoinClause) {
|
|
261
|
+
joinClause = {
|
|
262
|
+
custom: relation.customJoinClause
|
|
263
|
+
};
|
|
264
|
+
} else {
|
|
265
|
+
let from, to;
|
|
266
|
+
if (isOneToOneRelationProp(relation)) {
|
|
267
|
+
if (relation.hasJoinColumn) {
|
|
268
|
+
from = `${fromTable}.${relation.name}_id`;
|
|
269
|
+
to = `${joinAs}.id`;
|
|
270
|
+
} else {
|
|
271
|
+
from = `${fromTable}.id`;
|
|
272
|
+
to = `${joinAs}.${inflection.underscore(this.names.fs.replace(/-/g, "_"))}_id`;
|
|
273
|
+
}
|
|
274
|
+
} else {
|
|
275
|
+
from = `${fromTable}.${relation.name}_id`;
|
|
276
|
+
to = `${joinAs}.id`;
|
|
277
|
+
}
|
|
278
|
+
joinClause = {
|
|
279
|
+
from,
|
|
280
|
+
to
|
|
281
|
+
};
|
|
282
|
+
}
|
|
283
|
+
r.joins.push({
|
|
284
|
+
as: joinAs,
|
|
285
|
+
join: innerOrOuter,
|
|
286
|
+
table: relEntity.table,
|
|
287
|
+
...joinClause
|
|
288
|
+
});
|
|
289
|
+
// BelongsToOne 밑에 HasMany가 붙은 경우
|
|
290
|
+
if (relSubsetQuery.loaders.length > 0) {
|
|
291
|
+
const convertedLoaders = relSubsetQuery.loaders.map((loader)=>{
|
|
292
|
+
const newAs = [
|
|
293
|
+
groupKey,
|
|
294
|
+
loader.as
|
|
295
|
+
].join("__");
|
|
296
|
+
return {
|
|
297
|
+
as: newAs,
|
|
298
|
+
table: loader.table,
|
|
299
|
+
manyJoin: loader.manyJoin,
|
|
300
|
+
oneJoins: loader.oneJoins,
|
|
301
|
+
select: loader.select,
|
|
302
|
+
loaders: loader.loaders
|
|
303
|
+
};
|
|
304
|
+
});
|
|
305
|
+
r.loaders = [
|
|
306
|
+
...r.loaders,
|
|
307
|
+
...convertedLoaders
|
|
308
|
+
];
|
|
309
|
+
}
|
|
310
|
+
r.joins = r.joins.concat(relSubsetQuery.joins);
|
|
311
|
+
} else if (isHasManyRelationProp(relation) || isManyToManyRelationProp(relation)) {
|
|
312
|
+
// -Many Relation: Loader 로 처리
|
|
313
|
+
const relFields = fields.map((field)=>field.split(".").slice(1).join("."));
|
|
314
|
+
const relSubsetQuery = relEntity.resolveSubsetQuery("", relFields);
|
|
315
|
+
let manyJoin;
|
|
316
|
+
if (isHasManyRelationProp(relation)) {
|
|
317
|
+
const fromCol = relation?.fromColumn ?? "id";
|
|
318
|
+
manyJoin = {
|
|
319
|
+
fromTable: this.table,
|
|
320
|
+
fromCol,
|
|
321
|
+
idField: prefix === "" ? `${fromCol}` : `${prefix}__${fromCol}`,
|
|
322
|
+
toTable: relEntity.table,
|
|
323
|
+
toCol: relation.joinColumn
|
|
324
|
+
};
|
|
325
|
+
} else if (isManyToManyRelationProp(relation)) {
|
|
326
|
+
manyJoin = {
|
|
327
|
+
fromTable: this.table,
|
|
328
|
+
fromCol: "id",
|
|
329
|
+
idField: prefix === "" ? `id` : `${prefix}__id`,
|
|
330
|
+
through: {
|
|
331
|
+
table: relation.joinTable,
|
|
332
|
+
fromCol: `${inflection.singularize(this.table)}_id`,
|
|
333
|
+
toCol: `${inflection.singularize(relEntity.table)}_id`
|
|
334
|
+
},
|
|
335
|
+
toTable: relEntity.table,
|
|
336
|
+
toCol: "id"
|
|
337
|
+
};
|
|
338
|
+
} else {
|
|
339
|
+
throw new Error();
|
|
340
|
+
}
|
|
341
|
+
r.loaders.push({
|
|
342
|
+
as: groupKey,
|
|
343
|
+
table: relEntity.table,
|
|
344
|
+
manyJoin,
|
|
345
|
+
oneJoins: relSubsetQuery.joins,
|
|
346
|
+
select: relSubsetQuery.select,
|
|
347
|
+
loaders: relSubsetQuery.loaders
|
|
348
|
+
});
|
|
349
|
+
}
|
|
350
|
+
return r;
|
|
351
|
+
}, {
|
|
352
|
+
select: [],
|
|
353
|
+
virtual: [],
|
|
354
|
+
joins: [],
|
|
355
|
+
loaders: []
|
|
356
|
+
});
|
|
357
|
+
return result;
|
|
358
|
+
}
|
|
359
|
+
/*
|
|
360
|
+
FieldExpr[] 을 EntityPropNode[] 로 변환
|
|
361
|
+
*/ fieldExprsToPropNodes(fieldExprs, entity = this) {
|
|
362
|
+
const groups = fieldExprs.reduce((result, fieldExpr)=>{
|
|
363
|
+
let key, value, elseExpr;
|
|
364
|
+
if (fieldExpr.includes(".")) {
|
|
365
|
+
[key, ...elseExpr] = fieldExpr.split(".");
|
|
366
|
+
value = elseExpr.join(".");
|
|
367
|
+
} else {
|
|
368
|
+
key = "";
|
|
369
|
+
value = fieldExpr;
|
|
370
|
+
}
|
|
371
|
+
result[key] = (result[key] ?? []).concat(value);
|
|
372
|
+
return result;
|
|
373
|
+
}, {});
|
|
374
|
+
return Object.keys(groups).flatMap((key)=>{
|
|
375
|
+
const group = groups[key];
|
|
376
|
+
// 일반 prop 처리
|
|
377
|
+
if (key === "") {
|
|
378
|
+
return group.map((propName)=>{
|
|
379
|
+
// uuid 개별 처리
|
|
380
|
+
if (propName === "uuid") {
|
|
381
|
+
return {
|
|
382
|
+
nodeType: "plain",
|
|
383
|
+
prop: {
|
|
384
|
+
type: "string",
|
|
385
|
+
name: "uuid",
|
|
386
|
+
length: 128
|
|
387
|
+
},
|
|
388
|
+
children: []
|
|
389
|
+
};
|
|
390
|
+
}
|
|
391
|
+
const prop = entity.props.find((p)=>p.name === propName);
|
|
392
|
+
if (prop === undefined) {
|
|
393
|
+
console.log({
|
|
394
|
+
propName,
|
|
395
|
+
groups
|
|
396
|
+
});
|
|
397
|
+
throw new Error(`${entity.id} -- 잘못된 FieldExpr ${propName}`);
|
|
398
|
+
}
|
|
399
|
+
return {
|
|
400
|
+
nodeType: "plain",
|
|
401
|
+
prop,
|
|
402
|
+
children: []
|
|
403
|
+
};
|
|
404
|
+
});
|
|
405
|
+
}
|
|
406
|
+
// relation prop 처리
|
|
407
|
+
const prop = entity.propsDict[key];
|
|
408
|
+
if (!isRelationProp(prop)) {
|
|
409
|
+
throw new Error(`잘못된 FieldExpr ${key}.${group[0]}`);
|
|
410
|
+
}
|
|
411
|
+
const relEntity = EntityManager.get(prop.with);
|
|
412
|
+
// relation -One 에 id 필드 하나인 경우
|
|
413
|
+
if (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop)) {
|
|
414
|
+
if (group.length === 1 && (group[0] === "id" || group[0] === "id?")) {
|
|
415
|
+
// id 하나만 있는지 체크해서, 하나만 있으면 상위 prop으로 id를 리턴
|
|
416
|
+
const idProp = relEntity.propsDict.id;
|
|
417
|
+
return {
|
|
418
|
+
nodeType: "plain",
|
|
419
|
+
prop: {
|
|
420
|
+
...idProp,
|
|
421
|
+
name: `${key}_id`,
|
|
422
|
+
nullable: prop.nullable
|
|
423
|
+
},
|
|
424
|
+
children: []
|
|
425
|
+
};
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
// -One 그외의 경우 object로 리턴
|
|
429
|
+
// -Many의 경우 array로 리턴
|
|
430
|
+
// Recursive 로 뎁스 처리
|
|
431
|
+
const children = this.fieldExprsToPropNodes(group, relEntity);
|
|
432
|
+
const nodeType = isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop) ? "object" : "array";
|
|
433
|
+
return {
|
|
434
|
+
prop,
|
|
435
|
+
children,
|
|
436
|
+
nodeType
|
|
437
|
+
};
|
|
438
|
+
});
|
|
439
|
+
}
|
|
440
|
+
getFieldExprs(prefix = "", maxDepth = 3, froms = []) {
|
|
441
|
+
return this.props.flatMap((prop)=>{
|
|
442
|
+
const propName = [
|
|
443
|
+
prefix,
|
|
444
|
+
prop.name
|
|
445
|
+
].filter((v)=>v !== "").join(".");
|
|
446
|
+
if (propName === prefix) {
|
|
447
|
+
return null;
|
|
448
|
+
}
|
|
449
|
+
if (isRelationProp(prop)) {
|
|
450
|
+
if (maxDepth < 0) {
|
|
451
|
+
return null;
|
|
452
|
+
}
|
|
453
|
+
if (froms.includes(prop.with)) {
|
|
454
|
+
// 역방향 relation인 경우 제외
|
|
455
|
+
return null;
|
|
456
|
+
}
|
|
457
|
+
// 정방향 relation인 경우 recursive 콜
|
|
458
|
+
const relMd = EntityManager.get(prop.with);
|
|
459
|
+
return relMd.getFieldExprs(propName, maxDepth - 1, [
|
|
460
|
+
...froms,
|
|
461
|
+
this.id
|
|
462
|
+
]);
|
|
463
|
+
}
|
|
464
|
+
return propName;
|
|
465
|
+
}).filter((f)=>f !== null);
|
|
466
|
+
}
|
|
467
|
+
getTableColumns() {
|
|
468
|
+
return this.props.map((prop)=>{
|
|
469
|
+
if (prop.type === "relation") {
|
|
470
|
+
if (prop.relationType === "BelongsToOne" || prop.relationType === "OneToOne" && prop.hasJoinColumn === true) {
|
|
471
|
+
return {
|
|
472
|
+
name: `${prop.name}_id`,
|
|
473
|
+
type: "int_unsigned"
|
|
474
|
+
};
|
|
475
|
+
} else {
|
|
476
|
+
return null;
|
|
477
|
+
}
|
|
478
|
+
}
|
|
479
|
+
return {
|
|
480
|
+
name: prop.name,
|
|
481
|
+
type: prop.type
|
|
482
|
+
};
|
|
483
|
+
}).filter(nonNullable);
|
|
484
|
+
}
|
|
485
|
+
async registerModulePaths() {
|
|
486
|
+
const basePath = `${this.names.parentFs}`;
|
|
487
|
+
// base-scheme
|
|
488
|
+
EntityManager.setModulePath(`${this.id}BaseSchema`, `sonamu.generated`);
|
|
489
|
+
// subset
|
|
490
|
+
if (Object.keys(this.subsets).length > 0) {
|
|
491
|
+
EntityManager.setModulePath(`${this.id}SubsetKey`, `sonamu.generated`);
|
|
492
|
+
EntityManager.setModulePath(`${this.id}SubsetMapping`, `sonamu.generated`);
|
|
493
|
+
for (const subsetKey of Object.keys(this.subsets)){
|
|
494
|
+
EntityManager.setModulePath(`${this.id}Subset${subsetKey.toUpperCase()}`, `sonamu.generated`);
|
|
495
|
+
}
|
|
496
|
+
}
|
|
497
|
+
// enums
|
|
498
|
+
for (const enumId of Object.keys(this.enumLabels)){
|
|
499
|
+
EntityManager.setModulePath(enumId, `sonamu.generated`);
|
|
500
|
+
}
|
|
501
|
+
// types
|
|
502
|
+
const typesModulePath = `${basePath}/${this.names.parentFs}.types`;
|
|
503
|
+
const typesFilePath = path.join(Sonamu.apiRootPath, runtimePath(`dist/application/${typesModulePath}.js`));
|
|
504
|
+
if (await exists(typesFilePath)) {
|
|
505
|
+
const importedMembers = await importMembers(typesFilePath);
|
|
506
|
+
this.types = Object.fromEntries(importedMembers.map(({ name, value })=>{
|
|
507
|
+
EntityManager.setModulePath(name, typesModulePath);
|
|
508
|
+
return [
|
|
509
|
+
name,
|
|
510
|
+
value
|
|
511
|
+
];
|
|
512
|
+
}));
|
|
513
|
+
}
|
|
514
|
+
}
|
|
515
|
+
registerTableSpecs() {
|
|
516
|
+
const uniqueIndexes = this.indexes.filter((idx)=>idx.type === "unique");
|
|
517
|
+
EntityManager.setTableSpec({
|
|
518
|
+
name: this.table,
|
|
519
|
+
uniqueIndexes
|
|
520
|
+
});
|
|
521
|
+
}
|
|
522
|
+
toJson() {
|
|
523
|
+
return {
|
|
524
|
+
id: this.id,
|
|
525
|
+
parentId: this.parentId,
|
|
526
|
+
table: this.table,
|
|
527
|
+
title: this.title,
|
|
528
|
+
props: this.props,
|
|
529
|
+
indexes: this.indexes,
|
|
530
|
+
subsets: this.subsets,
|
|
531
|
+
enums: this.enumLabels
|
|
532
|
+
};
|
|
533
|
+
}
|
|
534
|
+
async save() {
|
|
535
|
+
// sort: subsets
|
|
536
|
+
const subsetRows = this.getSubsetRows();
|
|
537
|
+
this.subsets = Object.fromEntries(Object.entries(this.subsets).map(([subsetKey])=>{
|
|
538
|
+
return [
|
|
539
|
+
subsetKey,
|
|
540
|
+
this.subsetRowsToSubsetFields(subsetRows, subsetKey)
|
|
541
|
+
];
|
|
542
|
+
}));
|
|
543
|
+
// save
|
|
544
|
+
const jsonPath = path.join(Sonamu.apiRootPath, `src/application/${this.names.parentFs}/${this.names.fs}.entity.json`);
|
|
545
|
+
const json = this.toJson();
|
|
546
|
+
await writeFile(jsonPath, formatCode(JSON.stringify(json), "json", jsonPath));
|
|
547
|
+
// reload
|
|
548
|
+
await EntityManager.register(json);
|
|
549
|
+
}
|
|
550
|
+
getSubsetRows(_subsets, prefixes = []) {
|
|
551
|
+
if (prefixes.length > 10) {
|
|
552
|
+
return [];
|
|
553
|
+
}
|
|
554
|
+
const subsets = _subsets ?? this.subsets;
|
|
555
|
+
const subsetKeys = Object.keys(subsets);
|
|
556
|
+
const allFields = unique(subsetKeys.flatMap((key)=>subsets[key]));
|
|
557
|
+
return this.props.map((prop)=>{
|
|
558
|
+
if (prop.type === "relation" && allFields.find((f)=>f.startsWith(`${[
|
|
559
|
+
...prefixes,
|
|
560
|
+
prop.name
|
|
561
|
+
].join(".")}.`))) {
|
|
562
|
+
const relEntity = EntityManager.get(prop.with);
|
|
563
|
+
const children = relEntity.getSubsetRows(subsets, [
|
|
564
|
+
...prefixes,
|
|
565
|
+
`${prop.name}`
|
|
566
|
+
]);
|
|
567
|
+
return {
|
|
568
|
+
field: prop.name,
|
|
569
|
+
children,
|
|
570
|
+
relationEntity: prop.with,
|
|
571
|
+
prefixes,
|
|
572
|
+
isOpen: children.length > 0,
|
|
573
|
+
has: Object.fromEntries(subsetKeys.map((subsetKey)=>{
|
|
574
|
+
return [
|
|
575
|
+
subsetKey,
|
|
576
|
+
children.every((child)=>child.has[subsetKey] === true)
|
|
577
|
+
];
|
|
578
|
+
}))
|
|
579
|
+
};
|
|
580
|
+
}
|
|
581
|
+
return {
|
|
582
|
+
field: prop.name,
|
|
583
|
+
children: [],
|
|
584
|
+
relationEntity: prop.type === "relation" ? prop.with : undefined,
|
|
585
|
+
prefixes,
|
|
586
|
+
has: Object.fromEntries(subsetKeys.map((subsetKey)=>{
|
|
587
|
+
const subsetFields = subsets[subsetKey];
|
|
588
|
+
const has = subsetFields.some((f)=>{
|
|
589
|
+
const field = [
|
|
590
|
+
...prefixes,
|
|
591
|
+
prop.name
|
|
592
|
+
].join(".");
|
|
593
|
+
return f === field || f.startsWith(`${field}.`);
|
|
594
|
+
});
|
|
595
|
+
return [
|
|
596
|
+
subsetKey,
|
|
597
|
+
has
|
|
598
|
+
];
|
|
599
|
+
}))
|
|
600
|
+
};
|
|
601
|
+
});
|
|
602
|
+
}
|
|
603
|
+
subsetRowsToSubsetFields(subsetRows, subsetKey) {
|
|
604
|
+
return subsetRows.map((subsetRow)=>{
|
|
605
|
+
if (subsetRow.children.length > 0) {
|
|
606
|
+
return this.subsetRowsToSubsetFields(subsetRow.children, subsetKey);
|
|
607
|
+
} else if (subsetRow.has[subsetKey]) {
|
|
608
|
+
return subsetRow.prefixes.concat(subsetRow.field).join(".");
|
|
609
|
+
} else {
|
|
610
|
+
return null;
|
|
611
|
+
}
|
|
612
|
+
}).filter(nonNullable).flat();
|
|
613
|
+
}
|
|
614
|
+
async createProp(prop, at) {
|
|
615
|
+
if (!at) {
|
|
616
|
+
this.props.push(prop);
|
|
617
|
+
} else {
|
|
618
|
+
this.props.splice(at, 0, prop);
|
|
619
|
+
}
|
|
620
|
+
await this.save();
|
|
621
|
+
}
|
|
622
|
+
analyzeSubsetField(subsetField) {
|
|
623
|
+
const arr = subsetField.split(".");
|
|
624
|
+
let entityId = this.id;
|
|
625
|
+
const result = [];
|
|
626
|
+
for(let i = 0; i < arr.length; i++){
|
|
627
|
+
const propName = arr[i];
|
|
628
|
+
result.push({
|
|
629
|
+
entityId,
|
|
630
|
+
propName
|
|
631
|
+
});
|
|
632
|
+
const prop = EntityManager.get(entityId).props.find((p)=>p.name === propName);
|
|
633
|
+
if (!prop) {
|
|
634
|
+
throw new Error(`${entityId}의 잘못된 서브셋키 ${subsetField}`);
|
|
635
|
+
}
|
|
636
|
+
if (isRelationProp(prop)) {
|
|
637
|
+
entityId = prop.with;
|
|
638
|
+
}
|
|
639
|
+
}
|
|
640
|
+
return result;
|
|
641
|
+
}
|
|
642
|
+
async modifyProp(newProp, at) {
|
|
643
|
+
// 이전 프롭 이름 저장
|
|
644
|
+
const oldName = this.props[at].name;
|
|
645
|
+
// 저장할 엔티티
|
|
646
|
+
const entities = [
|
|
647
|
+
this
|
|
648
|
+
];
|
|
649
|
+
// 이름이 바뀐 경우
|
|
650
|
+
if (oldName !== newProp.name) {
|
|
651
|
+
// 전체 엔티티에서 현재 수정된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 수정
|
|
652
|
+
const allEntityIds = EntityManager.getAllIds();
|
|
653
|
+
for (const relEntityId of allEntityIds){
|
|
654
|
+
const relEntity = EntityManager.get(relEntityId);
|
|
655
|
+
const relEntitySubsetKeys = Object.keys(relEntity.subsets);
|
|
656
|
+
for (const subsetKey of relEntitySubsetKeys){
|
|
657
|
+
const subset = relEntity.subsets[subsetKey];
|
|
658
|
+
// 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 수정 처리
|
|
659
|
+
const modifiedSubsetFields = subset.map((subsetField)=>{
|
|
660
|
+
const analyzed = relEntity.analyzeSubsetField(subsetField);
|
|
661
|
+
const modified = analyzed.map((a)=>a.propName === oldName && a.entityId === this.id ? {
|
|
662
|
+
...a,
|
|
663
|
+
propName: newProp.name
|
|
664
|
+
} : a);
|
|
665
|
+
// 분석한 필드를 다시 서브셋 필드로 복구
|
|
666
|
+
return modified.map((a)=>a.propName).join(".");
|
|
667
|
+
});
|
|
668
|
+
if (subset.join(",") !== modifiedSubsetFields.join(",")) {
|
|
669
|
+
relEntity.subsets[subsetKey] = modifiedSubsetFields;
|
|
670
|
+
entities.push(relEntity);
|
|
671
|
+
}
|
|
672
|
+
}
|
|
673
|
+
}
|
|
674
|
+
}
|
|
675
|
+
// 프롭 수정
|
|
676
|
+
this.props[at] = newProp;
|
|
677
|
+
await Promise.all(entities.map(async (entity)=>entity.save()));
|
|
678
|
+
}
|
|
679
|
+
async delProp(at) {
|
|
680
|
+
// 이전 프롭 이름 저장
|
|
681
|
+
const oldName = this.props[at].name;
|
|
682
|
+
// 저장할 엔티티
|
|
683
|
+
const entities = [
|
|
684
|
+
this
|
|
685
|
+
];
|
|
686
|
+
// 전체 엔티티에서 현재 삭제된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 제외
|
|
687
|
+
const allEntityIds = EntityManager.getAllIds();
|
|
688
|
+
for (const relEntityId of allEntityIds){
|
|
689
|
+
const relEntity = EntityManager.get(relEntityId);
|
|
690
|
+
const relEntitySubsetKeys = Object.keys(relEntity.subsets);
|
|
691
|
+
for (const subsetKey of relEntitySubsetKeys){
|
|
692
|
+
const subset = relEntity.subsets[subsetKey];
|
|
693
|
+
// 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 이후의 필드를 제외
|
|
694
|
+
const modifiedSubsetFields = subset.map((subsetField)=>{
|
|
695
|
+
const analyzed = relEntity.analyzeSubsetField(subsetField);
|
|
696
|
+
if (analyzed.find((a)=>a.propName === oldName && a.entityId === this.id)) {
|
|
697
|
+
return null;
|
|
698
|
+
} else {
|
|
699
|
+
return subsetField;
|
|
700
|
+
}
|
|
701
|
+
}).filter(nonNullable);
|
|
702
|
+
if (subset.join(",") !== modifiedSubsetFields.join(",")) {
|
|
703
|
+
relEntity.subsets[subsetKey] = modifiedSubsetFields;
|
|
704
|
+
entities.push(relEntity);
|
|
705
|
+
}
|
|
706
|
+
}
|
|
707
|
+
}
|
|
708
|
+
// 현재 엔티티의 인덱스에서 제외
|
|
709
|
+
for (const index of EntityManager.get(this.id).indexes){
|
|
710
|
+
index.columns = index.columns.filter((col)=>col !== oldName);
|
|
711
|
+
}
|
|
712
|
+
// 프롭 삭제
|
|
713
|
+
this.props.splice(at, 1);
|
|
714
|
+
await Promise.all(entities.map(async (entity)=>entity.save()));
|
|
715
|
+
}
|
|
716
|
+
getEntityIdFromSubsetField(subsetField) {
|
|
717
|
+
if (subsetField.includes(".") === false) {
|
|
718
|
+
return this.id;
|
|
719
|
+
}
|
|
720
|
+
// 서브셋 필드의 마지막은 프롭이므로 제외
|
|
721
|
+
const arr = subsetField.split(".").slice(0, -1);
|
|
722
|
+
// 서브셋 필드를 내려가면서 마지막으로 relation된 엔티티를 찾음
|
|
723
|
+
const lastEntityId = arr.reduce((entityId, field)=>{
|
|
724
|
+
const relProp = EntityManager.get(entityId).props.find((p)=>p.name === field);
|
|
725
|
+
if (!relProp || relProp.type !== "relation") {
|
|
726
|
+
console.debug({
|
|
727
|
+
arr,
|
|
728
|
+
thisId: this.id,
|
|
729
|
+
entityId,
|
|
730
|
+
field
|
|
731
|
+
});
|
|
732
|
+
throw new Error(`잘못된 서브셋키 ${subsetField}`);
|
|
733
|
+
}
|
|
734
|
+
return relProp.with;
|
|
735
|
+
}, this.id);
|
|
736
|
+
return lastEntityId;
|
|
737
|
+
}
|
|
738
|
+
async moveProp(at, to) {
|
|
739
|
+
const prop = this.props[at];
|
|
740
|
+
const newProps = [
|
|
741
|
+
...this.props
|
|
742
|
+
];
|
|
743
|
+
newProps.splice(to, 0, prop);
|
|
744
|
+
newProps.splice(at < to ? at : at + 1, 1);
|
|
745
|
+
this.props = newProps;
|
|
746
|
+
await this.save();
|
|
747
|
+
}
|
|
748
|
+
}
|
|
749
|
+
|
|
750
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9lbnRpdHkvZW50aXR5LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBhc3NlcnQgZnJvbSBcImFzc2VydFwiO1xuaW1wb3J0IHsgd3JpdGVGaWxlIH0gZnJvbSBcImZzL3Byb21pc2VzXCI7XG5pbXBvcnQgaW5mbGVjdGlvbiBmcm9tIFwiaW5mbGVjdGlvblwiO1xuaW1wb3J0IHBhdGggZnJvbSBcInBhdGhcIjtcbmltcG9ydCB7IGdyb3VwLCB1bmlxdWUgfSBmcm9tIFwicmFkYXNoaVwiO1xuaW1wb3J0IHsgeiB9IGZyb20gXCJ6b2RcIjtcbmltcG9ydCB7IFNvbmFtdSB9IGZyb20gXCIuLi9hcGkvc29uYW11XCI7XG5pbXBvcnQge1xuICB0eXBlIEVudGl0eUluZGV4LFxuICB0eXBlIEVudGl0eUpzb24sXG4gIHR5cGUgRW50aXR5UHJvcCxcbiAgdHlwZSBFbnRpdHlQcm9wTm9kZSxcbiAgdHlwZSBFbnRpdHlTdWJzZXRSb3csXG4gIGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wLFxuICBpc0VudW1Qcm9wLFxuICBpc0hhc01hbnlSZWxhdGlvblByb3AsXG4gIGlzTWFueVRvTWFueVJlbGF0aW9uUHJvcCxcbiAgaXNPbmVUb09uZVJlbGF0aW9uUHJvcCxcbiAgaXNSZWxhdGlvblByb3AsXG4gIGlzVmlydHVhbFByb3AsXG4gIHR5cGUgUmVsYXRpb25Qcm9wLFxuICB0eXBlIFN0cmluZ1Byb3AsXG4gIHR5cGUgU3Vic2V0UXVlcnksXG59IGZyb20gXCIuLi90eXBlcy90eXBlc1wiO1xuaW1wb3J0IHsgaW1wb3J0TWVtYmVycyB9IGZyb20gXCIuLi91dGlscy9lc20tdXRpbHNcIjtcbmltcG9ydCB7IGZvcm1hdENvZGUgfSBmcm9tIFwiLi4vdXRpbHMvZm9ybWF0dGVyXCI7XG5pbXBvcnQgeyBleGlzdHMgfSBmcm9tIFwiLi4vdXRpbHMvZnMtdXRpbHNcIjtcbmltcG9ydCB7IHJ1bnRpbWVQYXRoIH0gZnJvbSBcIi4uL3V0aWxzL3BhdGgtdXRpbHNcIjtcbmltcG9ydCB7IGFzc2VydERlZmluZWQsIG5vbk51bGxhYmxlIH0gZnJvbSBcIi4uL3V0aWxzL3V0aWxzXCI7XG5pbXBvcnQgeyBFbnRpdHlNYW5hZ2VyIH0gZnJvbSBcIi4vZW50aXR5LW1hbmFnZXJcIjtcblxuZXhwb3J0IGNsYXNzIEVudGl0eSB7XG4gIGlkOiBzdHJpbmc7XG4gIHBhcmVudElkPzogc3RyaW5nO1xuICB0YWJsZTogc3RyaW5nO1xuICB0aXRsZTogc3RyaW5nO1xuICBuYW1lczoge1xuICAgIHBhcmVudEZzOiBzdHJpbmc7XG4gICAgZnM6IHN0cmluZztcbiAgICBtb2R1bGU6IHN0cmluZztcbiAgfTtcbiAgcHJvcHM6IEVudGl0eVByb3BbXTtcbiAgcHJvcHNEaWN0OiB7XG4gICAgW2tleTogc3RyaW5nXTogRW50aXR5UHJvcDtcbiAgfTtcbiAgcmVsYXRpb25zOiB7XG4gICAgW2tleTogc3RyaW5nXTogUmVsYXRpb25Qcm9wO1xuICB9O1xuICBpbmRleGVzOiBFbnRpdHlJbmRleFtdO1xuICBzdWJzZXRzOiB7XG4gICAgW2tleTogc3RyaW5nXTogc3RyaW5nW107XG4gIH07XG4gIHR5cGVzOiB7XG4gICAgW25hbWU6IHN0cmluZ106IHouWm9kVHlwZUFueTtcbiAgfSA9IHt9O1xuICBlbnVtczoge1xuICAgIFtlbnVtSWQ6IHN0cmluZ106IHouWm9kRW51bTxSZWFkb25seTxSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+Pj47XG4gIH0gPSB7fTtcbiAgZW51bUxhYmVsczoge1xuICAgIFtlbnVtSWQ6IHN0cmluZ106IHtcbiAgICAgIFtrZXk6IHN0cmluZ106IHN0cmluZztcbiAgICB9O1xuICB9ID0ge307XG5cbiAgY29uc3RydWN0b3IoeyBpZCwgcGFyZW50SWQsIHRhYmxlLCB0aXRsZSwgcHJvcHMsIGluZGV4ZXMsIHN1YnNldHMsIGVudW1zIH06IEVudGl0eUpzb24pIHtcbiAgICAvLyBpZFxuICAgIHRoaXMuaWQgPSBpZDtcbiAgICB0aGlzLnBhcmVudElkID0gcGFyZW50SWQ7XG4gICAgdGhpcy50aXRsZSA9IHRpdGxlID8/IHRoaXMuaWQ7XG4gICAgdGhpcy50YWJsZSA9IHRhYmxlID8/IGluZmxlY3Rpb24udW5kZXJzY29yZShpbmZsZWN0aW9uLnBsdXJhbGl6ZShpZCkpO1xuXG4gICAgLy8gcHJvcHNcbiAgICBpZiAocHJvcHMpIHtcbiAgICAgIHRoaXMucHJvcHMgPSBwcm9wcy5tYXAoKHByb3ApID0+IHtcbiAgICAgICAgaWYgKGlzRW51bVByb3AocHJvcCkpIHtcbiAgICAgICAgICBpZiAocHJvcC5pZC5pbmNsdWRlcyhcIiRNb2RlbFwiKSkge1xuICAgICAgICAgICAgcHJvcC5pZCA9IHByb3AuaWQucmVwbGFjZShcIiRNb2RlbFwiLCBpZCk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiBwcm9wO1xuICAgICAgfSk7XG4gICAgICB0aGlzLnByb3BzRGljdCA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgICAgcHJvcHMubWFwKChwcm9wKSA9PiB7XG4gICAgICAgICAgcmV0dXJuIFtwcm9wLm5hbWUsIHByb3BdO1xuICAgICAgICB9KSxcbiAgICAgICk7XG5cbiAgICAgIC8vIHJlbGF0aW9uc1xuICAgICAgdGhpcy5yZWxhdGlvbnMgPSBPYmplY3QuZnJvbUVudHJpZXMoXG4gICAgICAgIHByb3BzLmZpbHRlcigocHJvcCkgPT4gaXNSZWxhdGlvblByb3AocHJvcCkpLm1hcCgocHJvcCkgPT4gW3Byb3AubmFtZSwgcHJvcF0pLFxuICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5wcm9wcyA9IFtdO1xuICAgICAgdGhpcy5wcm9wc0RpY3QgPSB7fTtcbiAgICAgIHRoaXMucmVsYXRpb25zID0ge307XG4gICAgfVxuXG4gICAgLy8gaW5kZXhlc1xuICAgIHRoaXMuaW5kZXhlcyA9IGluZGV4ZXMgPz8gW107XG5cbiAgICAvLyBzdWJzZXRzXG4gICAgdGhpcy5zdWJzZXRzID0gc3Vic2V0cyA/PyB7fTtcblxuICAgIC8vIGVudW1zXG4gICAgdGhpcy5lbnVtTGFiZWxzID0gZW51bXMgPz8ge307XG4gICAgdGhpcy5lbnVtcyA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgIE9iamVjdC5lbnRyaWVzKHRoaXMuZW51bUxhYmVscykubWFwKChba2V5LCBlbnVtTGFiZWxdKSA9PiB7XG4gICAgICAgIHJldHVybiBba2V5LCB6LmVudW0oT2JqZWN0LmtleXMoZW51bUxhYmVsKSBhcyB1bmtub3duIGFzIHJlYWRvbmx5IFtzdHJpbmcsIC4uLnN0cmluZ1tdXSldO1xuICAgICAgfSksXG4gICAgKTtcblxuICAgIC8vIG5hbWVzXG4gICAgdGhpcy5uYW1lcyA9IHtcbiAgICAgIHBhcmVudEZzOiBpbmZsZWN0aW9uLmRhc2hlcml6ZShpbmZsZWN0aW9uLnVuZGVyc2NvcmUocGFyZW50SWQgPz8gaWQpKS50b0xvd2VyQ2FzZSgpLFxuICAgICAgZnM6IGluZmxlY3Rpb24uZGFzaGVyaXplKGluZmxlY3Rpb24udW5kZXJzY29yZShpZCkpLnRvTG93ZXJDYXNlKCksXG4gICAgICBtb2R1bGU6IGlkLFxuICAgIH07XG4gIH1cblxuICAvKlxuICAgIHN1YnNldOydhCBQdXJpIOy9lOuTnOuhnCDrs4DtmZhcbiAgKi9cbiAgZ2V0UHVyaVN1YnNldFF1ZXJ5KHN1YnNldEtleTogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBjb25zdCBzdWJzZXQgPSB0aGlzLnN1YnNldHNbc3Vic2V0S2V5XTtcbiAgICBjb25zdCBzdWJzZXRRdWVyeSA9IHRoaXMucmVzb2x2ZVN1YnNldFF1ZXJ5KFwiXCIsIHN1YnNldCk7XG5cbiAgICBjb25zdCBsaW5lczogc3RyaW5nW10gPSBbXTtcblxuICAgIC8vIGZyb21cbiAgICBsaW5lcy5wdXNoKGByZXR1cm4gcWJXcmFwcGVyYCk7XG4gICAgbGluZXMucHVzaChgLmZyb20oXCIke3RoaXMudGFibGV9XCIpYCk7XG5cbiAgICAvLyBqb2luXG4gICAgZm9yIChjb25zdCBqb2luIG9mIHN1YnNldFF1ZXJ5LmpvaW5zKSB7XG4gICAgICBjb25zdCBqb2luTWV0aG9kID0gam9pbi5qb2luID09PSBcImlubmVyXCIgPyBcImpvaW5cIiA6IFwibGVmdEpvaW5cIjtcblxuICAgICAgaWYgKFwiY3VzdG9tXCIgaW4gam9pbikge1xuICAgICAgICAvLyBjdXN0b20gam9pbiBjbGF1c2XripQgcmF3IOyCrOyaqVxuICAgICAgICBsaW5lcy5wdXNoKFxuICAgICAgICAgIGAuJHtqb2luTWV0aG9kfSh7ICR7am9pbi5hc306IFwiJHtqb2luLnRhYmxlfVwiIH0sIHFiV3JhcHBlci5rbmV4LnJhdyhcXGAke2pvaW4uY3VzdG9tfVxcYCkpYCxcbiAgICAgICAgKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGxpbmVzLnB1c2goYC4ke2pvaW5NZXRob2R9KHsgJHtqb2luLmFzfTogXCIke2pvaW4udGFibGV9XCIgfSwgXCIke2pvaW4uZnJvbX1cIiwgXCIke2pvaW4udG99XCIpYCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gc2VsZWN0XG4gICAgY29uc3Qgc2VsZWN0T2JqOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge307XG4gICAgZm9yIChjb25zdCBzZWxlY3RJdGVtIG9mIHN1YnNldFF1ZXJ5LnNlbGVjdCkge1xuICAgICAgLy8gXCJ1c2Vycy5pZFwiIOuYkOuKlCBcInVzZXJzLmlkIGFzIHVzZXJfX2lkXCIg7ZiV7YOcXG4gICAgICBjb25zdCBtYXRjaCA9IHNlbGVjdEl0ZW0ubWF0Y2goL14oLis/KSg/OiBhcyAoLispKT8kLyk7XG4gICAgICBpZiAobWF0Y2gpIHtcbiAgICAgICAgY29uc3QgWywgY29sdW1uLCBhbGlhc10gPSBtYXRjaDtcbiAgICAgICAgY29uc3Qga2V5ID0gYWxpYXMgPz8gYXNzZXJ0RGVmaW5lZChjb2x1bW4uc3BsaXQoXCIuXCIpLnBvcCgpKTtcbiAgICAgICAgc2VsZWN0T2JqW2tleV0gPSBgXCIke2NvbHVtbi50cmltKCl9XCJgO1xuICAgICAgfVxuICAgIH1cblxuICAgIGxpbmVzLnB1c2goYC5zZWxlY3Qoe2ApO1xuICAgIE9iamVjdC5lbnRyaWVzKHNlbGVjdE9iaikuZm9yRWFjaCgoW2tleSwgdmFsdWVdKSA9PiB7XG4gICAgICBsaW5lcy5wdXNoKGAke2tleX06ICR7dmFsdWV9LGApO1xuICAgIH0pO1xuICAgIGxpbmVzLnB1c2goYH0pO2ApO1xuXG4gICAgcmV0dXJuIGxpbmVzLmpvaW4oXCJcXG5cIik7XG4gIH1cblxuICBnZXRQdXJpTG9hZGVyUXVlcnkoc3Vic2V0S2V5OiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIGNvbnN0IHN1YnNldCA9IHRoaXMuc3Vic2V0c1tzdWJzZXRLZXldO1xuICAgIGNvbnN0IHsgbG9hZGVycyB9ID0gdGhpcy5yZXNvbHZlU3Vic2V0UXVlcnkoXCJcIiwgc3Vic2V0KTtcblxuICAgIGNvbnN0IGxpbmVzOiBzdHJpbmdbXSA9IFtgW2BdO1xuXG4gICAgY29uc3QgcGFyc2VTZWxlY3QgPSAoc2VsZWN0OiBzdHJpbmcsIHRhYmxlOiBzdHJpbmcpID0+IHtcbiAgICAgIGNvbnN0IHRhYmxlUHJlZml4ID0gYCR7dGFibGV9LmA7XG4gICAgICBpZiAoc2VsZWN0LnN0YXJ0c1dpdGgodGFibGVQcmVmaXgpKSB7XG4gICAgICAgIHJldHVybiBgJHtzZWxlY3QucmVwbGFjZSh0YWJsZVByZWZpeCwgXCJcIil9OiBcIiR7c2VsZWN0fVwiYDtcbiAgICAgIH1cblxuICAgICAgaWYgKHNlbGVjdC5pbmNsdWRlcyhcIiBhcyBcIikpIHtcbiAgICAgICAgY29uc3QgW2NvbHVtbiwgYWxpYXNdID0gc2VsZWN0LnNwbGl0KFwiIGFzIFwiKTtcbiAgICAgICAgcmV0dXJuIGAke2FsaWFzfTogXCIke2NvbHVtbn1cImA7XG4gICAgICB9XG4gICAgICByZXR1cm4gYCR7c2VsZWN0fTogXCIke3NlbGVjdH1cImA7XG4gICAgfTtcblxuICAgIC8vIOyerOq3gOyggeycvOuhnCBsb2FkZXIg7IOd7ISx7ZWY64qUIO2XrO2NvCDtlajsiJhcbiAgICBjb25zdCBnZW5lcmF0ZUxvYWRlckNvZGUgPSAobG9hZGVyczogU3Vic2V0UXVlcnlbXCJsb2FkZXJzXCJdKTogc3RyaW5nW10gPT4ge1xuICAgICAgY29uc3QgbG9hZGVyTGluZXM6IHN0cmluZ1tdID0gW107XG5cbiAgICAgIGZvciAoY29uc3QgbG9hZGVyIG9mIGxvYWRlcnMpIHtcbiAgICAgICAgY29uc3QgeyB0b1RhYmxlLCB0b0NvbCwgdGhyb3VnaCB9ID0gbG9hZGVyLm1hbnlKb2luO1xuICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFxuICAgICAgICAgIFwie1wiLFxuICAgICAgICAgIGBhczogXCIke2xvYWRlci5hc31cIixgLFxuICAgICAgICAgIGByZWZJZDogXCIke2xvYWRlci5tYW55Sm9pbi5pZEZpZWxkfVwiLGAsXG4gICAgICAgICAgYHFiOiAocWJXcmFwcGVyOiBQdXJpV3JhcHBlcjxEYXRhYmFzZVNjaGVtYUV4dGVuZD4sIGZyb21JZHM6IG51bWJlcltdKSA9PiB7YCxcbiAgICAgICAgKTtcblxuICAgICAgICBpZiAodGhyb3VnaCA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgLy8gSGFzTWFueVxuICAgICAgICAgIGxvYWRlckxpbmVzLnB1c2goXG4gICAgICAgICAgICAvL1xuICAgICAgICAgICAgXCJyZXR1cm4gcWJXcmFwcGVyXCIsXG4gICAgICAgICAgICBgLmZyb20oXCIke3RvVGFibGV9XCIpYCxcbiAgICAgICAgICApO1xuXG4gICAgICAgICAgbG9hZGVyLm9uZUpvaW5zLmZvckVhY2goKGpvaW46IFN1YnNldFF1ZXJ5W1wiam9pbnNcIl1bbnVtYmVyXSkgPT4ge1xuICAgICAgICAgICAgY29uc3Qgam9pblR5cGUgPSBqb2luLmpvaW4gPT09IFwiaW5uZXJcIiA/IFwiam9pblwiIDogXCJsZWZ0Sm9pblwiO1xuICAgICAgICAgICAgaWYgKFwiY3VzdG9tXCIgaW4gam9pbikge1xuICAgICAgICAgICAgICAvLyBGSVhNRTog6rKA7KadIO2VhOyalFxuICAgICAgICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFxuICAgICAgICAgICAgICAgIGAuJHtqb2luVHlwZX0oeyAke2pvaW4uYXN9OiBcIiR7am9pbi50YWJsZX1cIiB9LCAoaikgPT4ge2AsXG4gICAgICAgICAgICAgICAgYGoub24oUHVyaS5yYXdTdHJpbmcoXCIke2pvaW4uY3VzdG9tfVwiKSk7YCxcbiAgICAgICAgICAgICAgICBcIn0pXCIsXG4gICAgICAgICAgICAgICk7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFxuICAgICAgICAgICAgICAgIGAuJHtqb2luVHlwZX0oeyAke2pvaW4uYXN9OiBcIiR7am9pbi50YWJsZX1cIiB9LCBcIiR7am9pbi5mcm9tfVwiLCBcIiR7am9pbi50b31cIilgLFxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgbG9hZGVyTGluZXMucHVzaChcbiAgICAgICAgICAgIGAud2hlcmVJbihcIiR7dG9UYWJsZX0uJHt0b0NvbH1cIiwgZnJvbUlkcylgLFxuICAgICAgICAgICAgYC5zZWxlY3Qoe2AsXG4gICAgICAgICAgICBgJHtsb2FkZXIuc2VsZWN0Lm1hcCgoc2VsZWN0OiBzdHJpbmcpID0+IHBhcnNlU2VsZWN0KHNlbGVjdCwgdG9UYWJsZSkpLmpvaW4oXCIsXCIpfSxgLFxuICAgICAgICAgICAgYHJlZklkOiBcIiR7dG9UYWJsZX0uJHt0b0NvbH1cIixgLFxuICAgICAgICAgICAgYH0pO2AsXG4gICAgICAgICAgKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAvLyBNYW55VG9NYW55XG4gICAgICAgICAgbG9hZGVyTGluZXMucHVzaChcbiAgICAgICAgICAgIFwicmV0dXJuIHFiV3JhcHBlclwiLFxuICAgICAgICAgICAgYC5mcm9tKFwiJHt0aHJvdWdoLnRhYmxlfVwiKWAsXG4gICAgICAgICAgICBgLmpvaW4oXCIke3RvVGFibGV9XCIsIFwiJHt0aHJvdWdoLnRhYmxlfS4ke3Rocm91Z2gudG9Db2x9XCIsIFwiJHt0b1RhYmxlfS4ke3RvQ29sfVwiKWAsXG4gICAgICAgICAgKTtcblxuICAgICAgICAgIGxvYWRlci5vbmVKb2lucy5mb3JFYWNoKChqb2luOiBTdWJzZXRRdWVyeVtcImpvaW5zXCJdW251bWJlcl0pID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGpvaW5UeXBlID0gam9pbi5qb2luID09PSBcImlubmVyXCIgPyBcImpvaW5cIiA6IFwibGVmdEpvaW5cIjtcbiAgICAgICAgICAgIGlmIChcImN1c3RvbVwiIGluIGpvaW4pIHtcbiAgICAgICAgICAgICAgLy8gRklYTUU6IOqygOymnSDtlYTsmpRcbiAgICAgICAgICAgICAgbG9hZGVyTGluZXMucHVzaChcbiAgICAgICAgICAgICAgICBgLiR7am9pblR5cGV9KHsgJHtqb2luLmFzfTogXCIke2pvaW4udGFibGV9XCIgfSwgKGopID0+IHtgLFxuICAgICAgICAgICAgICAgIGBqLm9uKFB1cmkucmF3U3RyaW5nKFwiJHtqb2luLmN1c3RvbX1cIikpO2AsXG4gICAgICAgICAgICAgICAgXCJ9KVwiLFxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgbG9hZGVyTGluZXMucHVzaChcbiAgICAgICAgICAgICAgICBgLiR7am9pblR5cGV9KHsgJHtqb2luLmFzfTogXCIke2pvaW4udGFibGV9XCIgfSwgXCIke2pvaW4uZnJvbX1cIiwgXCIke2pvaW4udG99XCIpYCxcbiAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KTtcbiAgICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFxuICAgICAgICAgICAgYC53aGVyZUluKFwiJHt0aHJvdWdoLnRhYmxlfS4ke3Rocm91Z2guZnJvbUNvbH1cIiwgZnJvbUlkcylgLFxuICAgICAgICAgICAgYC5zZWxlY3Qoe2AsXG4gICAgICAgICAgICBgJHtsb2FkZXIuc2VsZWN0Lm1hcCgoc2VsZWN0OiBzdHJpbmcpID0+IHBhcnNlU2VsZWN0KHNlbGVjdCwgdG9UYWJsZSkpLmpvaW4oXCIsXCIpfSxgLFxuICAgICAgICAgICAgYHJlZklkOiBcIiR7dGhyb3VnaC50YWJsZX0uJHt0aHJvdWdoLmZyb21Db2x9XCIsYCxcbiAgICAgICAgICAgIGB9KTtgLFxuICAgICAgICAgICk7XG4gICAgICAgIH1cblxuICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKGB9LGApO1xuXG4gICAgICAgIC8vIOykkeyyqSBsb2FkZXJzIOyymOumrFxuICAgICAgICBpZiAobG9hZGVyLmxvYWRlcnMgJiYgbG9hZGVyLmxvYWRlcnMubGVuZ3RoID4gMCkge1xuICAgICAgICAgIGxvYWRlckxpbmVzLnB1c2goXCJsb2FkZXJzOiBbXCIsIC4uLmdlbmVyYXRlTG9hZGVyQ29kZShsb2FkZXIubG9hZGVycyksIFwiXSxcIik7XG4gICAgICAgIH1cblxuICAgICAgICBsb2FkZXJMaW5lcy5wdXNoKFwifSxcIik7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiBsb2FkZXJMaW5lcztcbiAgICB9O1xuXG4gICAgbGluZXMucHVzaCguLi5nZW5lcmF0ZUxvYWRlckNvZGUobG9hZGVycykpO1xuICAgIGxpbmVzLnB1c2goYF1gKTtcblxuICAgIHJldHVybiBsaW5lcy5qb2luKFwiXFxuXCIpO1xuICB9XG5cbiAgLypcbiAgICBzdWJzZXQgU0VMRUNUL0pPSU4vTE9BREVSIOqysOqzvCDrpqzthLRcbiAgKi9cbiAgZ2V0U3Vic2V0UXVlcnkoc3Vic2V0S2V5OiBzdHJpbmcpOiBTdWJzZXRRdWVyeSB7XG4gICAgY29uc3Qgc3Vic2V0ID0gdGhpcy5zdWJzZXRzW3N1YnNldEtleV07XG5cbiAgICBjb25zdCByZXN1bHQ6IFN1YnNldFF1ZXJ5ID0gdGhpcy5yZXNvbHZlU3Vic2V0UXVlcnkoXCJcIiwgc3Vic2V0KTtcbiAgICByZXR1cm4gcmVzdWx0O1xuICB9XG5cbiAgLypcbiAgICovXG4gIHJlc29sdmVTdWJzZXRRdWVyeShcbiAgICBwcmVmaXg6IHN0cmluZyxcbiAgICBmaWVsZHM6IHN0cmluZ1tdLFxuICAgIGlzQWxyZWFkeU91dGVySm9pbmVkOiBib29sZWFuID0gZmFsc2UsXG4gICk6IFN1YnNldFF1ZXJ5IHtcbiAgICAvLyBwcmVmaXgg7LmY7ZmYIChwcmVmaXjripQgVG9PbmVSZWxhdGlvbuydtCDrs7XsiJjroZwg67aZ7J2AIOqyveyasCDrqqjrkZAgX1/roZwg67OA6rK965CoKVxuICAgIHByZWZpeCA9IHByZWZpeC5yZXBsYWNlKC9cXC4vZywgXCJfX1wiKTtcblxuICAgIC8vIOyEnOu4jOyFi+ydhCAx646B7Iqk66eMIOu2hOumrO2VmOyXrCDqt7jro7ntlZFcbiAgICBjb25zdCBzdWJzZXRHcm91cCA9IGdyb3VwKGZpZWxkcywgKGZpZWxkKSA9PiB7XG4gICAgICBpZiAoZmllbGQuaW5jbHVkZXMoXCIuXCIpKSB7XG4gICAgICAgIGNvbnN0IFtyZWxdID0gZmllbGQuc3BsaXQoXCIuXCIpO1xuICAgICAgICByZXR1cm4gcmVsO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgcmV0dXJuIFwiXCI7XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICBjb25zdCByZXN1bHQgPSBPYmplY3Qua2V5cyhzdWJzZXRHcm91cCkucmVkdWNlKFxuICAgICAgKHIsIGdyb3VwS2V5KSA9PiB7XG4gICAgICAgIGNvbnN0IGZpZWxkcyA9IHN1YnNldEdyb3VwW2dyb3VwS2V5XTtcbiAgICAgICAgYXNzZXJ0KGZpZWxkcyAhPT0gdW5kZWZpbmVkLCBcImZpZWxkcyBpcyB1bmRlZmluZWRcIik7XG5cbiAgICAgICAgLy8g7ZiE7J6sIO2FjOydtOu4lCDtlYTrk5zshYvsnYAgc2VsZWN0LCB2aXJ0dWFs7JeQIOy2lOqwgO2VmOqzoCDrpqzthLRcbiAgICAgICAgaWYgKGdyb3VwS2V5ID09PSBcIlwiKSB7XG4gICAgICAgICAgY29uc3QgcmVhbEZpZWxkcyA9IGZpZWxkcy5maWx0ZXIoKGZpZWxkKSA9PiAhaXNWaXJ0dWFsUHJvcCh0aGlzLnByb3BzRGljdFtmaWVsZF0pKTtcbiAgICAgICAgICBjb25zdCB2aXJ0dWFsRmllbGRzID0gZmllbGRzLmZpbHRlcigoZmllbGQpID0+IGlzVmlydHVhbFByb3AodGhpcy5wcm9wc0RpY3RbZmllbGRdKSk7XG5cbiAgICAgICAgICBpZiAocHJlZml4ID09PSBcIlwiKSB7XG4gICAgICAgICAgICAvLyDtmITsnqwg7YWM7J2067iU7J24IOqyveyasFxuICAgICAgICAgICAgci5zZWxlY3QgPSByLnNlbGVjdC5jb25jYXQocmVhbEZpZWxkcy5tYXAoKGZpZWxkKSA9PiBgJHt0aGlzLnRhYmxlfS4ke2ZpZWxkfWApKTtcbiAgICAgICAgICAgIHIudmlydHVhbCA9IHIudmlydHVhbC5jb25jYXQodmlydHVhbEZpZWxkcyk7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIC8vIOuEmOyWtOyYqCDthYzsnbTruJTsnbgg6rK97JqwXG4gICAgICAgICAgICByLnNlbGVjdCA9IHIuc2VsZWN0LmNvbmNhdChcbiAgICAgICAgICAgICAgcmVhbEZpZWxkcy5tYXAoKGZpZWxkKSA9PiBgJHtwcmVmaXh9LiR7ZmllbGR9IGFzICR7cHJlZml4fV9fJHtmaWVsZH1gKSxcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgcmV0dXJuIHI7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCByZWxhdGlvbiA9IHRoaXMucmVsYXRpb25zW2dyb3VwS2V5XTtcbiAgICAgICAgaWYgKHJlbGF0aW9uID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYOyhtOyerO2VmOyngCDslYrripQgcmVsYXRpb24g7LC47KGwICR7Z3JvdXBLZXl9YCk7XG4gICAgICAgIH1cbiAgICAgICAgY29uc3QgcmVsRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocmVsYXRpb24ud2l0aCk7XG5cbiAgICAgICAgaWYgKGlzT25lVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pIHx8IGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wKHJlbGF0aW9uKSkge1xuICAgICAgICAgIC8vIC1PbmUgUmVsYXRpb246IEpPSU4g7Jy866GcIOyymOumrFxuICAgICAgICAgIGNvbnN0IHJlbEZpZWxkcyA9IGZpZWxkcy5tYXAoKGZpZWxkKSA9PiBmaWVsZC5zcGxpdChcIi5cIikuc2xpY2UoMSkuam9pbihcIi5cIikpO1xuXG4gICAgICAgICAgLy8gLU9uZSBSZWxhdGlvbuyXkOyEnCBpZCDtlYTrk5zrp4wg7LC47KGw7ZWY64qUIOqyveyasCDrprTroIjsnbTshZgg64SY6riw7KeAIOyViuqzoCDrpqzthLRcbiAgICAgICAgICBpZiAocmVsRmllbGRzLmxlbmd0aCA9PT0gMSAmJiByZWxGaWVsZHNbMF0gPT09IFwiaWRcIikge1xuICAgICAgICAgICAgaWYgKHByZWZpeCA9PT0gXCJcIikge1xuICAgICAgICAgICAgICByLnNlbGVjdCA9IHIuc2VsZWN0LmNvbmNhdChgJHt0aGlzLnRhYmxlfS4ke2dyb3VwS2V5fV9pZGApO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgci5zZWxlY3QgPSByLnNlbGVjdC5jb25jYXQoYCR7cHJlZml4fS4ke2dyb3VwS2V5fV9pZCBhcyAke3ByZWZpeH1fXyR7Z3JvdXBLZXl9X2lkYCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gcjtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBpbm5lck9yT3V0ZXJcbiAgICAgICAgICBjb25zdCBpbm5lck9yT3V0ZXIgPSAoKCkgPT4ge1xuICAgICAgICAgICAgaWYgKGlzQWxyZWFkeU91dGVySm9pbmVkKSB7XG4gICAgICAgICAgICAgIHJldHVybiBcIm91dGVyXCI7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmIChpc09uZVRvT25lUmVsYXRpb25Qcm9wKHJlbGF0aW9uKSkge1xuICAgICAgICAgICAgICBpZiAocmVsYXRpb24uaGFzSm9pbkNvbHVtbiA9PT0gdHJ1ZSAmJiAocmVsYXRpb24ubnVsbGFibGUgPz8gZmFsc2UpID09PSBmYWxzZSkge1xuICAgICAgICAgICAgICAgIHJldHVybiBcImlubmVyXCI7XG4gICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIFwib3V0ZXJcIjtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgaWYgKHJlbGF0aW9uLm51bGxhYmxlKSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIFwib3V0ZXJcIjtcbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICByZXR1cm4gXCJpbm5lclwiO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSkoKTtcbiAgICAgICAgICBjb25zdCByZWxTdWJzZXRRdWVyeSA9IHJlbEVudGl0eS5yZXNvbHZlU3Vic2V0UXVlcnkoXG4gICAgICAgICAgICBgJHtwcmVmaXggIT09IFwiXCIgPyBgJHtwcmVmaXh9LmAgOiBcIlwifSR7Z3JvdXBLZXl9YCxcbiAgICAgICAgICAgIHJlbEZpZWxkcyxcbiAgICAgICAgICAgIGlubmVyT3JPdXRlciA9PT0gXCJvdXRlclwiLFxuICAgICAgICAgICk7XG4gICAgICAgICAgci5zZWxlY3QgPSByLnNlbGVjdC5jb25jYXQocmVsU3Vic2V0UXVlcnkuc2VsZWN0KTtcbiAgICAgICAgICByLnZpcnR1YWwgPSByLnZpcnR1YWwuY29uY2F0KHJlbFN1YnNldFF1ZXJ5LnZpcnR1YWwpO1xuXG4gICAgICAgICAgY29uc3Qgam9pbkFzID0gcHJlZml4ID09PSBcIlwiID8gZ3JvdXBLZXkgOiBgJHtwcmVmaXh9X18ke2dyb3VwS2V5fWA7XG4gICAgICAgICAgY29uc3QgZnJvbVRhYmxlID0gcHJlZml4ID09PSBcIlwiID8gdGhpcy50YWJsZSA6IHByZWZpeDtcblxuICAgICAgICAgIGxldCBqb2luQ2xhdXNlOlxuICAgICAgICAgICAgfCB7XG4gICAgICAgICAgICAgICAgZnJvbTogc3RyaW5nO1xuICAgICAgICAgICAgICAgIHRvOiBzdHJpbmc7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHwge1xuICAgICAgICAgICAgICAgIGN1c3RvbTogc3RyaW5nO1xuICAgICAgICAgICAgICB9O1xuICAgICAgICAgIGlmIChyZWxhdGlvbi5jdXN0b21Kb2luQ2xhdXNlKSB7XG4gICAgICAgICAgICBqb2luQ2xhdXNlID0ge1xuICAgICAgICAgICAgICBjdXN0b206IHJlbGF0aW9uLmN1c3RvbUpvaW5DbGF1c2UsXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBsZXQgZnJvbTogc3RyaW5nLCB0bzogc3RyaW5nO1xuICAgICAgICAgICAgaWYgKGlzT25lVG9PbmVSZWxhdGlvblByb3AocmVsYXRpb24pKSB7XG4gICAgICAgICAgICAgIGlmIChyZWxhdGlvbi5oYXNKb2luQ29sdW1uKSB7XG4gICAgICAgICAgICAgICAgZnJvbSA9IGAke2Zyb21UYWJsZX0uJHtyZWxhdGlvbi5uYW1lfV9pZGA7XG4gICAgICAgICAgICAgICAgdG8gPSBgJHtqb2luQXN9LmlkYDtcbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICBmcm9tID0gYCR7ZnJvbVRhYmxlfS5pZGA7XG4gICAgICAgICAgICAgICAgdG8gPSBgJHtqb2luQXN9LiR7aW5mbGVjdGlvbi51bmRlcnNjb3JlKHRoaXMubmFtZXMuZnMucmVwbGFjZSgvLS9nLCBcIl9cIikpfV9pZGA7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIGZyb20gPSBgJHtmcm9tVGFibGV9LiR7cmVsYXRpb24ubmFtZX1faWRgO1xuICAgICAgICAgICAgICB0byA9IGAke2pvaW5Bc30uaWRgO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgam9pbkNsYXVzZSA9IHtcbiAgICAgICAgICAgICAgZnJvbSxcbiAgICAgICAgICAgICAgdG8sXG4gICAgICAgICAgICB9O1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHIuam9pbnMucHVzaCh7XG4gICAgICAgICAgICBhczogam9pbkFzLFxuICAgICAgICAgICAgam9pbjogaW5uZXJPck91dGVyLFxuICAgICAgICAgICAgdGFibGU6IHJlbEVudGl0eS50YWJsZSxcbiAgICAgICAgICAgIC4uLmpvaW5DbGF1c2UsXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICAvLyBCZWxvbmdzVG9PbmUg67CR7JeQIEhhc01hbnnqsIAg67aZ7J2AIOqyveyasFxuICAgICAgICAgIGlmIChyZWxTdWJzZXRRdWVyeS5sb2FkZXJzLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgIGNvbnN0IGNvbnZlcnRlZExvYWRlcnMgPSByZWxTdWJzZXRRdWVyeS5sb2FkZXJzLm1hcCgobG9hZGVyKSA9PiB7XG4gICAgICAgICAgICAgIGNvbnN0IG5ld0FzID0gW2dyb3VwS2V5LCBsb2FkZXIuYXNdLmpvaW4oXCJfX1wiKTtcbiAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICBhczogbmV3QXMsXG4gICAgICAgICAgICAgICAgdGFibGU6IGxvYWRlci50YWJsZSxcbiAgICAgICAgICAgICAgICBtYW55Sm9pbjogbG9hZGVyLm1hbnlKb2luLFxuICAgICAgICAgICAgICAgIG9uZUpvaW5zOiBsb2FkZXIub25lSm9pbnMsXG4gICAgICAgICAgICAgICAgc2VsZWN0OiBsb2FkZXIuc2VsZWN0LFxuICAgICAgICAgICAgICAgIGxvYWRlcnM6IGxvYWRlci5sb2FkZXJzLFxuICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgIHIubG9hZGVycyA9IFsuLi5yLmxvYWRlcnMsIC4uLmNvbnZlcnRlZExvYWRlcnNdO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHIuam9pbnMgPSByLmpvaW5zLmNvbmNhdChyZWxTdWJzZXRRdWVyeS5qb2lucyk7XG4gICAgICAgIH0gZWxzZSBpZiAoaXNIYXNNYW55UmVsYXRpb25Qcm9wKHJlbGF0aW9uKSB8fCBpc01hbnlUb01hbnlSZWxhdGlvblByb3AocmVsYXRpb24pKSB7XG4gICAgICAgICAgLy8gLU1hbnkgUmVsYXRpb246IExvYWRlciDroZwg7LKY66asXG4gICAgICAgICAgY29uc3QgcmVsRmllbGRzID0gZmllbGRzLm1hcCgoZmllbGQpID0+IGZpZWxkLnNwbGl0KFwiLlwiKS5zbGljZSgxKS5qb2luKFwiLlwiKSk7XG4gICAgICAgICAgY29uc3QgcmVsU3Vic2V0UXVlcnkgPSByZWxFbnRpdHkucmVzb2x2ZVN1YnNldFF1ZXJ5KFwiXCIsIHJlbEZpZWxkcyk7XG5cbiAgICAgICAgICBsZXQgbWFueUpvaW46IFN1YnNldFF1ZXJ5W1wibG9hZGVyc1wiXVtudW1iZXJdW1wibWFueUpvaW5cIl07XG4gICAgICAgICAgaWYgKGlzSGFzTWFueVJlbGF0aW9uUHJvcChyZWxhdGlvbikpIHtcbiAgICAgICAgICAgIGNvbnN0IGZyb21Db2wgPSByZWxhdGlvbj8uZnJvbUNvbHVtbiA/PyBcImlkXCI7XG4gICAgICAgICAgICBtYW55Sm9pbiA9IHtcbiAgICAgICAgICAgICAgZnJvbVRhYmxlOiB0aGlzLnRhYmxlLFxuICAgICAgICAgICAgICBmcm9tQ29sLFxuICAgICAgICAgICAgICBpZEZpZWxkOiBwcmVmaXggPT09IFwiXCIgPyBgJHtmcm9tQ29sfWAgOiBgJHtwcmVmaXh9X18ke2Zyb21Db2x9YCxcbiAgICAgICAgICAgICAgdG9UYWJsZTogcmVsRW50aXR5LnRhYmxlLFxuICAgICAgICAgICAgICB0b0NvbDogcmVsYXRpb24uam9pbkNvbHVtbixcbiAgICAgICAgICAgIH07XG4gICAgICAgICAgfSBlbHNlIGlmIChpc01hbnlUb01hbnlSZWxhdGlvblByb3AocmVsYXRpb24pKSB7XG4gICAgICAgICAgICBtYW55Sm9pbiA9IHtcbiAgICAgICAgICAgICAgZnJvbVRhYmxlOiB0aGlzLnRhYmxlLFxuICAgICAgICAgICAgICBmcm9tQ29sOiBcImlkXCIsXG4gICAgICAgICAgICAgIGlkRmllbGQ6IHByZWZpeCA9PT0gXCJcIiA/IGBpZGAgOiBgJHtwcmVmaXh9X19pZGAsXG4gICAgICAgICAgICAgIHRocm91Z2g6IHtcbiAgICAgICAgICAgICAgICB0YWJsZTogcmVsYXRpb24uam9pblRhYmxlLFxuICAgICAgICAgICAgICAgIGZyb21Db2w6IGAke2luZmxlY3Rpb24uc2luZ3VsYXJpemUodGhpcy50YWJsZSl9X2lkYCxcbiAgICAgICAgICAgICAgICB0b0NvbDogYCR7aW5mbGVjdGlvbi5zaW5ndWxhcml6ZShyZWxFbnRpdHkudGFibGUpfV9pZGAsXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgIHRvVGFibGU6IHJlbEVudGl0eS50YWJsZSxcbiAgICAgICAgICAgICAgdG9Db2w6IFwiaWRcIixcbiAgICAgICAgICAgIH07XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcigpO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHIubG9hZGVycy5wdXNoKHtcbiAgICAgICAgICAgIGFzOiBncm91cEtleSxcbiAgICAgICAgICAgIHRhYmxlOiByZWxFbnRpdHkudGFibGUsXG4gICAgICAgICAgICBtYW55Sm9pbixcbiAgICAgICAgICAgIG9uZUpvaW5zOiByZWxTdWJzZXRRdWVyeS5qb2lucyxcbiAgICAgICAgICAgIHNlbGVjdDogcmVsU3Vic2V0UXVlcnkuc2VsZWN0LFxuICAgICAgICAgICAgbG9hZGVyczogcmVsU3Vic2V0UXVlcnkubG9hZGVycyxcbiAgICAgICAgICB9KTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiByO1xuICAgICAgfSxcbiAgICAgIHtcbiAgICAgICAgc2VsZWN0OiBbXSxcbiAgICAgICAgdmlydHVhbDogW10sXG4gICAgICAgIGpvaW5zOiBbXSxcbiAgICAgICAgbG9hZGVyczogW10sXG4gICAgICB9IGFzIFN1YnNldFF1ZXJ5LFxuICAgICk7XG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gIC8qXG4gICAgRmllbGRFeHByW10g7J2EIEVudGl0eVByb3BOb2RlW10g66GcIOuzgO2ZmFxuICAqL1xuICBmaWVsZEV4cHJzVG9Qcm9wTm9kZXMoZmllbGRFeHByczogc3RyaW5nW10sIGVudGl0eTogRW50aXR5ID0gdGhpcyk6IEVudGl0eVByb3BOb2RlW10ge1xuICAgIGNvbnN0IGdyb3VwcyA9IGZpZWxkRXhwcnMucmVkdWNlKFxuICAgICAgKHJlc3VsdCwgZmllbGRFeHByKSA9PiB7XG4gICAgICAgIGxldCBrZXk6IHN0cmluZywgdmFsdWU6IHN0cmluZywgZWxzZUV4cHI6IHN0cmluZ1tdO1xuICAgICAgICBpZiAoZmllbGRFeHByLmluY2x1ZGVzKFwiLlwiKSkge1xuICAgICAgICAgIFtrZXksIC4uLmVsc2VFeHByXSA9IGZpZWxkRXhwci5zcGxpdChcIi5cIik7XG4gICAgICAgICAgdmFsdWUgPSBlbHNlRXhwci5qb2luKFwiLlwiKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBrZXkgPSBcIlwiO1xuICAgICAgICAgIHZhbHVlID0gZmllbGRFeHByO1xuICAgICAgICB9XG4gICAgICAgIHJlc3VsdFtrZXldID0gKHJlc3VsdFtrZXldID8/IFtdKS5jb25jYXQodmFsdWUpO1xuXG4gICAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgICB9LFxuICAgICAge30gYXMge1xuICAgICAgICBbazogc3RyaW5nXTogc3RyaW5nW107XG4gICAgICB9LFxuICAgICk7XG5cbiAgICByZXR1cm4gT2JqZWN0LmtleXMoZ3JvdXBzKS5mbGF0TWFwKChrZXkpID0+IHtcbiAgICAgIGNvbnN0IGdyb3VwID0gZ3JvdXBzW2tleV07XG5cbiAgICAgIC8vIOydvOuwmCBwcm9wIOyymOumrFxuICAgICAgaWYgKGtleSA9PT0gXCJcIikge1xuICAgICAgICByZXR1cm4gZ3JvdXAubWFwKChwcm9wTmFtZSkgPT4ge1xuICAgICAgICAgIC8vIHV1aWQg6rCc67OEIOyymOumrFxuICAgICAgICAgIGlmIChwcm9wTmFtZSA9PT0gXCJ1dWlkXCIpIHtcbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIG5vZGVUeXBlOiBcInBsYWluXCIgYXMgY29uc3QsXG4gICAgICAgICAgICAgIHByb3A6IHtcbiAgICAgICAgICAgICAgICB0eXBlOiBcInN0cmluZ1wiLFxuICAgICAgICAgICAgICAgIG5hbWU6IFwidXVpZFwiLFxuICAgICAgICAgICAgICAgIGxlbmd0aDogMTI4LFxuICAgICAgICAgICAgICB9IGFzIFN0cmluZ1Byb3AsXG4gICAgICAgICAgICAgIGNoaWxkcmVuOiBbXSxcbiAgICAgICAgICAgIH0gYXMgRW50aXR5UHJvcE5vZGU7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgY29uc3QgcHJvcCA9IGVudGl0eS5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IHByb3BOYW1lKTtcbiAgICAgICAgICBpZiAocHJvcCA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICBjb25zb2xlLmxvZyh7IHByb3BOYW1lLCBncm91cHMgfSk7XG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7ZW50aXR5LmlkfSAtLSDsnpjrqrvrkJwgRmllbGRFeHByICR7cHJvcE5hbWV9YCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBub2RlVHlwZTogXCJwbGFpblwiIGFzIGNvbnN0LFxuICAgICAgICAgICAgcHJvcCxcbiAgICAgICAgICAgIGNoaWxkcmVuOiBbXSxcbiAgICAgICAgICB9O1xuICAgICAgICB9KTtcbiAgICAgIH1cblxuICAgICAgLy8gcmVsYXRpb24gcHJvcCDsspjrpqxcbiAgICAgIGNvbnN0IHByb3AgPSBlbnRpdHkucHJvcHNEaWN0W2tleV07XG4gICAgICBpZiAoIWlzUmVsYXRpb25Qcm9wKHByb3ApKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihg7J6Y66q765CcIEZpZWxkRXhwciAke2tleX0uJHtncm91cFswXX1gKTtcbiAgICAgIH1cbiAgICAgIGNvbnN0IHJlbEVudGl0eSA9IEVudGl0eU1hbmFnZXIuZ2V0KHByb3Aud2l0aCk7XG5cbiAgICAgIC8vIHJlbGF0aW9uIC1PbmUg7JeQIGlkIO2VhOuTnCDtlZjrgpjsnbgg6rK97JqwXG4gICAgICBpZiAoaXNCZWxvbmdzVG9PbmVSZWxhdGlvblByb3AocHJvcCkgfHwgaXNPbmVUb09uZVJlbGF0aW9uUHJvcChwcm9wKSkge1xuICAgICAgICBpZiAoZ3JvdXAubGVuZ3RoID09PSAxICYmIChncm91cFswXSA9PT0gXCJpZFwiIHx8IGdyb3VwWzBdID09PSBcImlkP1wiKSkge1xuICAgICAgICAgIC8vIGlkIO2VmOuCmOunjCDsnojripTsp4Ag7LK07YGs7ZW07IScLCDtlZjrgpjrp4wg7J6I7Jy866m0IOyDgeychCBwcm9w7Jy866GcIGlk66W8IOumrO2EtFxuICAgICAgICAgIGNvbnN0IGlkUHJvcCA9IHJlbEVudGl0eS5wcm9wc0RpY3QuaWQ7XG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIG5vZGVUeXBlOiBcInBsYWluXCIgYXMgY29uc3QsXG4gICAgICAgICAgICBwcm9wOiB7XG4gICAgICAgICAgICAgIC4uLmlkUHJvcCxcbiAgICAgICAgICAgICAgbmFtZTogYCR7a2V5fV9pZGAsXG4gICAgICAgICAgICAgIG51bGxhYmxlOiBwcm9wLm51bGxhYmxlLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIGNoaWxkcmVuOiBbXSxcbiAgICAgICAgICB9O1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIC1PbmUg6re47Jm47J2YIOqyveyasCBvYmplY3TroZwg66as7YS0XG4gICAgICAvLyAtTWFueeydmCDqsr3smrAgYXJyYXnroZwg66as7YS0XG4gICAgICAvLyBSZWN1cnNpdmUg66GcIOuOgeyKpCDsspjrpqxcbiAgICAgIGNvbnN0IGNoaWxkcmVuID0gdGhpcy5maWVsZEV4cHJzVG9Qcm9wTm9kZXMoZ3JvdXAsIHJlbEVudGl0eSk7XG4gICAgICBjb25zdCBub2RlVHlwZSA9XG4gICAgICAgIGlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wKHByb3ApIHx8IGlzT25lVG9PbmVSZWxhdGlvblByb3AocHJvcClcbiAgICAgICAgICA/IChcIm9iamVjdFwiIGFzIGNvbnN0KVxuICAgICAgICAgIDogKFwiYXJyYXlcIiBhcyBjb25zdCk7XG5cbiAgICAgIHJldHVybiB7XG4gICAgICAgIHByb3AsXG4gICAgICAgIGNoaWxkcmVuLFxuICAgICAgICBub2RlVHlwZSxcbiAgICAgIH07XG4gICAgfSk7XG4gIH1cblxuICBnZXRGaWVsZEV4cHJzKHByZWZpeCA9IFwiXCIsIG1heERlcHRoOiBudW1iZXIgPSAzLCBmcm9tczogc3RyaW5nW10gPSBbXSk6IHN0cmluZ1tdIHtcbiAgICByZXR1cm4gdGhpcy5wcm9wc1xuICAgICAgLmZsYXRNYXAoKHByb3ApID0+IHtcbiAgICAgICAgY29uc3QgcHJvcE5hbWUgPSBbcHJlZml4LCBwcm9wLm5hbWVdLmZpbHRlcigodikgPT4gdiAhPT0gXCJcIikuam9pbihcIi5cIik7XG4gICAgICAgIGlmIChwcm9wTmFtZSA9PT0gcHJlZml4KSB7XG4gICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKGlzUmVsYXRpb25Qcm9wKHByb3ApKSB7XG4gICAgICAgICAgaWYgKG1heERlcHRoIDwgMCkge1xuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgICAgfVxuICAgICAgICAgIGlmIChmcm9tcy5pbmNsdWRlcyhwcm9wLndpdGgpKSB7XG4gICAgICAgICAgICAvLyDsl63rsKntlqUgcmVsYXRpb27snbgg6rK97JqwIOygnOyZuFxuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgICAgfVxuICAgICAgICAgIC8vIOygleuwqe2WpSByZWxhdGlvbuyduCDqsr3smrAgcmVjdXJzaXZlIOy9nFxuICAgICAgICAgIGNvbnN0IHJlbE1kID0gRW50aXR5TWFuYWdlci5nZXQocHJvcC53aXRoKTtcbiAgICAgICAgICByZXR1cm4gcmVsTWQuZ2V0RmllbGRFeHBycyhwcm9wTmFtZSwgbWF4RGVwdGggLSAxLCBbLi4uZnJvbXMsIHRoaXMuaWRdKTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gcHJvcE5hbWU7XG4gICAgICB9KVxuICAgICAgLmZpbHRlcigoZikgPT4gZiAhPT0gbnVsbCkgYXMgc3RyaW5nW107XG4gIH1cblxuICBnZXRUYWJsZUNvbHVtbnMoKTogeyBuYW1lOiBzdHJpbmc7IHR5cGU6IHN0cmluZyB9W10ge1xuICAgIHJldHVybiB0aGlzLnByb3BzXG4gICAgICAubWFwKChwcm9wKSA9PiB7XG4gICAgICAgIGlmIChwcm9wLnR5cGUgPT09IFwicmVsYXRpb25cIikge1xuICAgICAgICAgIGlmIChcbiAgICAgICAgICAgIHByb3AucmVsYXRpb25UeXBlID09PSBcIkJlbG9uZ3NUb09uZVwiIHx8XG4gICAgICAgICAgICAocHJvcC5yZWxhdGlvblR5cGUgPT09IFwiT25lVG9PbmVcIiAmJiBwcm9wLmhhc0pvaW5Db2x1bW4gPT09IHRydWUpXG4gICAgICAgICAgKSB7XG4gICAgICAgICAgICByZXR1cm4geyBuYW1lOiBgJHtwcm9wLm5hbWV9X2lkYCwgdHlwZTogXCJpbnRfdW5zaWduZWRcIiB9O1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHsgbmFtZTogcHJvcC5uYW1lLCB0eXBlOiBwcm9wLnR5cGUgfTtcbiAgICAgIH0pXG4gICAgICAuZmlsdGVyKG5vbk51bGxhYmxlKTtcbiAgfVxuXG4gIGFzeW5jIHJlZ2lzdGVyTW9kdWxlUGF0aHMoKSB7XG4gICAgY29uc3QgYmFzZVBhdGggPSBgJHt0aGlzLm5hbWVzLnBhcmVudEZzfWA7XG5cbiAgICAvLyBiYXNlLXNjaGVtZVxuICAgIEVudGl0eU1hbmFnZXIuc2V0TW9kdWxlUGF0aChgJHt0aGlzLmlkfUJhc2VTY2hlbWFgLCBgc29uYW11LmdlbmVyYXRlZGApO1xuXG4gICAgLy8gc3Vic2V0XG4gICAgaWYgKE9iamVjdC5rZXlzKHRoaXMuc3Vic2V0cykubGVuZ3RoID4gMCkge1xuICAgICAgRW50aXR5TWFuYWdlci5zZXRNb2R1bGVQYXRoKGAke3RoaXMuaWR9U3Vic2V0S2V5YCwgYHNvbmFtdS5nZW5lcmF0ZWRgKTtcbiAgICAgIEVudGl0eU1hbmFnZXIuc2V0TW9kdWxlUGF0aChgJHt0aGlzLmlkfVN1YnNldE1hcHBpbmdgLCBgc29uYW11LmdlbmVyYXRlZGApO1xuICAgICAgZm9yIChjb25zdCBzdWJzZXRLZXkgb2YgT2JqZWN0LmtleXModGhpcy5zdWJzZXRzKSkge1xuICAgICAgICBFbnRpdHlNYW5hZ2VyLnNldE1vZHVsZVBhdGgoXG4gICAgICAgICAgYCR7dGhpcy5pZH1TdWJzZXQke3N1YnNldEtleS50b1VwcGVyQ2FzZSgpfWAsXG4gICAgICAgICAgYHNvbmFtdS5nZW5lcmF0ZWRgLFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH1cblxuICAgIC8vIGVudW1zXG4gICAgZm9yIChjb25zdCBlbnVtSWQgb2YgT2JqZWN0LmtleXModGhpcy5lbnVtTGFiZWxzKSkge1xuICAgICAgRW50aXR5TWFuYWdlci5zZXRNb2R1bGVQYXRoKGVudW1JZCwgYHNvbmFtdS5nZW5lcmF0ZWRgKTtcbiAgICB9XG5cbiAgICAvLyB0eXBlc1xuICAgIGNvbnN0IHR5cGVzTW9kdWxlUGF0aCA9IGAke2Jhc2VQYXRofS8ke3RoaXMubmFtZXMucGFyZW50RnN9LnR5cGVzYDtcbiAgICBjb25zdCB0eXBlc0ZpbGVQYXRoID0gcGF0aC5qb2luKFxuICAgICAgU29uYW11LmFwaVJvb3RQYXRoLFxuICAgICAgcnVudGltZVBhdGgoYGRpc3QvYXBwbGljYXRpb24vJHt0eXBlc01vZHVsZVBhdGh9LmpzYCksXG4gICAgKTtcblxuICAgIGlmIChhd2FpdCBleGlzdHModHlwZXNGaWxlUGF0aCkpIHtcbiAgICAgIGNvbnN0IGltcG9ydGVkTWVtYmVycyA9IGF3YWl0IGltcG9ydE1lbWJlcnM8ei5ab2RUeXBlQW55Pih0eXBlc0ZpbGVQYXRoKTtcbiAgICAgIHRoaXMudHlwZXMgPSBPYmplY3QuZnJvbUVudHJpZXMoXG4gICAgICAgIGltcG9ydGVkTWVtYmVycy5tYXAoKHsgbmFtZSwgdmFsdWUgfSkgPT4ge1xuICAgICAgICAgIEVudGl0eU1hbmFnZXIuc2V0TW9kdWxlUGF0aChuYW1lLCB0eXBlc01vZHVsZVBhdGgpO1xuICAgICAgICAgIHJldHVybiBbbmFtZSwgdmFsdWVdO1xuICAgICAgICB9KSxcbiAgICAgICkgYXMgeyBbbmFtZTogc3RyaW5nXTogei5ab2RUeXBlQW55IH07XG4gICAgfVxuICB9XG5cbiAgcmVnaXN0ZXJUYWJsZVNwZWNzKCk6IHZvaWQge1xuICAgIGNvbnN0IHVuaXF1ZUluZGV4ZXMgPSB0aGlzLmluZGV4ZXMuZmlsdGVyKChpZHgpID0+IGlkeC50eXBlID09PSBcInVuaXF1ZVwiKTtcblxuICAgIEVudGl0eU1hbmFnZXIuc2V0VGFibGVTcGVjKHtcbiAgICAgIG5hbWU6IHRoaXMudGFibGUsXG4gICAgICB1bmlxdWVJbmRleGVzLFxuICAgIH0pO1xuICB9XG5cbiAgdG9Kc29uKCk6IEVudGl0eUpzb24ge1xuICAgIHJldHVybiB7XG4gICAgICBpZDogdGhpcy5pZCxcbiAgICAgIHBhcmVudElkOiB0aGlzLnBhcmVudElkLFxuICAgICAgdGFibGU6IHRoaXMudGFibGUsXG4gICAgICB0aXRsZTogdGhpcy50aXRsZSxcbiAgICAgIHByb3BzOiB0aGlzLnByb3BzLFxuICAgICAgaW5kZXhlczogdGhpcy5pbmRleGVzLFxuICAgICAgc3Vic2V0czogdGhpcy5zdWJzZXRzLFxuICAgICAgZW51bXM6IHRoaXMuZW51bUxhYmVscyxcbiAgICB9O1xuICB9XG5cbiAgYXN5bmMgc2F2ZSgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyBzb3J0OiBzdWJzZXRzXG4gICAgY29uc3Qgc3Vic2V0Um93cyA9IHRoaXMuZ2V0U3Vic2V0Um93cygpO1xuICAgIHRoaXMuc3Vic2V0cyA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgIE9iamVjdC5lbnRyaWVzKHRoaXMuc3Vic2V0cykubWFwKChbc3Vic2V0S2V5XSkgPT4ge1xuICAgICAgICByZXR1cm4gW3N1YnNldEtleSwgdGhpcy5zdWJzZXRSb3dzVG9TdWJzZXRGaWVsZHMoc3Vic2V0Um93cywgc3Vic2V0S2V5KV07XG4gICAgICB9KSxcbiAgICApO1xuXG4gICAgLy8gc2F2ZVxuICAgIGNvbnN0IGpzb25QYXRoID0gcGF0aC5qb2luKFxuICAgICAgU29uYW11LmFwaVJvb3RQYXRoLFxuICAgICAgYHNyYy9hcHBsaWNhdGlvbi8ke3RoaXMubmFtZXMucGFyZW50RnN9LyR7dGhpcy5uYW1lcy5mc30uZW50aXR5Lmpzb25gLFxuICAgICk7XG4gICAgY29uc3QganNvbiA9IHRoaXMudG9Kc29uKCk7XG4gICAgYXdhaXQgd3JpdGVGaWxlKGpzb25QYXRoLCBmb3JtYXRDb2RlKEpTT04uc3RyaW5naWZ5KGpzb24pLCBcImpzb25cIiwganNvblBhdGgpKTtcblxuICAgIC8vIHJlbG9hZFxuICAgIGF3YWl0IEVudGl0eU1hbmFnZXIucmVnaXN0ZXIoanNvbik7XG4gIH1cblxuICBnZXRTdWJzZXRSb3dzKFxuICAgIF9zdWJzZXRzPzogeyBba2V5OiBzdHJpbmddOiBzdHJpbmdbXSB9LFxuICAgIHByZWZpeGVzOiBzdHJpbmdbXSA9IFtdLFxuICApOiBFbnRpdHlTdWJzZXRSb3dbXSB7XG4gICAgaWYgKHByZWZpeGVzLmxlbmd0aCA+IDEwKSB7XG4gICAgICByZXR1cm4gW107XG4gICAgfVxuXG4gICAgY29uc3Qgc3Vic2V0cyA9IF9zdWJzZXRzID8/IHRoaXMuc3Vic2V0cztcbiAgICBjb25zdCBzdWJzZXRLZXlzID0gT2JqZWN0LmtleXMoc3Vic2V0cyk7XG4gICAgY29uc3QgYWxsRmllbGRzID0gdW5pcXVlKHN1YnNldEtleXMuZmxhdE1hcCgoa2V5KSA9PiBzdWJzZXRzW2tleV0pKTtcblxuICAgIHJldHVybiB0aGlzLnByb3BzLm1hcCgocHJvcCkgPT4ge1xuICAgICAgaWYgKFxuICAgICAgICBwcm9wLnR5cGUgPT09IFwicmVsYXRpb25cIiAmJlxuICAgICAgICBhbGxGaWVsZHMuZmluZCgoZikgPT4gZi5zdGFydHNXaXRoKGAke1suLi5wcmVmaXhlcywgcHJvcC5uYW1lXS5qb2luKFwiLlwiKX0uYCkpXG4gICAgICApIHtcbiAgICAgICAgY29uc3QgcmVsRW50aXR5ID0gRW50aXR5TWFuYWdlci5nZXQocHJvcC53aXRoKTtcbiAgICAgICAgY29uc3QgY2hpbGRyZW4gPSByZWxFbnRpdHkuZ2V0U3Vic2V0Um93cyhzdWJzZXRzLCBbLi4ucHJlZml4ZXMsIGAke3Byb3AubmFtZX1gXSk7XG5cbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBmaWVsZDogcHJvcC5uYW1lLFxuICAgICAgICAgIGNoaWxkcmVuLFxuICAgICAgICAgIHJlbGF0aW9uRW50aXR5OiBwcm9wLndpdGgsXG4gICAgICAgICAgcHJlZml4ZXMsXG4gICAgICAgICAgaXNPcGVuOiBjaGlsZHJlbi5sZW5ndGggPiAwLFxuICAgICAgICAgIGhhczogT2JqZWN0LmZyb21FbnRyaWVzKFxuICAgICAgICAgICAgc3Vic2V0S2V5cy5tYXAoKHN1YnNldEtleSkgPT4ge1xuICAgICAgICAgICAgICByZXR1cm4gW3N1YnNldEtleSwgY2hpbGRyZW4uZXZlcnkoKGNoaWxkKSA9PiBjaGlsZC5oYXNbc3Vic2V0S2V5XSA9PT0gdHJ1ZSldO1xuICAgICAgICAgICAgfSksXG4gICAgICAgICAgKSxcbiAgICAgICAgfTtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgZmllbGQ6IHByb3AubmFtZSxcbiAgICAgICAgY2hpbGRyZW46IFtdLFxuICAgICAgICByZWxhdGlvbkVudGl0eTogcHJvcC50eXBlID09PSBcInJlbGF0aW9uXCIgPyBwcm9wLndpdGggOiB1bmRlZmluZWQsXG4gICAgICAgIHByZWZpeGVzLFxuICAgICAgICBoYXM6IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICAgICAgICBzdWJzZXRLZXlzLm1hcCgoc3Vic2V0S2V5KSA9PiB7XG4gICAgICAgICAgICBjb25zdCBzdWJzZXRGaWVsZHMgPSBzdWJzZXRzW3N1YnNldEtleV07XG4gICAgICAgICAgICBjb25zdCBoYXMgPSBzdWJzZXRGaWVsZHMuc29tZSgoZikgPT4ge1xuICAgICAgICAgICAgICBjb25zdCBmaWVsZCA9IFsuLi5wcmVmaXhlcywgcHJvcC5uYW1lXS5qb2luKFwiLlwiKTtcbiAgICAgICAgICAgICAgcmV0dXJuIGYgPT09IGZpZWxkIHx8IGYuc3RhcnRzV2l0aChgJHtmaWVsZH0uYCk7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIHJldHVybiBbc3Vic2V0S2V5LCBoYXNdO1xuICAgICAgICAgIH0pLFxuICAgICAgICApLFxuICAgICAgfTtcbiAgICB9KTtcbiAgfVxuXG4gIHN1YnNldFJvd3NUb1N1YnNldEZpZWxkcyhzdWJzZXRSb3dzOiBFbnRpdHlTdWJzZXRSb3dbXSwgc3Vic2V0S2V5OiBzdHJpbmcpOiBzdHJpbmdbXSB7XG4gICAgcmV0dXJuIHN1YnNldFJvd3NcbiAgICAgIC5tYXAoKHN1YnNldFJvdykgPT4ge1xuICAgICAgICBpZiAoc3Vic2V0Um93LmNoaWxkcmVuLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICByZXR1cm4gdGhpcy5zdWJzZXRSb3dzVG9TdWJzZXRGaWVsZHMoc3Vic2V0Um93LmNoaWxkcmVuLCBzdWJzZXRLZXkpO1xuICAgICAgICB9IGVsc2UgaWYgKHN1YnNldFJvdy5oYXNbc3Vic2V0S2V5XSkge1xuICAgICAgICAgIHJldHVybiBzdWJzZXRSb3cucHJlZml4ZXMuY29uY2F0KHN1YnNldFJvdy5maWVsZCkuam9pbihcIi5cIik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgICAuZmlsdGVyKG5vbk51bGxhYmxlKVxuICAgICAgLmZsYXQoKTtcbiAgfVxuXG4gIGFzeW5jIGNyZWF0ZVByb3AocHJvcDogRW50aXR5UHJvcCwgYXQ/OiBudW1iZXIpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBpZiAoIWF0KSB7XG4gICAgICB0aGlzLnByb3BzLnB1c2gocHJvcCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMucHJvcHMuc3BsaWNlKGF0LCAwLCBwcm9wKTtcbiAgICB9XG4gICAgYXdhaXQgdGhpcy5zYXZlKCk7XG4gIH1cblxuICBhbmFseXplU3Vic2V0RmllbGQoc3Vic2V0RmllbGQ6IHN0cmluZyk6IHtcbiAgICBlbnRpdHlJZDogc3RyaW5nO1xuICAgIHByb3BOYW1lOiBzdHJpbmc7XG4gIH1bXSB7XG4gICAgY29uc3QgYXJyID0gc3Vic2V0RmllbGQuc3BsaXQoXCIuXCIpO1xuXG4gICAgbGV0IGVudGl0eUlkID0gdGhpcy5pZDtcbiAgICBjb25zdCByZXN1bHQ6IHtcbiAgICAgIGVudGl0eUlkOiBzdHJpbmc7XG4gICAgICBwcm9wTmFtZTogc3RyaW5nO1xuICAgIH1bXSA9IFtdO1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgYXJyLmxlbmd0aDsgaSsrKSB7XG4gICAgICBjb25zdCBwcm9wTmFtZSA9IGFycltpXTtcbiAgICAgIHJlc3VsdC5wdXNoKHtcbiAgICAgICAgZW50aXR5SWQsXG4gICAgICAgIHByb3BOYW1lLFxuICAgICAgfSk7XG5cbiAgICAgIGNvbnN0IHByb3AgPSBFbnRpdHlNYW5hZ2VyLmdldChlbnRpdHlJZCkucHJvcHMuZmluZCgocCkgPT4gcC5uYW1lID09PSBwcm9wTmFtZSk7XG4gICAgICBpZiAoIXByb3ApIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGAke2VudGl0eUlkfeydmCDsnpjrqrvrkJwg7ISc67iM7IWL7YKkICR7c3Vic2V0RmllbGR9YCk7XG4gICAgICB9XG4gICAgICBpZiAoaXNSZWxhdGlvblByb3AocHJvcCkpIHtcbiAgICAgICAgZW50aXR5SWQgPSBwcm9wLndpdGg7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICBhc3luYyBtb2RpZnlQcm9wKG5ld1Byb3A6IEVudGl0eVByb3AsIGF0OiBudW1iZXIpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyDsnbTsoIQg7ZSE66GtIOydtOumhCDsoIDsnqVcbiAgICBjb25zdCBvbGROYW1lID0gdGhpcy5wcm9wc1thdF0ubmFtZTtcblxuICAgIC8vIOyggOyepe2VoCDsl5Tti7Dti7BcbiAgICBjb25zdCBlbnRpdGllczogRW50aXR5W10gPSBbdGhpc107XG5cbiAgICAvLyDsnbTrpoTsnbQg67CU64CQIOqyveyasFxuICAgIGlmIChvbGROYW1lICE9PSBuZXdQcm9wLm5hbWUpIHtcbiAgICAgIC8vIOyghOyytCDsl5Tti7Dti7Dsl5DshJwg7ZiE7J6sIOyImOygleuQnCDtlITroa3snYQg7LC47KGw7ZWY6rOgIOyeiOuKlCDrqqjrk6Ag7ISc67iM7IWL7ZWE65OcIOywvuyVhOyEnCDsiJjsoJVcbiAgICAgIGNvbnN0IGFsbEVudGl0eUlkcyA9IEVudGl0eU1hbmFnZXIuZ2V0QWxsSWRzKCk7XG4gICAgICBmb3IgKGNvbnN0IHJlbEVudGl0eUlkIG9mIGFsbEVudGl0eUlkcykge1xuICAgICAgICBjb25zdCByZWxFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChyZWxFbnRpdHlJZCk7XG4gICAgICAgIGNvbnN0IHJlbEVudGl0eVN1YnNldEtleXMgPSBPYmplY3Qua2V5cyhyZWxFbnRpdHkuc3Vic2V0cyk7XG4gICAgICAgIGZvciAoY29uc3Qgc3Vic2V0S2V5IG9mIHJlbEVudGl0eVN1YnNldEtleXMpIHtcbiAgICAgICAgICBjb25zdCBzdWJzZXQgPSByZWxFbnRpdHkuc3Vic2V0c1tzdWJzZXRLZXldO1xuXG4gICAgICAgICAgLy8g7ISc67iM7IWLIO2VhOuTnOulvCDsiJztmoztlZjrqbAsIOyXlO2LsO2LsC3tlITroa0g64uo7JyE66GcIOu2hOyEne2VnCDtm4Qg7ZiE7J6sIOyXlO2LsO2LsC3tlITroa3qs7wg7J287LmY7ZWY64qUIOqyveyasCDsiJjsoJUg7LKY66asXG4gICAgICAgICAgY29uc3QgbW9kaWZpZWRTdWJzZXRGaWVsZHMgPSBzdWJzZXQubWFwKChzdWJzZXRGaWVsZCkgPT4ge1xuICAgICAgICAgICAgY29uc3QgYW5hbHl6ZWQgPSByZWxFbnRpdHkuYW5hbHl6ZVN1YnNldEZpZWxkKHN1YnNldEZpZWxkKTtcbiAgICAgICAgICAgIGNvbnN0IG1vZGlmaWVkID0gYW5hbHl6ZWQubWFwKChhKSA9PlxuICAgICAgICAgICAgICBhLnByb3BOYW1lID09PSBvbGROYW1lICYmIGEuZW50aXR5SWQgPT09IHRoaXMuaWRcbiAgICAgICAgICAgICAgICA/IHtcbiAgICAgICAgICAgICAgICAgICAgLi4uYSxcbiAgICAgICAgICAgICAgICAgICAgcHJvcE5hbWU6IG5ld1Byb3AubmFtZSxcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICA6IGEsXG4gICAgICAgICAgICApO1xuICAgICAgICAgICAgLy8g67aE7ISd7ZWcIO2VhOuTnOulvCDri6Tsi5wg7ISc67iM7IWLIO2VhOuTnOuhnCDrs7XqtaxcbiAgICAgICAgICAgIHJldHVybiBtb2RpZmllZC5tYXAoKGEpID0+IGEucHJvcE5hbWUpLmpvaW4oXCIuXCIpO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgaWYgKHN1YnNldC5qb2luKFwiLFwiKSAhPT0gbW9kaWZpZWRTdWJzZXRGaWVsZHMuam9pbihcIixcIikpIHtcbiAgICAgICAgICAgIHJlbEVudGl0eS5zdWJzZXRzW3N1YnNldEtleV0gPSBtb2RpZmllZFN1YnNldEZpZWxkcztcbiAgICAgICAgICAgIGVudGl0aWVzLnB1c2gocmVsRW50aXR5KTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyDtlITroa0g7IiY7KCVXG4gICAgdGhpcy5wcm9wc1thdF0gPSBuZXdQcm9wO1xuXG4gICAgYXdhaXQgUHJvbWlzZS5hbGwoZW50aXRpZXMubWFwKGFzeW5jIChlbnRpdHkpID0+IGVudGl0eS5zYXZlKCkpKTtcbiAgfVxuXG4gIGFzeW5jIGRlbFByb3AoYXQ6IG51bWJlcik6IFByb21pc2U8dm9pZD4ge1xuICAgIC8vIOydtOyghCDtlITroa0g7J2066aEIOyggOyepVxuICAgIGNvbnN0IG9sZE5hbWUgPSB0aGlzLnByb3BzW2F0XS5uYW1lO1xuXG4gICAgLy8g7KCA7J6l7ZWgIOyXlO2LsO2LsFxuICAgIGNvbnN0IGVudGl0aWVzOiBFbnRpdHlbXSA9IFt0aGlzXTtcblxuICAgIC8vIOyghOyytCDsl5Tti7Dti7Dsl5DshJwg7ZiE7J6sIOyCreygnOuQnCDtlITroa3snYQg7LC47KGw7ZWY6rOgIOyeiOuKlCDrqqjrk6Ag7ISc67iM7IWL7ZWE65OcIOywvuyVhOyEnCDsoJzsmbhcbiAgICBjb25zdCBhbGxFbnRpdHlJZHMgPSBFbnRpdHlNYW5hZ2VyLmdldEFsbElkcygpO1xuICAgIGZvciAoY29uc3QgcmVsRW50aXR5SWQgb2YgYWxsRW50aXR5SWRzKSB7XG4gICAgICBjb25zdCByZWxFbnRpdHkgPSBFbnRpdHlNYW5hZ2VyLmdldChyZWxFbnRpdHlJZCk7XG4gICAgICBjb25zdCByZWxFbnRpdHlTdWJzZXRLZXlzID0gT2JqZWN0LmtleXMocmVsRW50aXR5LnN1YnNldHMpO1xuICAgICAgZm9yIChjb25zdCBzdWJzZXRLZXkgb2YgcmVsRW50aXR5U3Vic2V0S2V5cykge1xuICAgICAgICBjb25zdCBzdWJzZXQgPSByZWxFbnRpdHkuc3Vic2V0c1tzdWJzZXRLZXldO1xuICAgICAgICAvLyDshJzruIzshYsg7ZWE65Oc66W8IOyInO2ajO2VmOupsCwg7JeU7Yuw7YuwLe2UhOuhrSDri6jsnITroZwg67aE7ISd7ZWcIO2bhCDtmITsnqwg7JeU7Yuw7YuwLe2UhOuhreqzvCDsnbzsuZjtlZjripQg6rK97JqwIOydtO2bhOydmCDtlYTrk5zrpbwg7KCc7Jm4XG4gICAgICAgIGNvbnN0IG1vZGlmaWVkU3Vic2V0RmllbGRzID0gc3Vic2V0XG4gICAgICAgICAgLm1hcCgoc3Vic2V0RmllbGQpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGFuYWx5emVkID0gcmVsRW50aXR5LmFuYWx5emVTdWJzZXRGaWVsZChzdWJzZXRGaWVsZCk7XG4gICAgICAgICAgICBpZiAoYW5hbHl6ZWQuZmluZCgoYSkgPT4gYS5wcm9wTmFtZSA9PT0gb2xkTmFtZSAmJiBhLmVudGl0eUlkID09PSB0aGlzLmlkKSkge1xuICAgICAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIHJldHVybiBzdWJzZXRGaWVsZDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KVxuICAgICAgICAgIC5maWx0ZXIobm9uTnVsbGFibGUpO1xuXG4gICAgICAgIGlmIChzdWJzZXQuam9pbihcIixcIikgIT09IG1vZGlmaWVkU3Vic2V0RmllbGRzLmpvaW4oXCIsXCIpKSB7XG4gICAgICAgICAgcmVsRW50aXR5LnN1YnNldHNbc3Vic2V0S2V5XSA9IG1vZGlmaWVkU3Vic2V0RmllbGRzO1xuICAgICAgICAgIGVudGl0aWVzLnB1c2gocmVsRW50aXR5KTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIC8vIO2YhOyerCDsl5Tti7Dti7DsnZgg7J24642x7Iqk7JeQ7IScIOygnOyZuFxuICAgIGZvciAoY29uc3QgaW5kZXggb2YgRW50aXR5TWFuYWdlci5nZXQodGhpcy5pZCkuaW5kZXhlcykge1xuICAgICAgaW5kZXguY29sdW1ucyA9IGluZGV4LmNvbHVtbnMuZmlsdGVyKChjb2wpID0+IGNvbCAhPT0gb2xkTmFtZSk7XG4gICAgfVxuXG4gICAgLy8g7ZSE66GtIOyCreygnFxuICAgIHRoaXMucHJvcHMuc3BsaWNlKGF0LCAxKTtcblxuICAgIGF3YWl0IFByb21pc2UuYWxsKGVudGl0aWVzLm1hcChhc3luYyAoZW50aXR5KSA9PiBlbnRpdHkuc2F2ZSgpKSk7XG4gIH1cblxuICBnZXRFbnRpdHlJZEZyb21TdWJzZXRGaWVsZChzdWJzZXRGaWVsZDogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBpZiAoc3Vic2V0RmllbGQuaW5jbHVkZXMoXCIuXCIpID09PSBmYWxzZSkge1xuICAgICAgcmV0dXJuIHRoaXMuaWQ7XG4gICAgfVxuXG4gICAgLy8g7ISc67iM7IWLIO2VhOuTnOydmCDrp4jsp4Drp4nsnYAg7ZSE66Gt7J2066+A66GcIOygnOyZuFxuICAgIGNvbnN0IGFyciA9IHN1YnNldEZpZWxkLnNwbGl0KFwiLlwiKS5zbGljZSgwLCAtMSk7XG5cbiAgICAvLyDshJzruIzshYsg7ZWE65Oc66W8IOuCtOugpOqwgOuptOyEnCDrp4jsp4Drp4nsnLzroZwgcmVsYXRpb27rkJwg7JeU7Yuw7Yuw66W8IOywvuydjFxuICAgIGNvbnN0IGxhc3RFbnRpdHlJZCA9IGFyci5yZWR1Y2UoKGVudGl0eUlkLCBmaWVsZCkgPT4ge1xuICAgICAgY29uc3QgcmVsUHJvcCA9IEVudGl0eU1hbmFnZXIuZ2V0KGVudGl0eUlkKS5wcm9wcy5maW5kKChwKSA9PiBwLm5hbWUgPT09IGZpZWxkKTtcbiAgICAgIGlmICghcmVsUHJvcCB8fCByZWxQcm9wLnR5cGUgIT09IFwicmVsYXRpb25cIikge1xuICAgICAgICBjb25zb2xlLmRlYnVnKHsgYXJyLCB0aGlzSWQ6IHRoaXMuaWQsIGVudGl0eUlkLCBmaWVsZCB9KTtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGDsnpjrqrvrkJwg7ISc67iM7IWL7YKkICR7c3Vic2V0RmllbGR9YCk7XG4gICAgICB9XG4gICAgICByZXR1cm4gcmVsUHJvcC53aXRoO1xuICAgIH0sIHRoaXMuaWQpO1xuICAgIHJldHVybiBsYXN0RW50aXR5SWQ7XG4gIH1cblxuICBhc3luYyBtb3ZlUHJvcChhdDogbnVtYmVyLCB0bzogbnVtYmVyKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgcHJvcCA9IHRoaXMucHJvcHNbYXRdO1xuICAgIGNvbnN0IG5ld1Byb3BzID0gWy4uLnRoaXMucHJvcHNdO1xuICAgIG5ld1Byb3BzLnNwbGljZSh0bywgMCwgcHJvcCk7XG4gICAgbmV3UHJvcHMuc3BsaWNlKGF0IDwgdG8gPyBhdCA6IGF0ICsgMSwgMSk7XG4gICAgdGhpcy5wcm9wcyA9IG5ld1Byb3BzO1xuXG4gICAgYXdhaXQgdGhpcy5zYXZlKCk7XG4gIH1cbn1cbiJdLCJuYW1lcyI6WyJhc3NlcnQiLCJ3cml0ZUZpbGUiLCJpbmZsZWN0aW9uIiwicGF0aCIsImdyb3VwIiwidW5pcXVlIiwieiIsIlNvbmFtdSIsImlzQmVsb25nc1RvT25lUmVsYXRpb25Qcm9wIiwiaXNFbnVtUHJvcCIsImlzSGFzTWFueVJlbGF0aW9uUHJvcCIsImlzTWFueVRvTWFueVJlbGF0aW9uUHJvcCIsImlzT25lVG9PbmVSZWxhdGlvblByb3AiLCJpc1JlbGF0aW9uUHJvcCIsImlzVmlydHVhbFByb3AiLCJpbXBvcnRNZW1iZXJzIiwiZm9ybWF0Q29kZSIsImV4aXN0cyIsInJ1bnRpbWVQYXRoIiwiYXNzZXJ0RGVmaW5lZCIsIm5vbk51bGxhYmxlIiwiRW50aXR5TWFuYWdlciIsIkVudGl0eSIsImlkIiwicGFyZW50SWQiLCJ0YWJsZSIsInRpdGxlIiwibmFtZXMiLCJwcm9wcyIsInByb3BzRGljdCIsInJlbGF0aW9ucyIsImluZGV4ZXMiLCJzdWJzZXRzIiwidHlwZXMiLCJlbnVtcyIsImVudW1MYWJlbHMiLCJ1bmRlcnNjb3JlIiwicGx1cmFsaXplIiwibWFwIiwicHJvcCIsImluY2x1ZGVzIiwicmVwbGFjZSIsIk9iamVjdCIsImZyb21FbnRyaWVzIiwibmFtZSIsImZpbHRlciIsImVudHJpZXMiLCJrZXkiLCJlbnVtTGFiZWwiLCJlbnVtIiwia2V5cyIsInBhcmVudEZzIiwiZGFzaGVyaXplIiwidG9Mb3dlckNhc2UiLCJmcyIsIm1vZHVsZSIsImdldFB1cmlTdWJzZXRRdWVyeSIsInN1YnNldEtleSIsInN1YnNldCIsInN1YnNldFF1ZXJ5IiwicmVzb2x2ZVN1YnNldFF1ZXJ5IiwibGluZXMiLCJwdXNoIiwiam9pbiIsImpvaW5zIiwiam9pbk1ldGhvZCIsImFzIiwiY3VzdG9tIiwiZnJvbSIsInRvIiwic2VsZWN0T2JqIiwic2VsZWN0SXRlbSIsInNlbGVjdCIsIm1hdGNoIiwiY29sdW1uIiwiYWxpYXMiLCJzcGxpdCIsInBvcCIsInRyaW0iLCJmb3JFYWNoIiwidmFsdWUiLCJnZXRQdXJpTG9hZGVyUXVlcnkiLCJsb2FkZXJzIiwicGFyc2VTZWxlY3QiLCJ0YWJsZVByZWZpeCIsInN0YXJ0c1dpdGgiLCJnZW5lcmF0ZUxvYWRlckNvZGUiLCJsb2FkZXJMaW5lcyIsImxvYWRlciIsInRvVGFibGUiLCJ0b0NvbCIsInRocm91Z2giLCJtYW55Sm9pbiIsImlkRmllbGQiLCJ1bmRlZmluZWQiLCJvbmVKb2lucyIsImpvaW5UeXBlIiwiZnJvbUNvbCIsImxlbmd0aCIsImdldFN1YnNldFF1ZXJ5IiwicmVzdWx0IiwicHJlZml4IiwiZmllbGRzIiwiaXNBbHJlYWR5T3V0ZXJKb2luZWQiLCJzdWJzZXRHcm91cCIsImZpZWxkIiwicmVsIiwicmVkdWNlIiwiciIsImdyb3VwS2V5IiwicmVhbEZpZWxkcyIsInZpcnR1YWxGaWVsZHMiLCJjb25jYXQiLCJ2aXJ0dWFsIiwicmVsYXRpb24iLCJFcnJvciIsInJlbEVudGl0eSIsImdldCIsIndpdGgiLCJyZWxGaWVsZHMiLCJzbGljZSIsImlubmVyT3JPdXRlciIsImhhc0pvaW5Db2x1bW4iLCJudWxsYWJsZSIsInJlbFN1YnNldFF1ZXJ5Iiwiam9pbkFzIiwiZnJvbVRhYmxlIiwiam9pbkNsYXVzZSIsImN1c3RvbUpvaW5DbGF1c2UiLCJjb252ZXJ0ZWRMb2FkZXJzIiwibmV3QXMiLCJmcm9tQ29sdW1uIiwiam9pbkNvbHVtbiIsImpvaW5UYWJsZSIsInNpbmd1bGFyaXplIiwiZmllbGRFeHByc1RvUHJvcE5vZGVzIiwiZmllbGRFeHBycyIsImVudGl0eSIsImdyb3VwcyIsImZpZWxkRXhwciIsImVsc2VFeHByIiwiZmxhdE1hcCIsInByb3BOYW1lIiwibm9kZVR5cGUiLCJ0eXBlIiwiY2hpbGRyZW4iLCJmaW5kIiwicCIsImNvbnNvbGUiLCJsb2ciLCJpZFByb3AiLCJnZXRGaWVsZEV4cHJzIiwibWF4RGVwdGgiLCJmcm9tcyIsInYiLCJyZWxNZCIsImYiLCJnZXRUYWJsZUNvbHVtbnMiLCJyZWxhdGlvblR5cGUiLCJyZWdpc3Rlck1vZHVsZVBhdGhzIiwiYmFzZVBhdGgiLCJzZXRNb2R1bGVQYXRoIiwidG9VcHBlckNhc2UiLCJlbnVtSWQiLCJ0eXBlc01vZHVsZVBhdGgiLCJ0eXBlc0ZpbGVQYXRoIiwiYXBpUm9vdFBhdGgiLCJpbXBvcnRlZE1lbWJlcnMiLCJyZWdpc3RlclRhYmxlU3BlY3MiLCJ1bmlxdWVJbmRleGVzIiwiaWR4Iiwic2V0VGFibGVTcGVjIiwidG9Kc29uIiwic2F2ZSIsInN1YnNldFJvd3MiLCJnZXRTdWJzZXRSb3dzIiwic3Vic2V0Um93c1RvU3Vic2V0RmllbGRzIiwianNvblBhdGgiLCJqc29uIiwiSlNPTiIsInN0cmluZ2lmeSIsInJlZ2lzdGVyIiwiX3N1YnNldHMiLCJwcmVmaXhlcyIsInN1YnNldEtleXMiLCJhbGxGaWVsZHMiLCJyZWxhdGlvbkVudGl0eSIsImlzT3BlbiIsImhhcyIsImV2ZXJ5IiwiY2hpbGQiLCJzdWJzZXRGaWVsZHMiLCJzb21lIiwic3Vic2V0Um93IiwiZmxhdCIsImNyZWF0ZVByb3AiLCJhdCIsInNwbGljZSIsImFuYWx5emVTdWJzZXRGaWVsZCIsInN1YnNldEZpZWxkIiwiYXJyIiwiZW50aXR5SWQiLCJpIiwibW9kaWZ5UHJvcCIsIm5ld1Byb3AiLCJvbGROYW1lIiwiZW50aXRpZXMiLCJhbGxFbnRpdHlJZHMiLCJnZXRBbGxJZHMiLCJyZWxFbnRpdHlJZCIsInJlbEVudGl0eVN1YnNldEtleXMiLCJtb2RpZmllZFN1YnNldEZpZWxkcyIsImFuYWx5emVkIiwibW9kaWZpZWQiLCJhIiwiUHJvbWlzZSIsImFsbCIsImRlbFByb3AiLCJpbmRleCIsImNvbHVtbnMiLCJjb2wiLCJnZXRFbnRpdHlJZEZyb21TdWJzZXRGaWVsZCIsImxhc3RFbnRpdHlJZCIsInJlbFByb3AiLCJkZWJ1ZyIsInRoaXNJZCIsIm1vdmVQcm9wIiwibmV3UHJvcHMiXSwibWFwcGluZ3MiOiJBQUFBLE9BQU9BLFlBQVksU0FBUztBQUM1QixTQUFTQyxTQUFTLFFBQVEsbUJBQWM7QUFDeEMsT0FBT0MsZ0JBQWdCLGFBQWE7QUFDcEMsT0FBT0MsVUFBVSxPQUFPO0FBQ3hCLFNBQVNDLEtBQUssRUFBRUMsTUFBTSxRQUFRLFVBQVU7QUFDeEMsU0FBU0MsQ0FBQyxRQUFRLE1BQU07QUFDeEIsU0FBU0MsTUFBTSxRQUFRLG1CQUFnQjtBQUN2QyxTQU1FQywwQkFBMEIsRUFDMUJDLFVBQVUsRUFDVkMscUJBQXFCLEVBQ3JCQyx3QkFBd0IsRUFDeEJDLHNCQUFzQixFQUN0QkMsY0FBYyxFQUNkQyxhQUFhLFFBSVIsb0JBQWlCO0FBQ3hCLFNBQVNDLGFBQWEsUUFBUSx3QkFBcUI7QUFDbkQsU0FBU0MsVUFBVSxRQUFRLHdCQUFxQjtBQUNoRCxTQUFTQyxNQUFNLFFBQVEsdUJBQW9CO0FBQzNDLFNBQVNDLFdBQVcsUUFBUSx5QkFBc0I7QUFDbEQsU0FBU0MsYUFBYSxFQUFFQyxXQUFXLFFBQVEsb0JBQWlCO0FBQzVELFNBQVNDLGFBQWEsUUFBUSxzQkFBbUI7QUFFakQsT0FBTyxNQUFNQztJQUNYQyxHQUFXO0lBQ1hDLFNBQWtCO0lBQ2xCQyxNQUFjO0lBQ2RDLE1BQWM7SUFDZEMsTUFJRTtJQUNGQyxNQUFvQjtJQUNwQkMsVUFFRTtJQUNGQyxVQUVFO0lBQ0ZDLFFBQXVCO0lBQ3ZCQyxRQUVFO0lBQ0ZDLFFBRUksQ0FBQyxFQUFFO0lBQ1BDLFFBRUksQ0FBQyxFQUFFO0lBQ1BDLGFBSUksQ0FBQyxFQUFFO0lBRVAsWUFBWSxFQUFFWixFQUFFLEVBQUVDLFFBQVEsRUFBRUMsS0FBSyxFQUFFQyxLQUFLLEVBQUVFLEtBQUssRUFBRUcsT0FBTyxFQUFFQyxP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
|