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/bin/cli.js
CHANGED
|
@@ -1,2 +1,519 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,"__esModule",{value:true});var _chalk=/*#__PURE__*/_interop_require_default(require("chalk"));var _dotenv=/*#__PURE__*/_interop_require_default(require("dotenv"));var _path=/*#__PURE__*/_interop_require_default(require("path"));var _tsicli=require("tsicli");var _child_process=require("child_process");var _promises=require("fs/promises");var _fsutils=require("../utils/fs-utils");var _process=/*#__PURE__*/_interop_require_default(require("process"));var _api=require("../api");var _knex=/*#__PURE__*/_interop_require_default(require("knex"));var _utils=require("../utils/utils");var _entitymanager=require("../entity/entity-manager");var _migrator=require("../migration/migrator");var _fixturemanager=require("../testing/fixture-manager");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 _async_iterator(iterable){var method,async,sync,retry=2;for("undefined"!=typeof Symbol&&(async=Symbol.asyncIterator,sync=Symbol.iterator);retry--;){if(async&&null!=(method=iterable[async]))return method.call(iterable);if(sync&&null!=(method=iterable[sync]))return new AsyncFromSyncIterator(method.call(iterable));async="@@asyncIterator",sync="@@iterator"}throw new TypeError("Object is not async iterable")}function AsyncFromSyncIterator(s){function AsyncFromSyncIteratorContinuation(r){if(Object(r)!==r)return Promise.reject(new TypeError(r+" is not an object."));var done=r.done;return Promise.resolve(r.value).then(function(value){return{value:value,done:done}})}return AsyncFromSyncIterator=function(s){this.s=s,this.n=s.next},AsyncFromSyncIterator.prototype={s:null,n:null,next:function(){return AsyncFromSyncIteratorContinuation(this.n.apply(this.s,arguments))},return:function(value){var ret=this.s.return;return void 0===ret?Promise.resolve({value:value,done:!0}):AsyncFromSyncIteratorContinuation(ret.apply(this.s,arguments))},throw:function(value){var thr=this.s.return;return void 0===thr?Promise.reject(value):AsyncFromSyncIteratorContinuation(thr.apply(this.s,arguments))}},new AsyncFromSyncIterator(s)}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 _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 _instanceof(left,right){if(right!=null&&typeof Symbol!=="undefined"&&right[Symbol.hasInstance]){return!!right[Symbol.hasInstance](left)}else{return left instanceof right}}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_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 _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 _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}}}_dotenv.default.config();var migrator;function bootstrap(){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:if(!(_process.default.argv[2]!=="dev:serve"))return[3,2];return[4,_api.Sonamu.init(false,false)];case 1:_state.sent();_state.label=2;case 2:return[4,(0,_tsicli.tsicli)(_process.default.argv,{types:{"#entityId":{type:"autocomplete",name:"#entityId",message:"Please input #entityId",choices:_entitymanager.EntityManager.getAllParentIds().map(function(entityId){return{title:entityId,value:entityId}})},"#recordIds":"number[]","#name":"string"},args:[["fixture","init"],["fixture","import","#entityId","#recordIds"],["fixture","sync"],["migrate","run"],["migrate","check"],["migrate","rollback"],["migrate","reset"],["migrate","clear"],["migrate","status"],["stub","practice","#name"],["stub","entity","#name"],["scaffold","model","#entityId"],["scaffold","model_test","#entityId"],["scaffold","view_list","#entityId"],["scaffold","view_form","#entityId"],["ui"],["dev:serve"],["serve"]],runners:{migrate_run:migrate_run,migrate_check:migrate_check,migrate_rollback:migrate_rollback,migrate_clear:migrate_clear,migrate_reset:migrate_reset,migrate_status:migrate_status,fixture_init:fixture_init,fixture_import:fixture_import,fixture_sync:fixture_sync,stub_practice:stub_practice,stub_entity:stub_entity,scaffold_model:scaffold_model,scaffold_model_test:scaffold_model_test,ui:ui,"dev:serve":dev_serve,serve:serve}})];case 3:_state.sent();return[2]}})})()}bootstrap().finally(function(){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:if(!migrator)return[3,2];return[4,migrator.destroy()];case 1:_state.sent();_state.label=2;case 2:return[4,_fixturemanager.FixtureManager.destroy()];case 3:_state.sent();return[2]}})})()});function dev_serve(){return _async_to_generator(function(){var nodemon,nodemonConfig,cleanup;return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,Promise.resolve().then(function(){return /*#__PURE__*/_interop_require_wildcard(require("nodemon"))})];case 1:nodemon=_state.sent();return[4,function(){return _async_to_generator(function(){var projectNodemonPath,hasProjectNodemon,_;return _ts_generator(this,function(_state){switch(_state.label){case 0:projectNodemonPath=_path.default.join((0,_utils.findApiRootPath)(),"nodemon.json");return[4,(0,_fsutils.exists)(projectNodemonPath)];case 1:hasProjectNodemon=_state.sent();if(!hasProjectNodemon)return[3,3];_=JSON.parse;return[4,(0,_promises.readFile)(projectNodemonPath,"utf8")];case 2:return[2,_.apply(JSON,[_state.sent()])];case 3:return[2,{watch:["src/index.ts"],ignore:["dist/**","**/*.js","**/*.d.ts"],exec:["node --no-warnings -r source-map-support/register -r dotenv/config dist/index.js"].join(" && ")}]}})})()}()];case 2:nodemonConfig=_state.sent();nodemon.default(nodemonConfig);cleanup=function(){return _async_to_generator(function(){var _Sonamu_server;return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,(_Sonamu_server=_api.Sonamu.server)===null||_Sonamu_server===void 0?void 0:_Sonamu_server.close()];case 1:_state.sent();_process.default.exit(0);return[2]}})})()};_process.default.on("SIGINT",cleanup);_process.default.on("SIGTERM",cleanup);_process.default.on("SIGUSR2",cleanup);return[2]}})})()}function serve(){return _async_to_generator(function(){var distIndexPath,spawn,serverProcess;return _ts_generator(this,function(_state){switch(_state.label){case 0:distIndexPath=_path.default.join(_api.Sonamu.apiRootPath,"dist","index.js");return[4,(0,_fsutils.exists)(distIndexPath)];case 1:if(!_state.sent()){console.log(_chalk.default.red("dist/index.js not found. Please build your project first."));console.log(_chalk.default.blue("Run: yarn sonamu build"));return[2]}return[4,Promise.resolve().then(function(){return /*#__PURE__*/_interop_require_wildcard(require("child_process"))})];case 2:spawn=_state.sent().spawn;serverProcess=spawn("node",["-r","source-map-support/register","-r","dotenv/config",distIndexPath],{cwd:_api.Sonamu.apiRootPath,stdio:"inherit"});_process.default.on("SIGINT",function(){serverProcess.kill("SIGTERM");_process.default.exit(0)});return[2]}})})()}function setupMigrator(){return _async_to_generator(function(){return _ts_generator(this,function(_state){migrator=new _migrator.Migrator({mode:"dev"});return[2]})})()}function setupFixtureManager(){return _async_to_generator(function(){return _ts_generator(this,function(_state){_fixturemanager.FixtureManager.init();return[2]})})()}function migrate_run(){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,setupMigrator()];case 1:_state.sent();return[4,migrator.cleanUpDist()];case 2:_state.sent();return[4,migrator.run()];case 3:_state.sent();return[2]}})})()}function migrate_check(){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,setupMigrator()];case 1:_state.sent();return[4,migrator.cleanUpDist()];case 2:_state.sent();return[4,migrator.check()];case 3:_state.sent();return[2]}})})()}function migrate_status(){return _async_to_generator(function(){var status;return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,setupMigrator()];case 1:_state.sent();return[4,migrator.getStatus()];case 2:status=_state.sent();console.log(status);return[2]}})})()}function migrate_rollback(){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,setupMigrator()];case 1:_state.sent();return[4,migrator.rollback()];case 2:_state.sent();return[2]}})})()}function migrate_clear(){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,setupMigrator()];case 1:_state.sent();return[4,migrator.clearPendingList()];case 2:_state.sent();return[2]}})})()}function migrate_reset(){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,setupMigrator()];case 1:_state.sent();return[4,migrator.resetAll()];case 2:_state.sent();return[2]}})})()}function fixture_init(){return _async_to_generator(function(){var srcConfig,targets,dumpFilename,srcConn,migrationsDump,_db,_ref,_ref_,migrations,_iteratorAbruptCompletion,_didIteratorError,_iteratorError,_iterator,_step,_value,label,config,toSkip,conn,_config_connection,db,_ref1,_ref_1,row,mysqlCmd,err;return _ts_generator(this,function(_state){switch(_state.label){case 0:srcConfig=_api.Sonamu.dbConfig.development_master;targets=[{label:"(REMOTE) Fixture DB",config:_api.Sonamu.dbConfig.fixture_remote},{label:"(LOCAL) Fixture DB",config:_api.Sonamu.dbConfig.fixture_local,toSkip:function(){var remoteConn=_api.Sonamu.dbConfig.fixture_remote.connection;var localConn=_api.Sonamu.dbConfig.fixture_local.connection;return remoteConn.host===localConn.host&&remoteConn.database===localConn.database}()},{label:"(LOCAL) Testing DB",config:_api.Sonamu.dbConfig.test}];console.log("DUMP...");dumpFilename="/tmp/sonamu-fixture-init-".concat(Date.now(),".sql");srcConn=srcConfig.connection;migrationsDump="/tmp/sonamu-fixture-init-migrations-".concat(Date.now(),".sql");(0,_child_process.execSync)("mysqldump -h".concat(srcConn.host," -u").concat(srcConn.user," -p").concat(srcConn.password," --single-transaction -d --no-create-db --triggers ").concat(srcConn.database," > ").concat(dumpFilename));_db=(0,_knex.default)(srcConfig);return[4,_db.raw("SELECT COUNT(*) as count FROM information_schema.tables WHERE table_schema = ? AND table_name = 'knex_migrations'",[srcConn.database])];case 1:_ref=_sliced_to_array.apply(void 0,[_state.sent(),1]),_ref_=_sliced_to_array(_ref[0],1),migrations=_ref_[0];if(migrations.count>0){(0,_child_process.execSync)("mysqldump -h".concat(srcConn.host," -u").concat(srcConn.user," -p").concat(srcConn.password," --single-transaction --no-create-db --triggers ").concat(srcConn.database," knex_migrations knex_migrations_lock > ").concat(migrationsDump))}_iteratorAbruptCompletion=false,_didIteratorError=false;_state.label=2;case 2:_state.trys.push([2,12,13,18]);_iterator=_async_iterator(targets);_state.label=3;case 3:return[4,_iterator.next()];case 4:if(!(_iteratorAbruptCompletion=!(_step=_state.sent()).done))return[3,11];_value=_step.value;label=_value.label,config=_value.config,toSkip=_value.toSkip;conn=config.connection;if(toSkip===true){console.log(_chalk.default.red("".concat(label,": Skipped!")));return[3,10]}db=(0,_knex.default)(_object_spread_props(_object_spread({},config),{connection:_object_spread_props(_object_spread({},(_config_connection=config.connection)!==null&&_config_connection!==void 0?_config_connection:{}),{database:undefined})}));return[4,db.raw('SHOW DATABASES LIKE "'.concat(conn.database,'"'))];case 5:_ref1=_sliced_to_array.apply(void 0,[_state.sent(),1]),_ref_1=_sliced_to_array(_ref1[0],1),row=_ref_1[0];if(!row)return[3,7];console.log(_chalk.default.yellow("".concat(label,': Database "').concat(conn.database,'" Already exists')));return[4,db.destroy()];case 6:_state.sent();return[3,10];case 7:console.log("SYNC to ".concat(label,"..."));mysqlCmd="mysql -h".concat(conn.host," -u").concat(conn.user," -p").concat(conn.password);(0,_child_process.execSync)("".concat(mysqlCmd," -e 'DROP DATABASE IF EXISTS `").concat(conn.database,"`'"));(0,_child_process.execSync)("".concat(mysqlCmd," -e 'CREATE DATABASE `").concat(conn.database,"`'"));(0,_child_process.execSync)("".concat(mysqlCmd," ").concat(conn.database," < ").concat(dumpFilename));return[4,(0,_fsutils.exists)(migrationsDump)];case 8:if(_state.sent()){(0,_child_process.execSync)("".concat(mysqlCmd," ").concat(conn.database," < ").concat(migrationsDump))}return[4,db.destroy()];case 9:_state.sent();_state.label=10;case 10:_iteratorAbruptCompletion=false;return[3,3];case 11:return[3,18];case 12:err=_state.sent();_didIteratorError=true;_iteratorError=err;return[3,18];case 13:_state.trys.push([13,,16,17]);if(!(_iteratorAbruptCompletion&&_iterator.return!=null))return[3,15];return[4,_iterator.return()];case 14:_state.sent();_state.label=15;case 15:return[3,17];case 16:if(_didIteratorError){throw _iteratorError}return[7];case 17:return[7];case 18:return[4,_db.destroy()];case 19:_state.sent();return[2]}})})()}function fixture_import(entityId,recordIds){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,setupFixtureManager()];case 1:_state.sent();return[4,_fixturemanager.FixtureManager.importFixture(entityId,recordIds)];case 2:_state.sent();return[4,_fixturemanager.FixtureManager.sync()];case 3:_state.sent();return[2]}})})()}function fixture_sync(){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,setupFixtureManager()];case 1:_state.sent();return[4,_fixturemanager.FixtureManager.sync()];case 2:_state.sent();return[2]}})})()}function stub_practice(name){return _async_to_generator(function(){var practiceDir,fileNames,maxSeqNo,currentSeqNo,fileName,dstPath,code,runCode;return _ts_generator(this,function(_state){switch(_state.label){case 0:practiceDir=_path.default.join(_api.Sonamu.apiRootPath,"src","practices");return[4,(0,_promises.readdir)(practiceDir)];case 1:fileNames=_state.sent();return[4,function(){return _async_to_generator(function(){var filteredSeqs;return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,(0,_fsutils.exists)(practiceDir)];case 1:if(!!_state.sent())return[3,3];return[4,(0,_promises.mkdir)(practiceDir,{recursive:true})];case 2:_state.sent();_state.label=3;case 3:filteredSeqs=fileNames.filter(function(fileName){return fileName.startsWith("p")&&fileName.endsWith(".ts")}).map(function(fileName){var _fileName_match;var _ref=_sliced_to_array((_fileName_match=fileName.match(/^p([0-9]+)\-/))!==null&&_fileName_match!==void 0?_fileName_match:["0","0"],2),seqNo=_ref[1];return parseInt(seqNo)}).sort(function(a,b){return b-a});if(filteredSeqs.length>0){return[2,filteredSeqs[0]]}return[2,0]}})})()}()];case 2:maxSeqNo=_state.sent();currentSeqNo=maxSeqNo+1;fileName="p".concat(currentSeqNo,"-").concat(name,".ts");dstPath=_path.default.join(practiceDir,fileName);code=['import { Sonamu } from "sonamu";',"","console.clear();",'console.log("'.concat(fileName,'");'),"","Sonamu.runScript(async () => {"," // TODO","});",""].join("\n");return[4,(0,_promises.writeFile)(dstPath,code)];case 3:_state.sent();(0,_child_process.execSync)("code ".concat(dstPath));runCode="yarn node -r dotenv/config -r source-map-support/register dist/practices/".concat(fileName.replace(".ts",".js"));console.log("".concat(_chalk.default.blue(runCode)," copied to clipboard."));(0,_child_process.execSync)('echo "'.concat(runCode,'" | pbcopy'));return[2]}})})()}function stub_entity(entityId){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,_api.Sonamu.syncer.createEntity({entityId:entityId})];case 1:_state.sent();return[2]}})})()}function scaffold_model(entityId){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,_api.Sonamu.syncer.generateTemplate("model",{entityId:entityId})];case 1:_state.sent();return[2]}})})()}function scaffold_model_test(entityId){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,_api.Sonamu.syncer.generateTemplate("model_test",{entityId:entityId})];case 1:_state.sent();return[2]}})})()}function ui(){return _async_to_generator(function(){var _Sonamu_config_ui,sonamuUI,_Sonamu_config_projectName,_Sonamu_config_ui_port,e;return _ts_generator(this,function(_state){switch(_state.label){case 0:_state.trys.push([0,2,,3]);return[4,Promise.resolve().then(function(){return /*#__PURE__*/_interop_require_wildcard(require("@sonamu-kit/ui"))})];case 1:sonamuUI=_state.sent();sonamuUI.startServers({projectName:(_Sonamu_config_projectName=_api.Sonamu.config.projectName)!==null&&_Sonamu_config_projectName!==void 0?_Sonamu_config_projectName:_path.default.basename(_api.Sonamu.apiRootPath),apiRootPath:_api.Sonamu.apiRootPath,port:(_Sonamu_config_ui_port=(_Sonamu_config_ui=_api.Sonamu.config.ui)===null||_Sonamu_config_ui===void 0?void 0:_Sonamu_config_ui.port)!==null&&_Sonamu_config_ui_port!==void 0?_Sonamu_config_ui_port:57e3});return[3,3];case 2:e=_state.sent();if(_instanceof(e,Error)&&e.message.includes("isn't declared")){console.log("You need to install ".concat(_chalk.default.blue("@sonamu-kit/ui")," first."));return[2]}throw e;case 3:return[2]}})})()}
|
|
2
|
-
|
|
1
|
+
import chalk from "chalk";
|
|
2
|
+
import dotenv from "dotenv";
|
|
3
|
+
dotenv.config();
|
|
4
|
+
import { execSync, spawn } from "child_process";
|
|
5
|
+
import { mkdir, readdir, rm, writeFile } from "node:fs/promises";
|
|
6
|
+
import knex from "knex";
|
|
7
|
+
import { createRequire } from "module";
|
|
8
|
+
import path from "path";
|
|
9
|
+
import process from "process";
|
|
10
|
+
import { tsicli } from "tsicli";
|
|
11
|
+
import { Sonamu } from "../api/index.js";
|
|
12
|
+
import { EntityManager } from "../entity/entity-manager.js";
|
|
13
|
+
import { Migrator } from "../migration/migrator.js";
|
|
14
|
+
import { FixtureManager } from "../testing/fixture-manager.js";
|
|
15
|
+
import { exists } from "../utils/fs-utils.js";
|
|
16
|
+
import { findApiRootPath } from "../utils/utils.js";
|
|
17
|
+
import { BUILD_DIR, SWC_BUILD_COMMAND, TSC_TYPE_CHECK_COMMAND } from "./build-config.js";
|
|
18
|
+
let migrator;
|
|
19
|
+
async function bootstrap() {
|
|
20
|
+
const notToInit = [
|
|
21
|
+
"dev",
|
|
22
|
+
"build",
|
|
23
|
+
"start"
|
|
24
|
+
].includes(process.argv[2] ?? "");
|
|
25
|
+
if (!notToInit) {
|
|
26
|
+
await Sonamu.init(false, false);
|
|
27
|
+
}
|
|
28
|
+
await tsicli(process.argv, {
|
|
29
|
+
types: {
|
|
30
|
+
"#entityId": {
|
|
31
|
+
type: "autocomplete",
|
|
32
|
+
name: "#entityId",
|
|
33
|
+
message: "Please input #entityId",
|
|
34
|
+
choices: EntityManager.getAllParentIds().map((entityId)=>({
|
|
35
|
+
title: entityId,
|
|
36
|
+
value: entityId
|
|
37
|
+
}))
|
|
38
|
+
},
|
|
39
|
+
"#recordIds": "number[]",
|
|
40
|
+
"#name": "string"
|
|
41
|
+
},
|
|
42
|
+
args: [
|
|
43
|
+
[
|
|
44
|
+
"fixture",
|
|
45
|
+
"init"
|
|
46
|
+
],
|
|
47
|
+
[
|
|
48
|
+
"fixture",
|
|
49
|
+
"import",
|
|
50
|
+
"#entityId",
|
|
51
|
+
"#recordIds"
|
|
52
|
+
],
|
|
53
|
+
[
|
|
54
|
+
"fixture",
|
|
55
|
+
"sync"
|
|
56
|
+
],
|
|
57
|
+
[
|
|
58
|
+
"migrate",
|
|
59
|
+
"run"
|
|
60
|
+
],
|
|
61
|
+
[
|
|
62
|
+
"migrate",
|
|
63
|
+
"check"
|
|
64
|
+
],
|
|
65
|
+
[
|
|
66
|
+
"migrate",
|
|
67
|
+
"rollback"
|
|
68
|
+
],
|
|
69
|
+
[
|
|
70
|
+
"migrate",
|
|
71
|
+
"reset"
|
|
72
|
+
],
|
|
73
|
+
[
|
|
74
|
+
"migrate",
|
|
75
|
+
"clear"
|
|
76
|
+
],
|
|
77
|
+
[
|
|
78
|
+
"migrate",
|
|
79
|
+
"status"
|
|
80
|
+
],
|
|
81
|
+
[
|
|
82
|
+
"stub",
|
|
83
|
+
"practice",
|
|
84
|
+
"#name"
|
|
85
|
+
],
|
|
86
|
+
[
|
|
87
|
+
"stub",
|
|
88
|
+
"entity",
|
|
89
|
+
"#name"
|
|
90
|
+
],
|
|
91
|
+
[
|
|
92
|
+
"scaffold",
|
|
93
|
+
"model",
|
|
94
|
+
"#entityId"
|
|
95
|
+
],
|
|
96
|
+
[
|
|
97
|
+
"scaffold",
|
|
98
|
+
"model_test",
|
|
99
|
+
"#entityId"
|
|
100
|
+
],
|
|
101
|
+
[
|
|
102
|
+
"scaffold",
|
|
103
|
+
"view_list",
|
|
104
|
+
"#entityId"
|
|
105
|
+
],
|
|
106
|
+
[
|
|
107
|
+
"scaffold",
|
|
108
|
+
"view_form",
|
|
109
|
+
"#entityId"
|
|
110
|
+
],
|
|
111
|
+
[
|
|
112
|
+
"ui"
|
|
113
|
+
],
|
|
114
|
+
[
|
|
115
|
+
"sync"
|
|
116
|
+
],
|
|
117
|
+
[
|
|
118
|
+
"dev"
|
|
119
|
+
],
|
|
120
|
+
[
|
|
121
|
+
"build"
|
|
122
|
+
],
|
|
123
|
+
[
|
|
124
|
+
"start"
|
|
125
|
+
]
|
|
126
|
+
],
|
|
127
|
+
runners: {
|
|
128
|
+
migrate_status,
|
|
129
|
+
migrate_run,
|
|
130
|
+
fixture_init,
|
|
131
|
+
fixture_import,
|
|
132
|
+
fixture_sync,
|
|
133
|
+
stub_practice,
|
|
134
|
+
stub_entity,
|
|
135
|
+
scaffold_model,
|
|
136
|
+
scaffold_model_test,
|
|
137
|
+
ui,
|
|
138
|
+
// scaffold_view_list,
|
|
139
|
+
// scaffold_view_form,
|
|
140
|
+
sync,
|
|
141
|
+
dev,
|
|
142
|
+
build,
|
|
143
|
+
start
|
|
144
|
+
}
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
bootstrap().finally(async ()=>{
|
|
148
|
+
await FixtureManager.destroy();
|
|
149
|
+
});
|
|
150
|
+
/**
|
|
151
|
+
* pnpm sync 하면 실행되는 함수입니다.
|
|
152
|
+
* 프로젝트를 싱크합니다.
|
|
153
|
+
*/ async function sync() {
|
|
154
|
+
await Sonamu.syncer.sync();
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* pnpm dev 하면 실행되는 함수입니다.
|
|
158
|
+
* 프로젝트에 대해 HMR 지원하는 개발 서버를 띄워줍니다.
|
|
159
|
+
*
|
|
160
|
+
* TypeScript를 바로 실행할 수 있도록 @sonamu-kit/loader를,
|
|
161
|
+
* HMR을 지원하기 위해 @sonamu-kit/hot-hook을 import하며,
|
|
162
|
+
* 소스맵 지원을 위해 --enable-source-maps 플래그를 포함하여 실행합니다.
|
|
163
|
+
*
|
|
164
|
+
* 이때 @sonamu-kit/loader와 @sonamu-kit/hot-hook는 sonamu가 자체적으로 가지고 있는 dependency입니다.
|
|
165
|
+
* 또한 실행에 사용하는 @sonamu-kit/hot-runner도 마찬가지로 sonamu가 자체적으로 가지고 있는 dependency입니다.
|
|
166
|
+
* 따라서 사용자 프로젝트에서는 이 세 패키지를 직접 설치할 필요가 없습니다.
|
|
167
|
+
*
|
|
168
|
+
* Sonamu.init 없이 호출될 것을 상정하여 구현되었습니다.
|
|
169
|
+
*/ async function dev() {
|
|
170
|
+
const apiRoot = findApiRootPath();
|
|
171
|
+
const entryPoint = "src/index.ts";
|
|
172
|
+
console.log(chalk.yellow.bold("🚀 Starting Sonamu dev server...\n"));
|
|
173
|
+
// 이 sonamu 패키지가 dependencies로 가지고 있는 @sonamu-kit/hot-runner의 bin/run.js를 사용합니다.
|
|
174
|
+
// 이 경로(/bin/run.js)는 @sonamu-kit/hot-runner의 package.json의 bin 필드에 명시되어 있는 그것과 같습니다.
|
|
175
|
+
const hotRunnerBinPath = createRequire(import.meta.url).resolve("@sonamu-kit/hot-runner/bin/run.js");
|
|
176
|
+
const serverProcess = spawn(process.execPath, [
|
|
177
|
+
hotRunnerBinPath,
|
|
178
|
+
"--clear-screen=false",
|
|
179
|
+
"--node-args=--import=sonamu/loader-register",
|
|
180
|
+
"--node-args=--import=sonamu/hot-hook-register",
|
|
181
|
+
"--node-args=--enable-source-maps",
|
|
182
|
+
"--on-key=r:restart:Restart server",
|
|
183
|
+
`--on-key=f:shell(rm ${path.join(apiRoot, "sonamu.lock")}):restart:Force restart`,
|
|
184
|
+
"--on-key=enter:shell(echo hi):Key binding test",
|
|
185
|
+
"--on-key=ctrl+f ctrl+f:shell(git pull && pnpm install && pnpm --filter sonamu build && echo 'Sonamu is now up-to-date!'):restart:Pull & install & build & restart",
|
|
186
|
+
entryPoint
|
|
187
|
+
], {
|
|
188
|
+
cwd: apiRoot,
|
|
189
|
+
stdio: "inherit",
|
|
190
|
+
env: {
|
|
191
|
+
...process.env,
|
|
192
|
+
NODE_ENV: "development",
|
|
193
|
+
HOT: "yes",
|
|
194
|
+
API_ROOT_PATH: apiRoot
|
|
195
|
+
}
|
|
196
|
+
});
|
|
197
|
+
// 종료 처리
|
|
198
|
+
const cleanup = ()=>{
|
|
199
|
+
console.log(chalk.yellow("\n\n👋 Shutting down..."));
|
|
200
|
+
serverProcess.kill("SIGTERM");
|
|
201
|
+
process.exit(0);
|
|
202
|
+
};
|
|
203
|
+
process.on("SIGINT", cleanup);
|
|
204
|
+
process.on("SIGTERM", cleanup);
|
|
205
|
+
serverProcess.on("exit", (code)=>{
|
|
206
|
+
if (code !== 0) {
|
|
207
|
+
console.error(chalk.red(`❌ Server exited with code ${code}`));
|
|
208
|
+
process.exit(code || 1);
|
|
209
|
+
}
|
|
210
|
+
});
|
|
211
|
+
}
|
|
212
|
+
/**
|
|
213
|
+
* pnpm build 하면 실행되는 함수입니다.
|
|
214
|
+
* 프로젝트를 빌드합니다.
|
|
215
|
+
*
|
|
216
|
+
* 빌드에 필요한 .swcrc는 프로젝트 루트에서 찾고, 없으면 sonamu가 관리하는 .swcrc.project-default를 사용합니다.
|
|
217
|
+
* sonamu.config.ts는 src에 들어있지 않기 때문에 SWC_BUILD_COMMAND로 빌드되지 않습니다.
|
|
218
|
+
* 따라서 따로 빌드해줍니다.
|
|
219
|
+
*
|
|
220
|
+
* Sonamu.init 없이 호출될 것을 상정하여 구현되었습니다.
|
|
221
|
+
*/ async function build() {
|
|
222
|
+
const apiRoot = findApiRootPath();
|
|
223
|
+
// 출력 디렉토리를 제거합니다.
|
|
224
|
+
try {
|
|
225
|
+
console.log(chalk.blue("Removing build directory..."));
|
|
226
|
+
if (await exists(BUILD_DIR)) {
|
|
227
|
+
await rm(BUILD_DIR, {
|
|
228
|
+
recursive: true,
|
|
229
|
+
force: true
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
} catch (error) {
|
|
233
|
+
console.error(chalk.red("Remove build directory failed."), error);
|
|
234
|
+
process.exit(1);
|
|
235
|
+
}
|
|
236
|
+
// .swcrc 파일을 지정합니다.
|
|
237
|
+
let swcFilePath = ".swcrc";
|
|
238
|
+
try {
|
|
239
|
+
if (await exists(swcFilePath)) {
|
|
240
|
+
// 사용자 프로젝트에 .swcrc가 있으면 우선으로 사용합니다.
|
|
241
|
+
console.log(chalk.blue("Using .swcrc from project root..."));
|
|
242
|
+
} else {
|
|
243
|
+
// 아니라면 sonamu가 관리하는 .swcrc.project-default를 가져다 씁니다.
|
|
244
|
+
console.log(chalk.blue("Using default .swcrc from sonamu package..."));
|
|
245
|
+
swcFilePath = path.join(import.meta.dirname, "..", "..", ".swcrc.project-default");
|
|
246
|
+
}
|
|
247
|
+
} catch (error) {
|
|
248
|
+
console.error(chalk.red("Setting up swc config file failed."), error);
|
|
249
|
+
process.exit(1);
|
|
250
|
+
}
|
|
251
|
+
// 소스 디렉토리를 빌드합니다.
|
|
252
|
+
try {
|
|
253
|
+
console.log(chalk.blue("Building with swc..."));
|
|
254
|
+
execSync(SWC_BUILD_COMMAND(swcFilePath), {
|
|
255
|
+
cwd: apiRoot,
|
|
256
|
+
stdio: "inherit"
|
|
257
|
+
});
|
|
258
|
+
} catch (error) {
|
|
259
|
+
console.error(chalk.red("Build failed."), error);
|
|
260
|
+
process.exit(1);
|
|
261
|
+
}
|
|
262
|
+
// sonamu.config.ts만 따로 빌드합니다.
|
|
263
|
+
// 이 친구는 src에 들어있지 않기 때문에 SWC_BUILD_COMMAND로 빌드되지 않습니다.
|
|
264
|
+
// 따라서 따로 빌드해줍니다.
|
|
265
|
+
try {
|
|
266
|
+
const configPath = path.join(apiRoot, "sonamu.config.ts");
|
|
267
|
+
if (await exists(configPath)) {
|
|
268
|
+
console.log(chalk.blue("Building sonamu.config.ts..."));
|
|
269
|
+
execSync(`swc ${configPath} -o ${BUILD_DIR}/sonamu.config.js`, {
|
|
270
|
+
cwd: apiRoot,
|
|
271
|
+
stdio: "inherit"
|
|
272
|
+
});
|
|
273
|
+
}
|
|
274
|
+
} catch (error) {
|
|
275
|
+
console.error(chalk.red("Building sonamu.config.ts failed."), error);
|
|
276
|
+
process.exit(1);
|
|
277
|
+
}
|
|
278
|
+
// 마지막에는 타입 체크를 해요.
|
|
279
|
+
try {
|
|
280
|
+
console.log(chalk.blue("Checking types with tsc..."));
|
|
281
|
+
execSync(TSC_TYPE_CHECK_COMMAND, {
|
|
282
|
+
cwd: apiRoot,
|
|
283
|
+
stdio: "inherit"
|
|
284
|
+
});
|
|
285
|
+
} catch (error) {
|
|
286
|
+
console.error(chalk.red("Type check failed."), error);
|
|
287
|
+
process.exit(1);
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
/**
|
|
291
|
+
* pnpm start 하면 실행되는 함수입니다.
|
|
292
|
+
* 빌드된 프로젝트를 실행합니다.
|
|
293
|
+
*
|
|
294
|
+
* 빌드된 결과물(dist 디렉토리의 index.js 엔트리포인트)이 없다면 실행을 중단합니다.
|
|
295
|
+
* 소스맵 지원과 dotenv 지원을 포함하여 실행합니다.
|
|
296
|
+
*
|
|
297
|
+
* Sonamu.init 없이 호출될 것을 상정하여 구현되었습니다.
|
|
298
|
+
*/ async function start() {
|
|
299
|
+
const apiRoot = findApiRootPath();
|
|
300
|
+
const entryPoint = "dist/index.js";
|
|
301
|
+
if (!await exists(entryPoint)) {
|
|
302
|
+
console.log(chalk.red(`${entryPoint} not found. Please build your project first.`));
|
|
303
|
+
console.log(chalk.blue("Run: yarn sonamu build"));
|
|
304
|
+
return;
|
|
305
|
+
}
|
|
306
|
+
const { spawn } = await import("child_process");
|
|
307
|
+
const serverProcess = spawn(process.execPath, [
|
|
308
|
+
"--enable-source-maps",
|
|
309
|
+
"-r",
|
|
310
|
+
"dotenv/config",
|
|
311
|
+
entryPoint
|
|
312
|
+
], {
|
|
313
|
+
cwd: apiRoot,
|
|
314
|
+
stdio: "inherit"
|
|
315
|
+
});
|
|
316
|
+
process.on("SIGINT", ()=>{
|
|
317
|
+
serverProcess.kill("SIGTERM");
|
|
318
|
+
process.exit(0);
|
|
319
|
+
});
|
|
320
|
+
}
|
|
321
|
+
async function setupMigrator() {
|
|
322
|
+
// migrator
|
|
323
|
+
migrator = new Migrator();
|
|
324
|
+
}
|
|
325
|
+
async function setupFixtureManager() {
|
|
326
|
+
FixtureManager.init();
|
|
327
|
+
}
|
|
328
|
+
async function migrate_run() {
|
|
329
|
+
await setupMigrator();
|
|
330
|
+
await migrator.runAction("apply", Object.keys(Sonamu.dbConfig));
|
|
331
|
+
}
|
|
332
|
+
async function migrate_status() {
|
|
333
|
+
await setupMigrator();
|
|
334
|
+
const status = await migrator.getStatus();
|
|
335
|
+
// status;
|
|
336
|
+
console.log(status);
|
|
337
|
+
}
|
|
338
|
+
async function fixture_init() {
|
|
339
|
+
const srcConfig = Sonamu.dbConfig.development_master;
|
|
340
|
+
const targets = [
|
|
341
|
+
{
|
|
342
|
+
label: "(REMOTE) Fixture DB",
|
|
343
|
+
config: Sonamu.dbConfig.fixture_remote
|
|
344
|
+
},
|
|
345
|
+
{
|
|
346
|
+
label: "(LOCAL) Testing DB",
|
|
347
|
+
config: Sonamu.dbConfig.test,
|
|
348
|
+
toSkip: (()=>{
|
|
349
|
+
const remoteConn = Sonamu.dbConfig.fixture_remote.connection;
|
|
350
|
+
const localConn = Sonamu.dbConfig.test.connection;
|
|
351
|
+
return remoteConn.host === localConn.host && remoteConn.database === localConn.database;
|
|
352
|
+
})()
|
|
353
|
+
}
|
|
354
|
+
];
|
|
355
|
+
// 1. 기준DB 스키마를 덤프
|
|
356
|
+
console.log("DUMP...");
|
|
357
|
+
const dumpFilename = `/tmp/sonamu-fixture-init-${Date.now()}.sql`;
|
|
358
|
+
const srcConn = srcConfig.connection;
|
|
359
|
+
const migrationsDump = `/tmp/sonamu-fixture-init-migrations-${Date.now()}.sql`;
|
|
360
|
+
execSync(`mysqldump -h${srcConn.host} -u${srcConn.user} -p${srcConn.password} --single-transaction -d --no-create-db --triggers ${srcConn.database} > ${dumpFilename}`);
|
|
361
|
+
const _db = knex(srcConfig);
|
|
362
|
+
const [[migrations]] = await _db.raw("SELECT COUNT(*) as count FROM information_schema.tables WHERE table_schema = ? AND table_name = 'knex_migrations'", [
|
|
363
|
+
srcConn.database
|
|
364
|
+
]);
|
|
365
|
+
if (migrations.count > 0) {
|
|
366
|
+
execSync(`mysqldump -h${srcConn.host} -u${srcConn.user} -p${srcConn.password} --single-transaction --no-create-db --triggers ${srcConn.database} knex_migrations knex_migrations_lock > ${migrationsDump}`);
|
|
367
|
+
}
|
|
368
|
+
// 2. 대상DB 각각에 대하여 존재여부 확인 후 붓기
|
|
369
|
+
for await (const { label, config, toSkip } of targets){
|
|
370
|
+
const conn = config.connection;
|
|
371
|
+
if (toSkip === true) {
|
|
372
|
+
console.log(chalk.red(`${label}: Skipped!`));
|
|
373
|
+
continue;
|
|
374
|
+
}
|
|
375
|
+
const db = knex({
|
|
376
|
+
...config,
|
|
377
|
+
connection: {
|
|
378
|
+
...config.connection ?? {},
|
|
379
|
+
database: undefined
|
|
380
|
+
}
|
|
381
|
+
});
|
|
382
|
+
const [[row]] = await db.raw(`SHOW DATABASES LIKE "${conn.database}"`);
|
|
383
|
+
if (row) {
|
|
384
|
+
console.log(chalk.yellow(`${label}: Database "${conn.database}" Already exists`));
|
|
385
|
+
await db.destroy();
|
|
386
|
+
continue;
|
|
387
|
+
}
|
|
388
|
+
console.log(`SYNC to ${label}...`);
|
|
389
|
+
const mysqlCmd = `mysql -h${conn.host} -u${conn.user} -p${conn.password}`;
|
|
390
|
+
execSync(`${mysqlCmd} -e 'DROP DATABASE IF EXISTS \`${conn.database}\`'`);
|
|
391
|
+
execSync(`${mysqlCmd} -e 'CREATE DATABASE \`${conn.database}\`'`);
|
|
392
|
+
execSync(`${mysqlCmd} ${conn.database} < ${dumpFilename}`);
|
|
393
|
+
if (await exists(migrationsDump)) {
|
|
394
|
+
execSync(`${mysqlCmd} ${conn.database} < ${migrationsDump}`);
|
|
395
|
+
}
|
|
396
|
+
await db.destroy();
|
|
397
|
+
}
|
|
398
|
+
await _db.destroy();
|
|
399
|
+
}
|
|
400
|
+
async function fixture_import(entityId, recordIds) {
|
|
401
|
+
await setupFixtureManager();
|
|
402
|
+
await FixtureManager.importFixture(entityId, recordIds);
|
|
403
|
+
await FixtureManager.sync();
|
|
404
|
+
}
|
|
405
|
+
async function fixture_sync() {
|
|
406
|
+
await setupFixtureManager();
|
|
407
|
+
await FixtureManager.sync();
|
|
408
|
+
}
|
|
409
|
+
async function stub_practice(name) {
|
|
410
|
+
const practiceDir = path.join(Sonamu.apiRootPath, "src", "practices");
|
|
411
|
+
const fileNames = await readdir(practiceDir);
|
|
412
|
+
const maxSeqNo = await (async ()=>{
|
|
413
|
+
if (!await exists(practiceDir)) {
|
|
414
|
+
await mkdir(practiceDir, {
|
|
415
|
+
recursive: true
|
|
416
|
+
});
|
|
417
|
+
}
|
|
418
|
+
const filteredSeqs = fileNames.filter((fileName)=>fileName.startsWith("p") && fileName.endsWith(".ts")).map((fileName)=>{
|
|
419
|
+
const [, seqNo] = fileName.match(/^p([0-9]+)-/) ?? [
|
|
420
|
+
"0",
|
|
421
|
+
"0"
|
|
422
|
+
];
|
|
423
|
+
return parseInt(seqNo);
|
|
424
|
+
}).sort((a, b)=>b - a);
|
|
425
|
+
if (filteredSeqs.length > 0) {
|
|
426
|
+
return filteredSeqs[0];
|
|
427
|
+
}
|
|
428
|
+
return 0;
|
|
429
|
+
})();
|
|
430
|
+
const currentSeqNo = maxSeqNo + 1;
|
|
431
|
+
const fileName = `p${currentSeqNo}-${name}.ts`;
|
|
432
|
+
const dstPath = path.join(practiceDir, fileName);
|
|
433
|
+
const code = [
|
|
434
|
+
`import { Sonamu } from "sonamu";`,
|
|
435
|
+
"",
|
|
436
|
+
`console.clear();`,
|
|
437
|
+
`console.log("${fileName}");`,
|
|
438
|
+
"",
|
|
439
|
+
`Sonamu.runScript(async () => {`,
|
|
440
|
+
` // TODO`,
|
|
441
|
+
`});`,
|
|
442
|
+
""
|
|
443
|
+
].join("\n");
|
|
444
|
+
await writeFile(dstPath, code);
|
|
445
|
+
execSync(`code ${dstPath}`);
|
|
446
|
+
const runCode = `yarn node -r dotenv/config --enable-source-maps dist/practices/${fileName.replace(".ts", ".js")}`;
|
|
447
|
+
console.log(`${chalk.blue(runCode)} copied to clipboard.`);
|
|
448
|
+
execSync(`echo "${runCode}" | pbcopy`);
|
|
449
|
+
}
|
|
450
|
+
async function stub_entity(entityId) {
|
|
451
|
+
await Sonamu.syncer.createEntity({
|
|
452
|
+
entityId,
|
|
453
|
+
title: entityId
|
|
454
|
+
});
|
|
455
|
+
}
|
|
456
|
+
async function scaffold_model(entityId) {
|
|
457
|
+
await Sonamu.syncer.generateTemplate("model", {
|
|
458
|
+
entityId
|
|
459
|
+
});
|
|
460
|
+
}
|
|
461
|
+
async function scaffold_model_test(entityId) {
|
|
462
|
+
await Sonamu.syncer.generateTemplate("model_test", {
|
|
463
|
+
entityId
|
|
464
|
+
});
|
|
465
|
+
}
|
|
466
|
+
async function ui() {
|
|
467
|
+
try {
|
|
468
|
+
// 사용자 프로젝트의 패키지들 중에서 @sonamu-kit/ui를 찾습니다.
|
|
469
|
+
// 이를 위해서 createRequire를 사용하여 프로젝트 경로 기준으로 resolve합니다.
|
|
470
|
+
const projectRequire = createRequire(path.join(Sonamu.apiRootPath, "package.json"));
|
|
471
|
+
const uiPackagePath = projectRequire.resolve("@sonamu-kit/ui"); // 없으면 여기서 터져요(MODULE_NOT_FOUND)
|
|
472
|
+
const uiNodePath = path.join(path.dirname(uiPackagePath), "run-ui.js");
|
|
473
|
+
if (!await exists(uiNodePath)) {
|
|
474
|
+
console.log(chalk.red(`UI runner script not found at ${uiNodePath}. Please rebuild @sonamu-kit/ui.`));
|
|
475
|
+
return;
|
|
476
|
+
}
|
|
477
|
+
// UI를 별도 프로세스로 실행 (hot-hook 활성화)
|
|
478
|
+
const uiProcess = spawn(process.execPath, [
|
|
479
|
+
"--import",
|
|
480
|
+
"sonamu/loader-register",
|
|
481
|
+
"--import",
|
|
482
|
+
"sonamu/hot-hook-register",
|
|
483
|
+
"--enable-source-maps",
|
|
484
|
+
"--no-warnings",
|
|
485
|
+
uiNodePath
|
|
486
|
+
], {
|
|
487
|
+
stdio: "inherit",
|
|
488
|
+
env: {
|
|
489
|
+
...process.env,
|
|
490
|
+
HOT: "yes",
|
|
491
|
+
PROJECT_NAME: Sonamu.config.projectName ?? path.basename(Sonamu.apiRootPath),
|
|
492
|
+
API_ROOT_PATH: Sonamu.apiRootPath,
|
|
493
|
+
UI_PORT: (Sonamu.config.ui?.port ?? 57000).toString()
|
|
494
|
+
}
|
|
495
|
+
});
|
|
496
|
+
// 종료 처리
|
|
497
|
+
const cleanup = ()=>{
|
|
498
|
+
console.log(chalk.yellow("\n\n👋 Shutting down UI server..."));
|
|
499
|
+
uiProcess.kill("SIGTERM");
|
|
500
|
+
process.exit(0);
|
|
501
|
+
};
|
|
502
|
+
process.on("SIGINT", cleanup);
|
|
503
|
+
process.on("SIGTERM", cleanup);
|
|
504
|
+
uiProcess.on("exit", (code)=>{
|
|
505
|
+
if (code !== 0) {
|
|
506
|
+
console.error(chalk.red(`❌ UI server exited with code ${code}`));
|
|
507
|
+
process.exit(code || 1);
|
|
508
|
+
}
|
|
509
|
+
});
|
|
510
|
+
} catch (e) {
|
|
511
|
+
if (e instanceof Error && e.message.includes("isn't declared")) {
|
|
512
|
+
console.log(`You need to install ${chalk.blue(`@sonamu-kit/ui`)} first.`);
|
|
513
|
+
return;
|
|
514
|
+
}
|
|
515
|
+
throw e;
|
|
516
|
+
}
|
|
517
|
+
}
|
|
518
|
+
|
|
519
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9iaW4vY2xpLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBjaGFsayBmcm9tIFwiY2hhbGtcIjtcbmltcG9ydCBkb3RlbnYgZnJvbSBcImRvdGVudlwiO1xuXG5kb3RlbnYuY29uZmlnKCk7XG5cbmltcG9ydCB7IGV4ZWNTeW5jLCBzcGF3biB9IGZyb20gXCJjaGlsZF9wcm9jZXNzXCI7XG5pbXBvcnQgeyBta2RpciwgcmVhZGRpciwgcm0sIHdyaXRlRmlsZSB9IGZyb20gXCJmcy9wcm9taXNlc1wiO1xuaW1wb3J0IGtuZXgsIHsgdHlwZSBLbmV4IH0gZnJvbSBcImtuZXhcIjtcbmltcG9ydCB7IGNyZWF0ZVJlcXVpcmUgfSBmcm9tIFwibW9kdWxlXCI7XG5pbXBvcnQgcGF0aCBmcm9tIFwicGF0aFwiO1xuaW1wb3J0IHByb2Nlc3MgZnJvbSBcInByb2Nlc3NcIjtcbmltcG9ydCB7IHRzaWNsaSB9IGZyb20gXCJ0c2ljbGlcIjtcbmltcG9ydCB7IFNvbmFtdSB9IGZyb20gXCIuLi9hcGlcIjtcbmltcG9ydCB0eXBlIHsgU29uYW11REJDb25maWcgfSBmcm9tIFwiLi4vZGF0YWJhc2UvZGJcIjtcbmltcG9ydCB7IEVudGl0eU1hbmFnZXIgfSBmcm9tIFwiLi4vZW50aXR5L2VudGl0eS1tYW5hZ2VyXCI7XG5pbXBvcnQgeyBNaWdyYXRvciB9IGZyb20gXCIuLi9taWdyYXRpb24vbWlncmF0b3JcIjtcbmltcG9ydCB7IEZpeHR1cmVNYW5hZ2VyIH0gZnJvbSBcIi4uL3Rlc3RpbmcvZml4dHVyZS1tYW5hZ2VyXCI7XG5pbXBvcnQgeyBleGlzdHMgfSBmcm9tIFwiLi4vdXRpbHMvZnMtdXRpbHNcIjtcbmltcG9ydCB7IGZpbmRBcGlSb290UGF0aCB9IGZyb20gXCIuLi91dGlscy91dGlsc1wiO1xuaW1wb3J0IHsgQlVJTERfRElSLCBTV0NfQlVJTERfQ09NTUFORCwgVFNDX1RZUEVfQ0hFQ0tfQ09NTUFORCB9IGZyb20gXCIuL2J1aWxkLWNvbmZpZ1wiO1xuXG5sZXQgbWlncmF0b3I6IE1pZ3JhdG9yO1xuXG5hc3luYyBmdW5jdGlvbiBib290c3RyYXAoKSB7XG4gIGNvbnN0IG5vdFRvSW5pdCA9IFtcImRldlwiLCBcImJ1aWxkXCIsIFwic3RhcnRcIl0uaW5jbHVkZXMocHJvY2Vzcy5hcmd2WzJdID8/IFwiXCIpO1xuICBpZiAoIW5vdFRvSW5pdCkge1xuICAgIGF3YWl0IFNvbmFtdS5pbml0KGZhbHNlLCBmYWxzZSk7XG4gIH1cblxuICBhd2FpdCB0c2ljbGkocHJvY2Vzcy5hcmd2LCB7XG4gICAgdHlwZXM6IHtcbiAgICAgIFwiI2VudGl0eUlkXCI6IHtcbiAgICAgICAgdHlwZTogXCJhdXRvY29tcGxldGVcIixcbiAgICAgICAgbmFtZTogXCIjZW50aXR5SWRcIixcbiAgICAgICAgbWVzc2FnZTogXCJQbGVhc2UgaW5wdXQgI2VudGl0eUlkXCIsXG4gICAgICAgIGNob2ljZXM6IEVudGl0eU1hbmFnZXIuZ2V0QWxsUGFyZW50SWRzKCkubWFwKChlbnRpdHlJZCkgPT4gKHtcbiAgICAgICAgICB0aXRsZTogZW50aXR5SWQsXG4gICAgICAgICAgdmFsdWU6IGVudGl0eUlkLFxuICAgICAgICB9KSksXG4gICAgICB9LFxuICAgICAgXCIjcmVjb3JkSWRzXCI6IFwibnVtYmVyW11cIixcbiAgICAgIFwiI25hbWVcIjogXCJzdHJpbmdcIixcbiAgICB9LFxuICAgIGFyZ3M6IFtcbiAgICAgIFtcImZpeHR1cmVcIiwgXCJpbml0XCJdLFxuICAgICAgW1wiZml4dHVyZVwiLCBcImltcG9ydFwiLCBcIiNlbnRpdHlJZFwiLCBcIiNyZWNvcmRJZHNcIl0sXG4gICAgICBbXCJmaXh0dXJlXCIsIFwic3luY1wiXSxcbiAgICAgIFtcIm1pZ3JhdGVcIiwgXCJydW5cIl0sXG4gICAgICBbXCJtaWdyYXRlXCIsIFwiY2hlY2tcIl0sXG4gICAgICBbXCJtaWdyYXRlXCIsIFwicm9sbGJhY2tcIl0sXG4gICAgICBbXCJtaWdyYXRlXCIsIFwicmVzZXRcIl0sXG4gICAgICBbXCJtaWdyYXRlXCIsIFwiY2xlYXJcIl0sXG4gICAgICBbXCJtaWdyYXRlXCIsIFwic3RhdHVzXCJdLFxuICAgICAgW1wic3R1YlwiLCBcInByYWN0aWNlXCIsIFwiI25hbWVcIl0sXG4gICAgICBbXCJzdHViXCIsIFwiZW50aXR5XCIsIFwiI25hbWVcIl0sXG4gICAgICBbXCJzY2FmZm9sZFwiLCBcIm1vZGVsXCIsIFwiI2VudGl0eUlkXCJdLFxuICAgICAgW1wic2NhZmZvbGRcIiwgXCJtb2RlbF90ZXN0XCIsIFwiI2VudGl0eUlkXCJdLFxuICAgICAgW1wic2NhZmZvbGRcIiwgXCJ2aWV3X2xpc3RcIiwgXCIjZW50aXR5SWRcIl0sXG4gICAgICBbXCJzY2FmZm9sZFwiLCBcInZpZXdfZm9ybVwiLCBcIiNlbnRpdHlJZFwiXSxcbiAgICAgIFtcInVpXCJdLFxuICAgICAgW1wic3luY1wiXSxcbiAgICAgIFtcImRldlwiXSxcbiAgICAgIFtcImJ1aWxkXCJdLFxuICAgICAgW1wic3RhcnRcIl0sXG4gICAgXSxcbiAgICBydW5uZXJzOiB7XG4gICAgICBtaWdyYXRlX3N0YXR1cyxcbiAgICAgIG1pZ3JhdGVfcnVuLFxuICAgICAgZml4dHVyZV9pbml0LFxuICAgICAgZml4dHVyZV9pbXBvcnQsXG4gICAgICBmaXh0dXJlX3N5bmMsXG4gICAgICBzdHViX3ByYWN0aWNlLFxuICAgICAgc3R1Yl9lbnRpdHksXG4gICAgICBzY2FmZm9sZF9tb2RlbCxcbiAgICAgIHNjYWZmb2xkX21vZGVsX3Rlc3QsXG4gICAgICB1aSxcbiAgICAgIC8vIHNjYWZmb2xkX3ZpZXdfbGlzdCxcbiAgICAgIC8vIHNjYWZmb2xkX3ZpZXdfZm9ybSxcbiAgICAgIHN5bmMsXG4gICAgICBkZXYsXG4gICAgICBidWlsZCxcbiAgICAgIHN0YXJ0LFxuICAgIH0sXG4gIH0pO1xufVxuXG5ib290c3RyYXAoKS5maW5hbGx5KGFzeW5jICgpID0+IHtcbiAgYXdhaXQgRml4dHVyZU1hbmFnZXIuZGVzdHJveSgpO1xufSk7XG5cbi8qKlxuICogcG5wbSBzeW5jIO2VmOuptCDsi6TtlonrkJjripQg7ZWo7IiY7J6F64uI64ukLlxuICog7ZSE66Gc7KCd7Yq466W8IOyLse2BrO2VqeuLiOuLpC5cbiAqL1xuYXN5bmMgZnVuY3Rpb24gc3luYygpIHtcbiAgYXdhaXQgU29uYW11LnN5bmNlci5zeW5jKCk7XG59XG5cbi8qKlxuICogcG5wbSBkZXYg7ZWY66m0IOyLpO2WieuQmOuKlCDtlajsiJjsnoXri4jri6QuXG4gKiDtlITroZzsoJ3tirjsl5Ag64yA7ZW0IEhNUiDsp4Dsm5DtlZjripQg6rCc67CcIOyEnOuyhOulvCDrnYTsm4zspI3ri4jri6QuXG4gKlxuICogVHlwZVNjcmlwdOulvCDrsJTroZwg7Iuk7ZaJ7ZWgIOyImCDsnojrj4TroZ0gQHNvbmFtdS1raXQvbG9hZGVy66W8LFxuICogSE1S7J2EIOyngOybkO2VmOq4sCDsnITtlbQgQHNvbmFtdS1raXQvaG90LWhvb2vsnYQgaW1wb3J07ZWY66mwLFxuICog7IaM7Iqk66e1IOyngOybkOydhCDsnITtlbQgLS1lbmFibGUtc291cmNlLW1hcHMg7ZSM656Y6re466W8IO2PrO2VqO2VmOyXrCDsi6Ttlontlanri4jri6QuXG4gKlxuICog7J2065WMIEBzb25hbXUta2l0L2xvYWRlcuyZgCBAc29uYW11LWtpdC9ob3QtaG9va+uKlCBzb25hbXXqsIAg7J6Q7LK07KCB7Jy866GcIOqwgOyngOqzoCDsnojripQgZGVwZW5kZW5jeeyeheuLiOuLpC5cbiAqIOuYkO2VnCDsi6Ttlonsl5Ag7IKs7Jqp7ZWY64qUIEBzb25hbXUta2l0L2hvdC1ydW5uZXLrj4Qg66eI7LCs6rCA7KeA66GcIHNvbmFtdeqwgCDsnpDssrTsoIHsnLzroZwg6rCA7KeA6rOgIOyeiOuKlCBkZXBlbmRlbmN57J6F64uI64ukLlxuICog65Sw65287IScIOyCrOyaqeyekCDtlITroZzsoJ3tirjsl5DshJzripQg7J20IOyEuCDtjKjtgqTsp4Drpbwg7KeB7KCRIOyEpOy5mO2VoCDtlYTsmpTqsIAg7JeG7Iq164uI64ukLlxuICpcbiAqIFNvbmFtdS5pbml0IOyXhuydtCDtmLjstpzrkKAg6rKD7J2EIOyDgeygle2VmOyXrCDqtaztmITrkJjsl4jsirXri4jri6QuXG4gKi9cbmFzeW5jIGZ1bmN0aW9uIGRldigpIHtcbiAgY29uc3QgYXBpUm9vdCA9IGZpbmRBcGlSb290UGF0aCgpO1xuICBjb25zdCBlbnRyeVBvaW50ID0gXCJzcmMvaW5kZXgudHNcIjtcblxuICBjb25zb2xlLmxvZyhjaGFsay55ZWxsb3cuYm9sZChcIvCfmoAgU3RhcnRpbmcgU29uYW11IGRldiBzZXJ2ZXIuLi5cXG5cIikpO1xuXG4gIC8vIOydtCBzb25hbXUg7Yyo7YKk7KeA6rCAIGRlcGVuZGVuY2llc+uhnCDqsIDsp4Dqs6Ag7J6I64qUIEBzb25hbXUta2l0L2hvdC1ydW5uZXLsnZggYmluL3J1bi5qc+ulvCDsgqzsmqntlanri4jri6QuXG4gIC8vIOydtCDqsr3roZwoL2Jpbi9ydW4uanMp64qUIEBzb25hbXUta2l0L2hvdC1ydW5uZXLsnZggcGFja2FnZS5qc29u7J2YIGJpbiDtlYTrk5zsl5Ag66qF7Iuc65CY7Ja0IOyeiOuKlCDqt7jqsoPqs7wg6rCZ7Iq164uI64ukLlxuICBjb25zdCBob3RSdW5uZXJCaW5QYXRoID0gY3JlYXRlUmVxdWlyZShpbXBvcnQubWV0YS51cmwpLnJlc29sdmUoXG4gICAgXCJAc29uYW11LWtpdC9ob3QtcnVubmVyL2Jpbi9ydW4uanNcIixcbiAgKTtcblxuICBjb25zdCBzZXJ2ZXJQcm9jZXNzID0gc3Bhd24oXG4gICAgcHJvY2Vzcy5leGVjUGF0aCwgLy8gbm9kZVxuICAgIFtcbiAgICAgIGhvdFJ1bm5lckJpblBhdGgsIC8vIOydtOugh+qyjCDtlbTshJwgaG90LXJ1bm5lcuulvCDsi6TtlontlZjqtazsmpRcbiAgICAgIFwiLS1jbGVhci1zY3JlZW49ZmFsc2VcIiwgLy8g7J207ZWYIGhvdC1ydW5uZXLsl5Dqsowg64SY6rKo7KSEIOyduOyekOuTpOyeheuLiOuLpC5cbiAgICAgIFwiLS1ub2RlLWFyZ3M9LS1pbXBvcnQ9c29uYW11L2xvYWRlci1yZWdpc3RlclwiLCAvLyBUeXBlU2NyaXB0IOyEnO2PrO2KuOulvCDsnITtlZwg66Gc642ULFxuICAgICAgXCItLW5vZGUtYXJncz0tLWltcG9ydD1zb25hbXUvaG90LWhvb2stcmVnaXN0ZXJcIiwgLy8gSE1S7J2EIOyngOybkO2VmOq4sCDsnITtlZwgaG90LWhvb2ssXG4gICAgICBcIi0tbm9kZS1hcmdzPS0tZW5hYmxlLXNvdXJjZS1tYXBzXCIsIC8vIOq3uOumrOqzoCDshozsiqTrp7Ug7KeA7JuQ7J2EIOychO2VnCDtlIzrnpjqt7jsnoXri4jri6QuXG4gICAgICBcIi0tb24ta2V5PXI6cmVzdGFydDpSZXN0YXJ0IHNlcnZlclwiLCAvLyByIOuIhOultOuptCDshJzrsoQg7J6s7Iuc7J6R7ZWY6rKMIO2VtOykmOyalC5cbiAgICAgIGAtLW9uLWtleT1mOnNoZWxsKHJtICR7cGF0aC5qb2luKGFwaVJvb3QsIFwic29uYW11LmxvY2tcIil9KTpyZXN0YXJ0OkZvcmNlIHJlc3RhcnRgLCAvLyBmIOuIhOultOuptCBzb25hbXUubG9jayDtjIzsnbzsnYQg7KeA7Jqw6rOgIOyEnOuyhCDsnqzsi5zsnpHtlZjqsowg7ZW07KSY7JqULlxuXG4gICAgICBcIi0tb24ta2V5PWVudGVyOnNoZWxsKGVjaG8gaGkpOktleSBiaW5kaW5nIHRlc3RcIiwgLy8gZW50ZXLrpbwga2V566GcIOyTuCDsiJgg7J6I7J2M7J2EIOuztOydtOq4sCDsnITtlZwg7YWM7Iqk7Yq47J6F64uI64ukLlxuICAgICAgXCItLW9uLWtleT1jdHJsK2YgY3RybCtmOnNoZWxsKGdpdCBwdWxsICYmIHBucG0gaW5zdGFsbCAmJiBwbnBtIC0tZmlsdGVyIHNvbmFtdSBidWlsZCAmJiBlY2hvICdTb25hbXUgaXMgbm93IHVwLXRvLWRhdGUhJyk6cmVzdGFydDpQdWxsICYgaW5zdGFsbCAmIGJ1aWxkICYgcmVzdGFydFwiLCAvLyBtb2RpZmllcuyZgOydmCDsobDtlaksIOq3uOumrOqzoCDrkZAg6rCc7J2YIGNob3Jk66W8IOyCrOyaqe2VoCDsiJgg7J6I7J2M7J2EIOuztOydtOq4sCDsnITtlZwg7YWM7Iqk7Yq47J6F64uI64ukLlxuICAgICAgZW50cnlQb2ludCwgLy8g66eI7KeA66eJ7Jy866GcIOyLpOygnCDsi6TtlontlaAg7Iqk7YGs66a97Yq47J2YIOqyveuhnOulvCDrhJjqsqjspI3ri4jri6QuXG4gICAgXSxcbiAgICB7XG4gICAgICBjd2Q6IGFwaVJvb3QsXG4gICAgICBzdGRpbzogXCJpbmhlcml0XCIsXG4gICAgICBlbnY6IHtcbiAgICAgICAgLi4ucHJvY2Vzcy5lbnYsXG4gICAgICAgIE5PREVfRU5WOiBcImRldmVsb3BtZW50XCIsXG4gICAgICAgIEhPVDogXCJ5ZXNcIiwgLy8g7JaY6rCAIOyeiOyWtOyVvCBITVLsnbQg7Zmc7ISx7ZmU65Cp64uI64ukLlxuICAgICAgICBBUElfUk9PVF9QQVRIOiBhcGlSb290LCAvLyDsnbQg6rK966Gc6rCAIGhvdC1ob29r7J2YIOujqO2KuCDrlJTroInthqDrpqzqsIAg65Cp64uI64ukLlxuICAgICAgfSxcbiAgICB9LFxuICApO1xuXG4gIC8vIOyiheujjCDsspjrpqxcbiAgY29uc3QgY2xlYW51cCA9ICgpID0+IHtcbiAgICBjb25zb2xlLmxvZyhjaGFsay55ZWxsb3coXCJcXG5cXG7wn5GLIFNodXR0aW5nIGRvd24uLi5cIikpO1xuICAgIHNlcnZlclByb2Nlc3Mua2lsbChcIlNJR1RFUk1cIik7XG4gICAgcHJvY2Vzcy5leGl0KDApO1xuICB9O1xuXG4gIHByb2Nlc3Mub24oXCJTSUdJTlRcIiwgY2xlYW51cCk7XG4gIHByb2Nlc3Mub24oXCJTSUdURVJNXCIsIGNsZWFudXApO1xuXG4gIHNlcnZlclByb2Nlc3Mub24oXCJleGl0XCIsIChjb2RlKSA9PiB7XG4gICAgaWYgKGNvZGUgIT09IDApIHtcbiAgICAgIGNvbnNvbGUuZXJyb3IoY2hhbGsucmVkKGDinYwgU2VydmVyIGV4aXRlZCB3aXRoIGNvZGUgJHtjb2RlfWApKTtcbiAgICAgIHByb2Nlc3MuZXhpdChjb2RlIHx8IDEpO1xuICAgIH1cbiAgfSk7XG59XG5cbi8qKlxuICogcG5wbSBidWlsZCDtlZjrqbQg7Iuk7ZaJ65CY64qUIO2VqOyImOyeheuLiOuLpC5cbiAqIO2UhOuhnOygne2KuOulvCDruYzrk5ztlanri4jri6QuXG4gKlxuICog67mM65Oc7JeQIO2VhOyalO2VnCAuc3djcmPripQg7ZSE66Gc7KCd7Yq4IOujqO2KuOyXkOyEnCDssL7qs6AsIOyXhuycvOuptCBzb25hbXXqsIAg6rSA66as7ZWY64qUIC5zd2NyYy5wcm9qZWN0LWRlZmF1bHTrpbwg7IKs7Jqp7ZWp64uI64ukLlxuICogc29uYW11LmNvbmZpZy50c+uKlCBzcmPsl5Ag65Ok7Ja07J6I7KeAIOyViuq4sCDrlYzrrLjsl5AgU1dDX0JVSUxEX0NPTU1BTkTroZwg67mM65Oc65CY7KeAIOyViuyKteuLiOuLpC5cbiAqIOuUsOudvOyEnCDrlLDroZwg67mM65Oc7ZW07KSN64uI64ukLlxuICpcbiAqIFNvbmFtdS5pbml0IOyXhuydtCDtmLjstpzrkKAg6rKD7J2EIOyDgeygle2VmOyXrCDqtaztmITrkJjsl4jsirXri4jri6QuXG4gKi9cbmFzeW5jIGZ1bmN0aW9uIGJ1aWxkKCkge1xuICBjb25zdCBhcGlSb290ID0gZmluZEFwaVJvb3RQYXRoKCk7XG5cbiAgLy8g7Lac66ClIOuUlOugie2GoOumrOulvCDsoJzqsbDtlanri4jri6QuXG4gIHRyeSB7XG4gICAgY29uc29sZS5sb2coY2hhbGsuYmx1ZShcIlJlbW92aW5nIGJ1aWxkIGRpcmVjdG9yeS4uLlwiKSk7XG4gICAgaWYgKGF3YWl0IGV4aXN0cyhCVUlMRF9ESVIpKSB7XG4gICAgICBhd2FpdCBybShCVUlMRF9ESVIsIHsgcmVjdXJzaXZlOiB0cnVlLCBmb3JjZTogdHJ1ZSB9KTtcbiAgICB9XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihjaGFsay5yZWQoXCJSZW1vdmUgYnVpbGQgZGlyZWN0b3J5IGZhaWxlZC5cIiksIGVycm9yKTtcbiAgICBwcm9jZXNzLmV4aXQoMSk7XG4gIH1cblxuICAvLyAuc3djcmMg7YyM7J287J2EIOyngOygle2VqeuLiOuLpC5cbiAgbGV0IHN3Y0ZpbGVQYXRoID0gXCIuc3djcmNcIjtcbiAgdHJ5IHtcbiAgICBpZiAoYXdhaXQgZXhpc3RzKHN3Y0ZpbGVQYXRoKSkge1xuICAgICAgLy8g7IKs7Jqp7J6QIO2UhOuhnOygne2KuOyXkCAuc3djcmPqsIAg7J6I7Jy866m0IOyasOyEoOycvOuhnCDsgqzsmqntlanri4jri6QuXG4gICAgICBjb25zb2xlLmxvZyhjaGFsay5ibHVlKFwiVXNpbmcgLnN3Y3JjIGZyb20gcHJvamVjdCByb290Li4uXCIpKTtcbiAgICB9IGVsc2Uge1xuICAgICAgLy8g7JWE64uI652866m0IHNvbmFtdeqwgCDqtIDrpqztlZjripQgLnN3Y3JjLnByb2plY3QtZGVmYXVsdOulvCDqsIDsoLjri6Qg7JSB64uI64ukLlxuICAgICAgY29uc29sZS5sb2coY2hhbGsuYmx1ZShcIlVzaW5nIGRlZmF1bHQgLnN3Y3JjIGZyb20gc29uYW11IHBhY2thZ2UuLi5cIikpO1xuICAgICAgc3djRmlsZVBhdGggPSBwYXRoLmpvaW4oaW1wb3J0Lm1ldGEuZGlybmFtZSwgXCIuLlwiLCBcIi4uXCIsIFwiLnN3Y3JjLnByb2plY3QtZGVmYXVsdFwiKTtcbiAgICB9XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihjaGFsay5yZWQoXCJTZXR0aW5nIHVwIHN3YyBjb25maWcgZmlsZSBmYWlsZWQuXCIpLCBlcnJvcik7XG4gICAgcHJvY2Vzcy5leGl0KDEpO1xuICB9XG5cbiAgLy8g7IaM7IqkIOuUlOugie2GoOumrOulvCDruYzrk5ztlanri4jri6QuXG4gIHRyeSB7XG4gICAgY29uc29sZS5sb2coY2hhbGsuYmx1ZShcIkJ1aWxkaW5nIHdpdGggc3djLi4uXCIpKTtcbiAgICBleGVjU3luYyhTV0NfQlVJTERfQ09NTUFORChzd2NGaWxlUGF0aCksIHsgY3dkOiBhcGlSb290LCBzdGRpbzogXCJpbmhlcml0XCIgfSk7XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihjaGFsay5yZWQoXCJCdWlsZCBmYWlsZWQuXCIpLCBlcnJvcik7XG4gICAgcHJvY2Vzcy5leGl0KDEpO1xuICB9XG5cbiAgLy8gc29uYW11LmNvbmZpZy50c+unjCDrlLDroZwg67mM65Oc7ZWp64uI64ukLlxuICAvLyDsnbQg7Lmc6rWs64qUIHNyY+yXkCDrk6TslrTsnojsp4Ag7JWK6riwIOuVjOusuOyXkCBTV0NfQlVJTERfQ09NTUFOROuhnCDruYzrk5zrkJjsp4Ag7JWK7Iq164uI64ukLlxuICAvLyDrlLDrnbzshJwg65Sw66GcIOu5jOuTnO2VtOykjeuLiOuLpC5cbiAgdHJ5IHtcbiAgICBjb25zdCBjb25maWdQYXRoID0gcGF0aC5qb2luKGFwaVJvb3QsIFwic29uYW11LmNvbmZpZy50c1wiKTtcbiAgICBpZiAoYXdhaXQgZXhpc3RzKGNvbmZpZ1BhdGgpKSB7XG4gICAgICBjb25zb2xlLmxvZyhjaGFsay5ibHVlKFwiQnVpbGRpbmcgc29uYW11LmNvbmZpZy50cy4uLlwiKSk7XG4gICAgICBleGVjU3luYyhgc3djICR7Y29uZmlnUGF0aH0gLW8gJHtCVUlMRF9ESVJ9L3NvbmFtdS5jb25maWcuanNgLCB7XG4gICAgICAgIGN3ZDogYXBpUm9vdCxcbiAgICAgICAgc3RkaW86IFwiaW5oZXJpdFwiLFxuICAgICAgfSk7XG4gICAgfVxuICB9IGNhdGNoIChlcnJvcikge1xuICAgIGNvbnNvbGUuZXJyb3IoY2hhbGsucmVkKFwiQnVpbGRpbmcgc29uYW11LmNvbmZpZy50cyBmYWlsZWQuXCIpLCBlcnJvcik7XG4gICAgcHJvY2Vzcy5leGl0KDEpO1xuICB9XG5cbiAgLy8g66eI7KeA66eJ7JeQ64qUIO2DgOyehSDssrTtgazrpbwg7ZW07JqULlxuICB0cnkge1xuICAgIGNvbnNvbGUubG9nKGNoYWxrLmJsdWUoXCJDaGVja2luZyB0eXBlcyB3aXRoIHRzYy4uLlwiKSk7XG4gICAgZXhlY1N5bmMoVFNDX1RZUEVfQ0hFQ0tfQ09NTUFORCwge1xuICAgICAgY3dkOiBhcGlSb290LFxuICAgICAgc3RkaW86IFwiaW5oZXJpdFwiLFxuICAgIH0pO1xuICB9IGNhdGNoIChlcnJvcikge1xuICAgIGNvbnNvbGUuZXJyb3IoY2hhbGsucmVkKFwiVHlwZSBjaGVjayBmYWlsZWQuXCIpLCBlcnJvcik7XG4gICAgcHJvY2Vzcy5leGl0KDEpO1xuICB9XG59XG5cbi8qKlxuICogcG5wbSBzdGFydCDtlZjrqbQg7Iuk7ZaJ65CY64qUIO2VqOyImOyeheuLiOuLpC5cbiAqIOu5jOuTnOuQnCDtlITroZzsoJ3tirjrpbwg7Iuk7ZaJ7ZWp64uI64ukLlxuICpcbiAqIOu5jOuTnOuQnCDqsrDqs7zrrLwoZGlzdCDrlJTroInthqDrpqzsnZggaW5kZXguanMg7JeU7Yq466as7Y+s7J247Yq4KeydtCDsl4bri6TrqbQg7Iuk7ZaJ7J2EIOykkeuLqO2VqeuLiOuLpC5cbiAqIOyGjOyKpOuntSDsp4Dsm5Dqs7wgZG90ZW52IOyngOybkOydhCDtj6ztlajtlZjsl6wg7Iuk7ZaJ7ZWp64uI64ukLlxuICpcbiAqIFNvbmFtdS5pbml0IOyXhuydtCDtmLjstpzrkKAg6rKD7J2EIOyDgeygle2VmOyXrCDqtaztmITrkJjsl4jsirXri4jri6QuXG4gKi9cbmFzeW5jIGZ1bmN0aW9uIHN0YXJ0KCkge1xuICBjb25zdCBhcGlSb290ID0gZmluZEFwaVJvb3RQYXRoKCk7XG4gIGNvbnN0IGVudHJ5UG9pbnQgPSBcImRpc3QvaW5kZXguanNcIjtcblxuICBpZiAoIShhd2FpdCBleGlzdHMoZW50cnlQb2ludCkpKSB7XG4gICAgY29uc29sZS5sb2coY2hhbGsucmVkKGAke2VudHJ5UG9pbnR9IG5vdCBmb3VuZC4gUGxlYXNlIGJ1aWxkIHlvdXIgcHJvamVjdCBmaXJzdC5gKSk7XG4gICAgY29uc29sZS5sb2coY2hhbGsuYmx1ZShcIlJ1bjogeWFybiBzb25hbXUgYnVpbGRcIikpO1xuICAgIHJldHVybjtcbiAgfVxuXG4gIGNvbnN0IHsgc3Bhd24gfSA9IGF3YWl0IGltcG9ydChcImNoaWxkX3Byb2Nlc3NcIik7XG4gIGNvbnN0IHNlcnZlclByb2Nlc3MgPSBzcGF3bihcbiAgICBwcm9jZXNzLmV4ZWNQYXRoLFxuICAgIFtcIi0tZW5hYmxlLXNvdXJjZS1tYXBzXCIsIFwiLXJcIiwgXCJkb3RlbnYvY29uZmlnXCIsIGVudHJ5UG9pbnRdLFxuICAgIHtcbiAgICAgIGN3ZDogYXBpUm9vdCxcbiAgICAgIHN0ZGlvOiBcImluaGVyaXRcIixcbiAgICB9LFxuICApO1xuXG4gIHByb2Nlc3Mub24oXCJTSUdJTlRcIiwgKCkgPT4ge1xuICAgIHNlcnZlclByb2Nlc3Mua2lsbChcIlNJR1RFUk1cIik7XG4gICAgcHJvY2Vzcy5leGl0KDApO1xuICB9KTtcbn1cblxuYXN5bmMgZnVuY3Rpb24gc2V0dXBNaWdyYXRvcigpIHtcbiAgLy8gbWlncmF0b3JcbiAgbWlncmF0b3IgPSBuZXcgTWlncmF0b3IoKTtcbn1cblxuYXN5bmMgZnVuY3Rpb24gc2V0dXBGaXh0dXJlTWFuYWdlcigpIHtcbiAgRml4dHVyZU1hbmFnZXIuaW5pdCgpO1xufVxuXG5hc3luYyBmdW5jdGlvbiBtaWdyYXRlX3J1bigpIHtcbiAgYXdhaXQgc2V0dXBNaWdyYXRvcigpO1xuXG4gIGF3YWl0IG1pZ3JhdG9yLnJ1bkFjdGlvbihcbiAgICBcImFwcGx5XCIsXG4gICAgT2JqZWN0LmtleXMoU29uYW11LmRiQ29uZmlnKSBhcyAoa2V5b2YgU29uYW11REJDb25maWcpW10gLyrsi7kg64ukISovLFxuICApO1xufVxuXG5hc3luYyBmdW5jdGlvbiBtaWdyYXRlX3N0YXR1cygpIHtcbiAgYXdhaXQgc2V0dXBNaWdyYXRvcigpO1xuXG4gIGNvbnN0IHN0YXR1cyA9IGF3YWl0IG1pZ3JhdG9yLmdldFN0YXR1cygpO1xuICAvLyBzdGF0dXM7XG4gIGNvbnNvbGUubG9nKHN0YXR1cyk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIGZpeHR1cmVfaW5pdCgpIHtcbiAgY29uc3Qgc3JjQ29uZmlnID0gU29uYW11LmRiQ29uZmlnLmRldmVsb3BtZW50X21hc3RlcjtcbiAgY29uc3QgdGFyZ2V0cyA9IFtcbiAgICB7XG4gICAgICBsYWJlbDogXCIoUkVNT1RFKSBGaXh0dXJlIERCXCIsXG4gICAgICBjb25maWc6IFNvbmFtdS5kYkNvbmZpZy5maXh0dXJlX3JlbW90ZSxcbiAgICB9LFxuICAgIHtcbiAgICAgIGxhYmVsOiBcIihMT0NBTCkgVGVzdGluZyBEQlwiLFxuICAgICAgY29uZmlnOiBTb25hbXUuZGJDb25maWcudGVzdCxcbiAgICAgIHRvU2tpcDogKCgpID0+IHtcbiAgICAgICAgY29uc3QgcmVtb3RlQ29ubiA9IFNvbmFtdS5kYkNvbmZpZy5maXh0dXJlX3JlbW90ZS5jb25uZWN0aW9uIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZztcbiAgICAgICAgY29uc3QgbG9jYWxDb25uID0gU29uYW11LmRiQ29uZmlnLnRlc3QuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWc7XG4gICAgICAgIHJldHVybiByZW1vdGVDb25uLmhvc3QgPT09IGxvY2FsQ29ubi5ob3N0ICYmIHJlbW90ZUNvbm4uZGF0YWJhc2UgPT09IGxvY2FsQ29ubi5kYXRhYmFzZTtcbiAgICAgIH0pKCksXG4gICAgfSxcbiAgXSBhcyB7XG4gICAgbGFiZWw6IHN0cmluZztcbiAgICBjb25maWc6IEtuZXguQ29uZmlnO1xuICAgIHRvU2tpcD86IGJvb2xlYW47XG4gIH1bXTtcblxuICAvLyAxLiDquLDspIBEQiDsiqTtgqTrp4jrpbwg642k7ZSEXG4gIGNvbnNvbGUubG9nKFwiRFVNUC4uLlwiKTtcbiAgY29uc3QgZHVtcEZpbGVuYW1lID0gYC90bXAvc29uYW11LWZpeHR1cmUtaW5pdC0ke0RhdGUubm93KCl9LnNxbGA7XG4gIGNvbnN0IHNyY0Nvbm4gPSBzcmNDb25maWcuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWc7XG4gIGNvbnN0IG1pZ3JhdGlvbnNEdW1wID0gYC90bXAvc29uYW11LWZpeHR1cmUtaW5pdC1taWdyYXRpb25zLSR7RGF0ZS5ub3coKX0uc3FsYDtcbiAgZXhlY1N5bmMoXG4gICAgYG15c3FsZHVtcCAtaCR7c3JjQ29ubi5ob3N0fSAtdSR7c3JjQ29ubi51c2VyfSAtcCR7c3JjQ29ubi5wYXNzd29yZH0gLS1zaW5nbGUtdHJhbnNhY3Rpb24gLWQgLS1uby1jcmVhdGUtZGIgLS10cmlnZ2VycyAke3NyY0Nvbm4uZGF0YWJhc2V9ID4gJHtkdW1wRmlsZW5hbWV9YCxcbiAgKTtcbiAgY29uc3QgX2RiID0ga25leChzcmNDb25maWcpO1xuICBjb25zdCBbW21pZ3JhdGlvbnNdXSA9IGF3YWl0IF9kYi5yYXcoXG4gICAgXCJTRUxFQ1QgQ09VTlQoKikgYXMgY291bnQgRlJPTSBpbmZvcm1hdGlvbl9zY2hlbWEudGFibGVzIFdIRVJFIHRhYmxlX3NjaGVtYSA9ID8gQU5EIHRhYmxlX25hbWUgPSAna25leF9taWdyYXRpb25zJ1wiLFxuICAgIFtzcmNDb25uLmRhdGFiYXNlXSxcbiAgKTtcbiAgaWYgKG1pZ3JhdGlvbnMuY291bnQgPiAwKSB7XG4gICAgZXhlY1N5bmMoXG4gICAgICBgbXlzcWxkdW1wIC1oJHtzcmNDb25uLmhvc3R9IC11JHtzcmNDb25uLnVzZXJ9IC1wJHtzcmNDb25uLnBhc3N3b3JkfSAtLXNpbmdsZS10cmFuc2FjdGlvbiAtLW5vLWNyZWF0ZS1kYiAtLXRyaWdnZXJzICR7c3JjQ29ubi5kYXRhYmFzZX0ga25leF9taWdyYXRpb25zIGtuZXhfbWlncmF0aW9uc19sb2NrID4gJHttaWdyYXRpb25zRHVtcH1gLFxuICAgICk7XG4gIH1cblxuICAvLyAyLiDrjIDsg4FEQiDqsIHqsIHsl5Ag64yA7ZWY7JesIOyhtOyerOyXrOu2gCDtmZXsnbgg7ZuEIOu2k+q4sFxuICBmb3IgYXdhaXQgKGNvbnN0IHsgbGFiZWwsIGNvbmZpZywgdG9Ta2lwIH0gb2YgdGFyZ2V0cykge1xuICAgIGNvbnN0IGNvbm4gPSBjb25maWcuY29ubmVjdGlvbiBhcyBLbmV4LkNvbm5lY3Rpb25Db25maWc7XG5cbiAgICBpZiAodG9Ta2lwID09PSB0cnVlKSB7XG4gICAgICBjb25zb2xlLmxvZyhjaGFsay5yZWQoYCR7bGFiZWx9OiBTa2lwcGVkIWApKTtcbiAgICAgIGNvbnRpbnVlO1xuICAgIH1cblxuICAgIGNvbnN0IGRiID0ga25leCh7XG4gICAgICAuLi5jb25maWcsXG4gICAgICBjb25uZWN0aW9uOiB7XG4gICAgICAgIC4uLigoY29uZmlnLmNvbm5lY3Rpb24gPz8ge30pIGFzIEtuZXguQ29ubmVjdGlvbkNvbmZpZyksXG4gICAgICAgIGRhdGFiYXNlOiB1bmRlZmluZWQsXG4gICAgICB9LFxuICAgIH0pO1xuICAgIGNvbnN0IFtbcm93XV0gPSBhd2FpdCBkYi5yYXcoYFNIT1cgREFUQUJBU0VTIExJS0UgXCIke2Nvbm4uZGF0YWJhc2V9XCJgKTtcbiAgICBpZiAocm93KSB7XG4gICAgICBjb25zb2xlLmxvZyhjaGFsay55ZWxsb3coYCR7bGFiZWx9OiBEYXRhYmFzZSBcIiR7Y29ubi5kYXRhYmFzZX1cIiBBbHJlYWR5IGV4aXN0c2ApKTtcbiAgICAgIGF3YWl0IGRiLmRlc3Ryb3koKTtcbiAgICAgIGNvbnRpbnVlO1xuICAgIH1cblxuICAgIGNvbnNvbGUubG9nKGBTWU5DIHRvICR7bGFiZWx9Li4uYCk7XG4gICAgY29uc3QgbXlzcWxDbWQgPSBgbXlzcWwgLWgke2Nvbm4uaG9zdH0gLXUke2Nvbm4udXNlcn0gLXAke2Nvbm4ucGFzc3dvcmR9YDtcbiAgICBleGVjU3luYyhgJHtteXNxbENtZH0gLWUgJ0RST1AgREFUQUJBU0UgSUYgRVhJU1RTIFxcYCR7Y29ubi5kYXRhYmFzZX1cXGAnYCk7XG4gICAgZXhlY1N5bmMoYCR7bXlzcWxDbWR9IC1lICdDUkVBVEUgREFUQUJBU0UgXFxgJHtjb25uLmRhdGFiYXNlfVxcYCdgKTtcbiAgICBleGVjU3luYyhgJHtteXNxbENtZH0gJHtjb25uLmRhdGFiYXNlfSA8ICR7ZHVtcEZpbGVuYW1lfWApO1xuICAgIGlmIChhd2FpdCBleGlzdHMobWlncmF0aW9uc0R1bXApKSB7XG4gICAgICBleGVjU3luYyhgJHtteXNxbENtZH0gJHtjb25uLmRhdGFiYXNlfSA8ICR7bWlncmF0aW9uc0R1bXB9YCk7XG4gICAgfVxuXG4gICAgYXdhaXQgZGIuZGVzdHJveSgpO1xuICB9XG5cbiAgYXdhaXQgX2RiLmRlc3Ryb3koKTtcbn1cblxuYXN5bmMgZnVuY3Rpb24gZml4dHVyZV9pbXBvcnQoZW50aXR5SWQ6IHN0cmluZywgcmVjb3JkSWRzOiBudW1iZXJbXSkge1xuICBhd2FpdCBzZXR1cEZpeHR1cmVNYW5hZ2VyKCk7XG5cbiAgYXdhaXQgRml4dHVyZU1hbmFnZXIuaW1wb3J0Rml4dHVyZShlbnRpdHlJZCwgcmVjb3JkSWRzKTtcbiAgYXdhaXQgRml4dHVyZU1hbmFnZXIuc3luYygpO1xufVxuXG5hc3luYyBmdW5jdGlvbiBmaXh0dXJlX3N5bmMoKSB7XG4gIGF3YWl0IHNldHVwRml4dHVyZU1hbmFnZXIoKTtcblxuICBhd2FpdCBGaXh0dXJlTWFuYWdlci5zeW5jKCk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIHN0dWJfcHJhY3RpY2UobmFtZTogc3RyaW5nKSB7XG4gIGNvbnN0IHByYWN0aWNlRGlyID0gcGF0aC5qb2luKFNvbmFtdS5hcGlSb290UGF0aCwgXCJzcmNcIiwgXCJwcmFjdGljZXNcIik7XG4gIGNvbnN0IGZpbGVOYW1lcyA9IGF3YWl0IHJlYWRkaXIocHJhY3RpY2VEaXIpO1xuXG4gIGNvbnN0IG1heFNlcU5vID0gYXdhaXQgKGFzeW5jICgpID0+IHtcbiAgICBpZiAoIShhd2FpdCBleGlzdHMocHJhY3RpY2VEaXIpKSkge1xuICAgICAgYXdhaXQgbWtkaXIocHJhY3RpY2VEaXIsIHsgcmVjdXJzaXZlOiB0cnVlIH0pO1xuICAgIH1cblxuICAgIGNvbnN0IGZpbHRlcmVkU2VxcyA9IGZpbGVOYW1lc1xuICAgICAgLmZpbHRlcigoZmlsZU5hbWUpID0+IGZpbGVOYW1lLnN0YXJ0c1dpdGgoXCJwXCIpICYmIGZpbGVOYW1lLmVuZHNXaXRoKFwiLnRzXCIpKVxuICAgICAgLm1hcCgoZmlsZU5hbWUpID0+IHtcbiAgICAgICAgY29uc3QgWywgc2VxTm9dID0gZmlsZU5hbWUubWF0Y2goL15wKFswLTldKyktLykgPz8gW1wiMFwiLCBcIjBcIl07XG4gICAgICAgIHJldHVybiBwYXJzZUludChzZXFObyk7XG4gICAgICB9KVxuICAgICAgLnNvcnQoKGEsIGIpID0+IGIgLSBhKTtcblxuICAgIGlmIChmaWx0ZXJlZFNlcXMubGVuZ3RoID4gMCkge1xuICAgICAgcmV0dXJuIGZpbHRlcmVkU2Vxc1swXTtcbiAgICB9XG5cbiAgICByZXR1cm4gMDtcbiAgfSkoKTtcblxuICBjb25zdCBjdXJyZW50U2VxTm8gPSBtYXhTZXFObyArIDE7XG4gIGNvbnN0IGZpbGVOYW1lID0gYHAke2N1cnJlbnRTZXFOb30tJHtuYW1lfS50c2A7XG4gIGNvbnN0IGRzdFBhdGggPSBwYXRoLmpvaW4ocHJhY3RpY2VEaXIsIGZpbGVOYW1lKTtcblxuICBjb25zdCBjb2RlID0gW1xuICAgIGBpbXBvcnQgeyBTb25hbXUgfSBmcm9tIFwic29uYW11XCI7YCxcbiAgICBcIlwiLFxuICAgIGBjb25zb2xlLmNsZWFyKCk7YCxcbiAgICBgY29uc29sZS5sb2coXCIke2ZpbGVOYW1lfVwiKTtgLFxuICAgIFwiXCIsXG4gICAgYFNvbmFtdS5ydW5TY3JpcHQoYXN5bmMgKCkgPT4ge2AsXG4gICAgYCAvLyBUT0RPYCxcbiAgICBgfSk7YCxcbiAgICBcIlwiLFxuICBdLmpvaW4oXCJcXG5cIik7XG4gIGF3YWl0IHdyaXRlRmlsZShkc3RQYXRoLCBjb2RlKTtcblxuICBleGVjU3luYyhgY29kZSAke2RzdFBhdGh9YCk7XG5cbiAgY29uc3QgcnVuQ29kZSA9IGB5YXJuIG5vZGUgLXIgZG90ZW52L2NvbmZpZyAtLWVuYWJsZS1zb3VyY2UtbWFwcyBkaXN0L3ByYWN0aWNlcy8ke2ZpbGVOYW1lLnJlcGxhY2UoXG4gICAgXCIudHNcIixcbiAgICBcIi5qc1wiLFxuICApfWA7XG4gIGNvbnNvbGUubG9nKGAke2NoYWxrLmJsdWUocnVuQ29kZSl9IGNvcGllZCB0byBjbGlwYm9hcmQuYCk7XG4gIGV4ZWNTeW5jKGBlY2hvIFwiJHtydW5Db2RlfVwiIHwgcGJjb3B5YCk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIHN0dWJfZW50aXR5KGVudGl0eUlkOiBzdHJpbmcpIHtcbiAgYXdhaXQgU29uYW11LnN5bmNlci5jcmVhdGVFbnRpdHkoeyBlbnRpdHlJZCwgdGl0bGU6IGVudGl0eUlkIH0pO1xufVxuXG5hc3luYyBmdW5jdGlvbiBzY2FmZm9sZF9tb2RlbChlbnRpdHlJZDogc3RyaW5nKSB7XG4gIGF3YWl0IFNvbmFtdS5zeW5jZXIuZ2VuZXJhdGVUZW1wbGF0ZShcIm1vZGVsXCIsIHtcbiAgICBlbnRpdHlJZCxcbiAgfSk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIHNjYWZmb2xkX21vZGVsX3Rlc3QoZW50aXR5SWQ6IHN0cmluZykge1xuICBhd2FpdCBTb25hbXUuc3luY2VyLmdlbmVyYXRlVGVtcGxhdGUoXCJtb2RlbF90ZXN0XCIsIHtcbiAgICBlbnRpdHlJZCxcbiAgfSk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIHVpKCkge1xuICB0cnkge1xuICAgIC8vIOyCrOyaqeyekCDtlITroZzsoJ3tirjsnZgg7Yyo7YKk7KeA65OkIOykkeyXkOyEnCBAc29uYW11LWtpdC91aeulvCDssL7sirXri4jri6QuXG4gICAgLy8g7J2066W8IOychO2VtOyEnCBjcmVhdGVSZXF1aXJl66W8IOyCrOyaqe2VmOyXrCDtlITroZzsoJ3tirgg6rK966GcIOq4sOykgOycvOuhnCByZXNvbHZl7ZWp64uI64ukLlxuICAgIGNvbnN0IHByb2plY3RSZXF1aXJlID0gY3JlYXRlUmVxdWlyZShwYXRoLmpvaW4oU29uYW11LmFwaVJvb3RQYXRoLCBcInBhY2thZ2UuanNvblwiKSk7XG4gICAgY29uc3QgdWlQYWNrYWdlUGF0aCA9IHByb2plY3RSZXF1aXJlLnJlc29sdmUoXCJAc29uYW11LWtpdC91aVwiKTsgLy8g7JeG7Jy866m0IOyXrOq4sOyEnCDthLDsoLjsmpQoTU9EVUxFX05PVF9GT1VORClcbiAgICBjb25zdCB1aU5vZGVQYXRoID0gcGF0aC5qb2luKHBhdGguZGlybmFtZSh1aVBhY2thZ2VQYXRoKSwgXCJydW4tdWkuanNcIik7XG5cbiAgICBpZiAoIShhd2FpdCBleGlzdHModWlOb2RlUGF0aCkpKSB7XG4gICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgY2hhbGsucmVkKGBVSSBydW5uZXIgc2NyaXB0IG5vdCBmb3VuZCBhdCAke3VpTm9kZVBhdGh9LiBQbGVhc2UgcmVidWlsZCBAc29uYW11LWtpdC91aS5gKSxcbiAgICAgICk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLy8gVUnrpbwg67OE64+EIO2UhOuhnOyEuOyKpOuhnCDsi6TtlokgKGhvdC1ob29rIO2ZnOyEse2ZlClcbiAgICBjb25zdCB1aVByb2Nlc3MgPSBzcGF3bihcbiAgICAgIHByb2Nlc3MuZXhlY1BhdGgsXG4gICAgICBbXG4gICAgICAgIFwiLS1pbXBvcnRcIixcbiAgICAgICAgXCJzb25hbXUvbG9hZGVyLXJlZ2lzdGVyXCIsXG4gICAgICAgIFwiLS1pbXBvcnRcIixcbiAgICAgICAgXCJzb25hbXUvaG90LWhvb2stcmVnaXN0ZXJcIixcbiAgICAgICAgXCItLWVuYWJsZS1zb3VyY2UtbWFwc1wiLFxuICAgICAgICBcIi0tbm8td2FybmluZ3NcIixcbiAgICAgICAgdWlOb2RlUGF0aCxcbiAgICAgIF0sXG4gICAgICB7XG4gICAgICAgIHN0ZGlvOiBcImluaGVyaXRcIixcbiAgICAgICAgZW52OiB7XG4gICAgICAgICAgLi4ucHJvY2Vzcy5lbnYsXG4gICAgICAgICAgSE9UOiBcInllc1wiLFxuICAgICAgICAgIFBST0pFQ1RfTkFNRTogU29uYW11LmNvbmZpZy5wcm9qZWN0TmFtZSA/PyBwYXRoLmJhc2VuYW1lKFNvbmFtdS5hcGlSb290UGF0aCksXG4gICAgICAgICAgQVBJX1JPT1RfUEFUSDogU29uYW11LmFwaVJvb3RQYXRoLFxuICAgICAgICAgIFVJX1BPUlQ6IChTb25hbXUuY29uZmlnLnVpPy5wb3J0ID8/IDU3MDAwKS50b1N0cmluZygpLFxuICAgICAgICB9LFxuICAgICAgfSxcbiAgICApO1xuXG4gICAgLy8g7KKF66OMIOyymOumrFxuICAgIGNvbnN0IGNsZWFudXAgPSAoKSA9PiB7XG4gICAgICBjb25zb2xlLmxvZyhjaGFsay55ZWxsb3coXCJcXG5cXG7wn5GLIFNodXR0aW5nIGRvd24gVUkgc2VydmVyLi4uXCIpKTtcbiAgICAgIHVpUHJvY2Vzcy5raWxsKFwiU0lHVEVSTVwiKTtcbiAgICAgIHByb2Nlc3MuZXhpdCgwKTtcbiAgICB9O1xuXG4gICAgcHJvY2Vzcy5vbihcIlNJR0lOVFwiLCBjbGVhbnVwKTtcbiAgICBwcm9jZXNzLm9uKFwiU0lHVEVSTVwiLCBjbGVhbnVwKTtcblxuICAgIHVpUHJvY2Vzcy5vbihcImV4aXRcIiwgKGNvZGUpID0+IHtcbiAgICAgIGlmIChjb2RlICE9PSAwKSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoY2hhbGsucmVkKGDinYwgVUkgc2VydmVyIGV4aXRlZCB3aXRoIGNvZGUgJHtjb2RlfWApKTtcbiAgICAgICAgcHJvY2Vzcy5leGl0KGNvZGUgfHwgMSk7XG4gICAgICB9XG4gICAgfSk7XG4gIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICBpZiAoZSBpbnN0YW5jZW9mIEVycm9yICYmIGUubWVzc2FnZS5pbmNsdWRlcyhcImlzbid0IGRlY2xhcmVkXCIpKSB7XG4gICAgICBjb25zb2xlLmxvZyhgWW91IG5lZWQgdG8gaW5zdGFsbCAke2NoYWxrLmJsdWUoYEBzb25hbXUta2l0L3VpYCl9IGZpcnN0LmApO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aHJvdyBlO1xuICB9XG59XG4iXSwibmFtZXMiOlsiY2hhbGsiLCJkb3RlbnYiLCJjb25maWciLCJleGVjU3luYyIsInNwYXduIiwibWtkaXIiLCJyZWFkZGlyIiwicm0iLCJ3cml0ZUZpbGUiLCJrbmV4IiwiY3JlYXRlUmVxdWlyZSIsInBhdGgiLCJwcm9jZXNzIiwidHNpY2xpIiwiU29uYW11IiwiRW50aXR5TWFuYWdlciIsIk1pZ3JhdG9yIiwiRml4dHVyZU1hbmFnZXIiLCJleGlzdHMiLCJmaW5kQXBpUm9vdFBhdGgiLCJCVUlMRF9ESVIiLCJTV0NfQlVJTERfQ09NTUFORCIsIlRTQ19UWVBFX0NIRUNLX0NPTU1BTkQiLCJtaWdyYXRvciIsImJvb3RzdHJhcCIsIm5vdFRvSW5pdCIsImluY2x1ZGVzIiwiYXJndiIsImluaXQiLCJ0eXBlcyIsInR5cGUiLCJuYW1lIiwibWVzc2FnZSIsImNob2ljZXMiLCJnZXRBbGxQYXJlbnRJZHMiLCJtYXAiLCJlbnRpdHlJZCIsInRpdGxlIiwidmFsdWUiLCJhcmdzIiwicnVubmVycyIsIm1pZ3JhdGVfc3RhdHVzIiwibWlncmF0ZV9ydW4iLCJmaXh0dXJlX2luaXQiLCJmaXh0dXJlX2ltcG9ydCIsImZpeHR1cmVfc3luYyIsInN0dWJfcHJhY3RpY2UiLCJzdHViX2VudGl0eSIsInNjYWZmb2xkX21vZGVsIiwic2NhZmZvbGRfbW9kZWxfdGVzdCIsInVpIiwic3luYyIsImRldiIsImJ1aWxkIiwic3RhcnQiLCJmaW5hbGx5IiwiZGVzdHJveSIsInN5bmNlciIsImFwaVJvb3QiLCJlbnRyeVBvaW50IiwiY29uc29sZSIsImxvZyIsInllbGxvdyIsImJvbGQiLCJob3RSdW5uZXJCaW5QYXRoIiwidXJsIiwicmVzb2x2ZSIsInNlcnZlclByb2Nlc3MiLCJleGVjUGF0aCIsImpvaW4iLCJjd2QiLCJzdGRpbyIsImVudiIsIk5PREVfRU5WIiwiSE9UIiwiQVBJX1JPT1RfUEFUSCIsImNsZWFudXAiLCJraWxsIiwiZXhpdCIsIm9uIiwiY29kZSIsImVycm9yIiwicmVkIiwiYmx1ZSIsInJlY3Vyc2l2ZSIsImZvcmNlIiwic3djRmlsZVBhdGgiLCJkaXJuYW1lIiwiY29uZmlnUGF0aCIsInNldHVwTWlncmF0b3IiLCJzZXR1cEZpeHR1cmVNYW5hZ2VyIiwicnVuQWN0aW9uIiwiT2JqZWN0Iiwia2V5cyIsImRiQ29uZmlnIiwic3RhdHVzIiwiZ2V0U3RhdHVzIiwic3JjQ29uZmlnIiwiZGV2ZWxvcG1lbnRfbWFzdGVyIiwidGFyZ2V0cyIsImxhYmVsIiwiZml4dHVyZV9yZW1vdGUiLCJ0ZXN0IiwidG9Ta2lwIiwicmVtb3RlQ29ubiIsImNvbm5lY3Rpb24iLCJsb2NhbENvbm4iLCJob3N0IiwiZGF0YWJhc2UiLCJkdW1wRmlsZW5hbWUiLCJEYXRlIiwibm93Iiwic3JjQ29ubiIsIm1pZ3JhdGlvbnNEdW1wIiwidXNlciIsInBhc3N3b3JkIiwiX2RiIiwibWlncmF0aW9ucyIsInJhdyIsImNvdW50IiwiY29ubiIsImRiIiwidW5kZWZpbmVkIiwicm93IiwibXlzcWxDbWQiLCJyZWNvcmRJZHMiLCJpbXBvcnRGaXh0dXJlIiwicHJhY3RpY2VEaXIiLCJhcGlSb290UGF0aCIsImZpbGVOYW1lcyIsIm1heFNlcU5vIiwiZmlsdGVyZWRTZXFzIiwiZmlsdGVyIiwiZmlsZU5hbWUiLCJzdGFydHNXaXRoIiwiZW5kc1dpdGgiLCJzZXFObyIsIm1hdGNoIiwicGFyc2VJbnQiLCJzb3J0IiwiYSIsImIiLCJsZW5ndGgiLCJjdXJyZW50U2VxTm8iLCJkc3RQYXRoIiwicnVuQ29kZSIsInJlcGxhY2UiLCJjcmVhdGVFbnRpdHkiLCJnZW5lcmF0ZVRlbXBsYXRlIiwicHJvamVjdFJlcXVpcmUiLCJ1aVBhY2thZ2VQYXRoIiwidWlOb2RlUGF0aCIsInVpUHJvY2VzcyIsIlBST0pFQ1RfTkFNRSIsInByb2plY3ROYW1lIiwiYmFzZW5hbWUiLCJVSV9QT1JUIiwicG9ydCIsInRvU3RyaW5nIiwiZSIsIkVycm9yIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPQSxXQUFXLFFBQVE7QUFDMUIsT0FBT0MsWUFBWSxTQUFTO0FBRTVCQSxPQUFPQyxNQUFNO0FBRWIsU0FBU0MsUUFBUSxFQUFFQyxLQUFLLFFBQVEsZ0JBQWdCO0FBQ2hELFNBQVNDLEtBQUssRUFBRUMsT0FBTyxFQUFFQyxFQUFFLEVB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
|