sonamu 0.6.0 → 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 +2 -1
- package/dist/api/caster.d.ts.map +1 -1
- package/dist/api/caster.js +6 -1
- 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 +178 -409
- package/dist/api/config.d.ts +27 -13
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +19 -26
- package/dist/api/context.d.ts +4 -3
- package/dist/api/context.d.ts.map +1 -1
- package/dist/api/context.js +1 -1
- package/dist/api/decorators.d.ts +20 -6
- package/dist/api/decorators.d.ts.map +1 -1
- package/dist/api/decorators.js +111 -18
- package/dist/api/index.d.ts +2 -2
- package/dist/api/index.d.ts.map +1 -1
- package/dist/api/index.js +3 -3
- package/dist/api/sonamu.d.ts +7 -7
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +83 -51
- 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 +5 -1
- package/dist/bin/build-config.d.ts.map +1 -1
- package/dist/bin/build-config.js +5 -2
- package/dist/bin/cli.js +165 -64
- 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 +30 -13
- 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 +232 -89
- 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 +11 -10
- package/dist/database/db.d.ts +5 -6
- package/dist/database/db.d.ts.map +1 -1
- package/dist/database/db.js +22 -25
- 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 +2 -2
- package/dist/database/puri.d.ts +25 -14
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +83 -21
- package/dist/database/puri.types.d.ts +21 -7
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +4 -1
- 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 +1 -1
- 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 +228 -78
- package/dist/entity/entity-manager.d.ts +165 -2
- package/dist/entity/entity-manager.d.ts.map +1 -1
- package/dist/entity/entity-manager.js +26 -10
- package/dist/entity/entity.d.ts +5 -3
- package/dist/entity/entity.d.ts.map +1 -1
- package/dist/entity/entity.js +153 -54
- 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 +1 -1
- 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 +1 -1
- 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 +1 -1
- package/dist/file-storage/file-storage.js +2 -2
- package/dist/index.d.ts +18 -11
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +19 -13
- 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 +123 -67
- 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 +67 -218
- package/dist/migration/migrator.d.ts +24 -73
- package/dist/migration/migrator.d.ts.map +1 -1
- package/dist/migration/migrator.js +121 -301
- 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 +1 -1
- 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 +91 -8
- package/dist/naite/naite.d.ts.map +1 -1
- package/dist/naite/naite.js +285 -41
- package/dist/stream/sse.d.ts +2 -2
- package/dist/stream/sse.d.ts.map +1 -1
- package/dist/stream/sse.js +1 -1
- package/dist/syncer/api-parser.d.ts +3 -13
- package/dist/syncer/api-parser.d.ts.map +1 -1
- package/dist/syncer/api-parser.js +67 -56
- package/dist/syncer/checksum.d.ts +2 -2
- package/dist/syncer/checksum.d.ts.map +1 -1
- package/dist/syncer/checksum.js +11 -11
- package/dist/syncer/code-generator.d.ts +3 -3
- package/dist/syncer/code-generator.d.ts.map +1 -1
- package/dist/syncer/code-generator.js +37 -17
- package/dist/syncer/entity-operations.d.ts +2 -2
- package/dist/syncer/entity-operations.d.ts.map +1 -1
- package/dist/syncer/entity-operations.js +9 -8
- package/dist/syncer/file-patterns.d.ts +1 -1
- package/dist/syncer/file-patterns.d.ts.map +1 -1
- package/dist/syncer/file-patterns.js +1 -1
- package/dist/syncer/index.d.ts +4 -4
- package/dist/syncer/index.d.ts.map +1 -1
- package/dist/syncer/index.js +5 -5
- package/dist/syncer/module-loader.d.ts +4 -4
- package/dist/syncer/module-loader.d.ts.map +1 -1
- package/dist/syncer/module-loader.js +17 -12
- package/dist/syncer/syncer.d.ts +31 -24
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +92 -45
- package/dist/template/entity-converter.d.ts +1 -1
- package/dist/template/entity-converter.d.ts.map +1 -1
- package/dist/template/entity-converter.js +15 -8
- package/dist/template/helpers.d.ts +2 -2
- package/dist/template/helpers.d.ts.map +1 -1
- package/dist/template/helpers.js +3 -3
- package/dist/template/implementations/entity.template.d.ts +2 -2
- package/dist/template/implementations/entity.template.d.ts.map +1 -1
- package/dist/template/implementations/entity.template.js +4 -5
- package/dist/template/implementations/generated.template.d.ts +2 -3
- package/dist/template/implementations/generated.template.d.ts.map +1 -1
- package/dist/template/implementations/generated.template.js +46 -29
- package/dist/template/implementations/generated_http.template.d.ts +2 -3
- package/dist/template/implementations/generated_http.template.d.ts.map +1 -1
- package/dist/template/implementations/generated_http.template.js +9 -9
- package/dist/template/implementations/generated_sso.template.d.ts +3 -4
- package/dist/template/implementations/generated_sso.template.d.ts.map +1 -1
- package/dist/template/implementations/generated_sso.template.js +54 -25
- package/dist/template/implementations/init_types.template.d.ts +2 -2
- package/dist/template/implementations/init_types.template.d.ts.map +1 -1
- package/dist/template/implementations/init_types.template.js +2 -2
- package/dist/template/implementations/model.template.d.ts +2 -2
- package/dist/template/implementations/model.template.d.ts.map +1 -1
- package/dist/template/implementations/model.template.js +47 -37
- package/dist/template/implementations/model_test.template.d.ts +2 -2
- package/dist/template/implementations/model_test.template.d.ts.map +1 -1
- package/dist/template/implementations/model_test.template.js +2 -2
- package/dist/template/implementations/service.template.d.ts +4 -4
- package/dist/template/implementations/service.template.d.ts.map +1 -1
- package/dist/template/implementations/service.template.js +24 -16
- package/dist/template/implementations/view_enums_buttonset.template.d.ts +2 -2
- package/dist/template/implementations/view_enums_buttonset.template.d.ts.map +1 -1
- package/dist/template/implementations/view_enums_buttonset.template.js +1 -1
- package/dist/template/implementations/view_enums_dropdown.template.d.ts +2 -2
- package/dist/template/implementations/view_enums_dropdown.template.d.ts.map +1 -1
- package/dist/template/implementations/view_enums_dropdown.template.js +2 -2
- package/dist/template/implementations/view_enums_select.template.d.ts +2 -2
- package/dist/template/implementations/view_enums_select.template.d.ts.map +1 -1
- package/dist/template/implementations/view_enums_select.template.js +2 -2
- package/dist/template/implementations/view_form.template.d.ts +2 -2
- package/dist/template/implementations/view_form.template.d.ts.map +1 -1
- package/dist/template/implementations/view_form.template.js +4 -4
- package/dist/template/implementations/view_id_all_select.template.d.ts +2 -2
- package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -1
- package/dist/template/implementations/view_id_all_select.template.js +1 -1
- package/dist/template/implementations/view_id_async_select.template.d.ts +2 -2
- package/dist/template/implementations/view_id_async_select.template.d.ts.map +1 -1
- package/dist/template/implementations/view_id_async_select.template.js +1 -1
- package/dist/template/implementations/view_list.template.d.ts +2 -2
- package/dist/template/implementations/view_list.template.d.ts.map +1 -1
- package/dist/template/implementations/view_list.template.js +29 -19
- package/dist/template/implementations/view_list_columns.template.d.ts +3 -3
- package/dist/template/implementations/view_list_columns.template.d.ts.map +1 -1
- package/dist/template/implementations/view_list_columns.template.js +1 -1
- package/dist/template/implementations/view_search_input.template.d.ts +2 -2
- package/dist/template/implementations/view_search_input.template.d.ts.map +1 -1
- package/dist/template/implementations/view_search_input.template.js +1 -1
- package/dist/template/index.d.ts +4 -2
- package/dist/template/index.d.ts.map +1 -1
- package/dist/template/index.js +5 -3
- 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 +12 -2
- package/dist/template/template.d.ts.map +1 -1
- package/dist/template/template.js +19 -6
- package/dist/template/zod-converter.d.ts +40 -7
- package/dist/template/zod-converter.d.ts.map +1 -1
- package/dist/template/zod-converter.js +341 -58
- 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 +12 -3
- 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 +338 -236
- package/dist/types/types.d.ts +709 -104
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +309 -52
- package/dist/typings/knex.d.js +2 -2
- package/dist/utils/async-utils.d.ts.map +1 -1
- package/dist/utils/async-utils.js +3 -3
- package/dist/utils/console-util.js +1 -1
- package/dist/utils/controller.d.ts +1 -0
- package/dist/utils/controller.d.ts.map +1 -1
- package/dist/utils/controller.js +4 -1
- package/dist/utils/esm-utils.d.ts +0 -6
- package/dist/utils/esm-utils.d.ts.map +1 -1
- package/dist/utils/esm-utils.js +2 -9
- 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 +1 -1
- package/dist/utils/lodash-able.d.ts.map +1 -1
- package/dist/utils/lodash-able.js +1 -1
- 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 +1 -1
- package/dist/utils/path-utils.d.ts.map +1 -1
- package/dist/utils/path-utils.js +3 -3
- package/dist/utils/process-utils.js +1 -1
- 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 +14 -3
- 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 +7 -1
- package/dist/utils/utils.d.ts.map +1 -1
- package/dist/utils/utils.js +44 -5
- 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 +1 -1
- package/package.json +54 -29
- 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 +176 -533
- package/src/api/config.ts +39 -56
- package/src/api/context.ts +7 -18
- package/src/api/decorators.ts +175 -46
- package/src/api/index.ts +2 -2
- package/src/api/sonamu.ts +133 -124
- package/src/api/validator.ts +83 -0
- package/src/bin/build-config.ts +7 -1
- package/src/bin/cli.ts +192 -110
- 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 +36 -50
- 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 +182 -126
- package/src/database/puri.types.ts +64 -31
- package/src/database/transaction-context.ts +1 -1
- package/src/database/upsert-builder.ts +262 -132
- package/src/entity/entity-manager.ts +36 -28
- package/src/entity/entity.ts +330 -249
- 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 -12
- package/src/migration/code-generation.ts +185 -172
- package/src/migration/migration-set.ts +80 -293
- package/src/migration/migrator.ts +182 -425
- 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 +378 -33
- package/src/shared/web.shared.ts.txt +20 -24
- package/src/stream/sse.ts +5 -5
- package/src/syncer/api-parser.ts +52 -69
- package/src/syncer/checksum.ts +25 -37
- package/src/syncer/code-generator.ts +58 -62
- package/src/syncer/entity-operations.ts +12 -15
- package/src/syncer/file-patterns.ts +2 -2
- package/src/syncer/index.ts +4 -4
- package/src/syncer/module-loader.ts +28 -25
- package/src/syncer/syncer.ts +155 -162
- package/src/template/entity-converter.ts +18 -27
- package/src/template/helpers.ts +8 -11
- package/src/template/implementations/entity.template.ts +6 -6
- package/src/template/implementations/generated.template.ts +99 -99
- package/src/template/implementations/generated_http.template.ts +21 -54
- package/src/template/implementations/generated_sso.template.ts +78 -65
- package/src/template/implementations/init_types.template.ts +4 -6
- package/src/template/implementations/model.template.ts +47 -38
- package/src/template/implementations/model_test.template.ts +3 -3
- package/src/template/implementations/service.template.ts +56 -80
- package/src/template/implementations/view_enums_buttonset.template.ts +2 -2
- package/src/template/implementations/view_enums_dropdown.template.ts +4 -4
- package/src/template/implementations/view_enums_select.template.ts +3 -3
- package/src/template/implementations/view_form.template.ts +34 -75
- package/src/template/implementations/view_id_all_select.template.ts +2 -2
- package/src/template/implementations/view_id_async_select.template.ts +9 -23
- package/src/template/implementations/view_list.template.ts +54 -95
- package/src/template/implementations/view_list_columns.template.ts +4 -10
- package/src/template/implementations/view_search_input.template.ts +2 -2
- package/src/template/index.ts +4 -2
- package/src/template/template-manager.ts +166 -0
- package/src/template/template-types.ts +16 -0
- package/src/template/template.ts +29 -10
- package/src/template/zod-converter.ts +407 -101
- package/src/testing/_relation-graph.ts +18 -11
- package/src/testing/fixture-manager.ts +468 -362
- package/src/types/types.ts +516 -248
- package/src/typings/knex.d.ts +7 -9
- package/src/utils/async-utils.ts +8 -12
- package/src/utils/console-util.ts +1 -1
- package/src/utils/controller.ts +3 -0
- package/src/utils/esm-utils.ts +8 -18
- 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 +3 -6
- package/src/utils/process-utils.ts +1 -1
- package/src/utils/sql-parser.ts +23 -5
- package/src/utils/type-utils.ts +83 -0
- package/src/utils/utils.ts +58 -9
- package/src/utils/zod-error.ts +3 -3
- 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 -72
- 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 -39
- 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 -210
- package/src/bin/cli-wrapper.ts +0 -82
- package/src/database/knex-plugins/knex-on-duplicate-update.ts +0 -45
- package/src/entity/entity-utils.ts +0 -291
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import type { Knex } from "knex";
|
|
2
|
+
import type { MigrationColumn, MigrationSet } from "../types/types";
|
|
3
|
+
export type PgColumn = {
|
|
4
|
+
column_name: string;
|
|
5
|
+
data_type: string;
|
|
6
|
+
udt_name: string;
|
|
7
|
+
character_maximum_length: number | null;
|
|
8
|
+
numeric_precision: number | null;
|
|
9
|
+
numeric_scale: number | null;
|
|
10
|
+
is_nullable: string;
|
|
11
|
+
column_default: string | null;
|
|
12
|
+
};
|
|
13
|
+
type PgIndex = {
|
|
14
|
+
index_name: string;
|
|
15
|
+
column_name: string;
|
|
16
|
+
is_unique: boolean;
|
|
17
|
+
is_primary: boolean;
|
|
18
|
+
index_type: string;
|
|
19
|
+
};
|
|
20
|
+
type PgForeign = {
|
|
21
|
+
constraint_name: string;
|
|
22
|
+
column_name: string;
|
|
23
|
+
foreign_table_name: string;
|
|
24
|
+
foreign_column_name: string;
|
|
25
|
+
update_rule: string;
|
|
26
|
+
delete_rule: string;
|
|
27
|
+
};
|
|
28
|
+
declare class PostgreSQLSchemaReaderClass {
|
|
29
|
+
/**
|
|
30
|
+
* DB에서 테이블 정보를 읽어서 MigrationSet을 만들어옵니다.
|
|
31
|
+
* @param compareDB Knex 인스턴스
|
|
32
|
+
* @param table 테이블 이름
|
|
33
|
+
* @returns MigrationSet 객체
|
|
34
|
+
*/
|
|
35
|
+
getMigrationSetFromDB(compareDB: Knex, table: string): Promise<MigrationSet | null>;
|
|
36
|
+
/**
|
|
37
|
+
* PostgreSQL의 constraint action을 Knex 형식으로 변환
|
|
38
|
+
*/
|
|
39
|
+
private mapConstraintAction;
|
|
40
|
+
/**
|
|
41
|
+
* 기존 테이블 읽어서 cols, indexes, foreigns 반환
|
|
42
|
+
*/
|
|
43
|
+
readTable(compareDB: Knex, tableName: string): Promise<[PgColumn[], PgIndex[], PgForeign[]]>;
|
|
44
|
+
/**
|
|
45
|
+
* PostgreSQL 컬럼 타입을 분석하여 MigrationColumn 객체로 변환합니다.
|
|
46
|
+
*/
|
|
47
|
+
resolveDBColType(dbColumn: PgColumn): Pick<MigrationColumn, "type" | "length" | "precision" | "scale" | "numberType">;
|
|
48
|
+
}
|
|
49
|
+
export declare const PostgreSQLSchemaReader: PostgreSQLSchemaReaderClass;
|
|
50
|
+
export {};
|
|
51
|
+
//# sourceMappingURL=postgresql-schema-reader.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"postgresql-schema-reader.d.ts","sourceRoot":"","sources":["../../src/migration/postgresql-schema-reader.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAEjC,OAAO,KAAK,EACV,eAAe,EAGf,YAAY,EAEb,MAAM,gBAAgB,CAAC;AAExB,MAAM,MAAM,QAAQ,GAAG;IACrB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,MAAM,CAAC;IACjB,wBAAwB,EAAE,MAAM,GAAG,IAAI,CAAC;IACxC,iBAAiB,EAAE,MAAM,GAAG,IAAI,CAAC;IACjC,aAAa,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7B,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,EAAE,MAAM,GAAG,IAAI,CAAC;CAC/B,CAAC;AAEF,KAAK,OAAO,GAAG;IACb,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,OAAO,CAAC;IACnB,UAAU,EAAE,OAAO,CAAC;IACpB,UAAU,EAAE,MAAM,CAAC;CACpB,CAAC;AAEF,KAAK,SAAS,GAAG;IACf,eAAe,EAAE,MAAM,CAAC;IACxB,WAAW,EAAE,MAAM,CAAC;IACpB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,mBAAmB,EAAE,MAAM,CAAC;IAC5B,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC;CACrB,CAAC;AAEF,cAAM,2BAA2B;IAC/B;;;;;OAKG;IACG,qBAAqB,CAAC,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC;IAuFzF;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAW3B;;OAEG;IACG,SAAS,CACb,SAAS,EAAE,IAAI,EACf,SAAS,EAAE,MAAM,GAChB,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE,OAAO,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;IAkEhD;;OAEG;IACH,gBAAgB,CACd,QAAQ,EAAE,QAAQ,GACjB,IAAI,CAAC,eAAe,EAAE,MAAM,GAAG,QAAQ,GAAG,WAAW,GAAG,OAAO,GAAG,YAAY,CAAC;CAmFnF;AAED,eAAO,MAAM,sBAAsB,6BAAoC,CAAC"}
|
|
@@ -0,0 +1,245 @@
|
|
|
1
|
+
import assert from "assert";
|
|
2
|
+
import { group } from "radashi";
|
|
3
|
+
class PostgreSQLSchemaReaderClass {
|
|
4
|
+
/**
|
|
5
|
+
* DB에서 테이블 정보를 읽어서 MigrationSet을 만들어옵니다.
|
|
6
|
+
* @param compareDB Knex 인스턴스
|
|
7
|
+
* @param table 테이블 이름
|
|
8
|
+
* @returns MigrationSet 객체
|
|
9
|
+
*/ async getMigrationSetFromDB(compareDB, table) {
|
|
10
|
+
let dbColumns, dbIndexes, dbForeigns;
|
|
11
|
+
try {
|
|
12
|
+
[dbColumns, dbIndexes, dbForeigns] = await this.readTable(compareDB, table);
|
|
13
|
+
} catch (e) {
|
|
14
|
+
if (e instanceof Error && e.message.includes("Table not found")) {
|
|
15
|
+
return null;
|
|
16
|
+
}
|
|
17
|
+
console.error(e);
|
|
18
|
+
return null;
|
|
19
|
+
}
|
|
20
|
+
const columns = dbColumns.map((dbColumn)=>{
|
|
21
|
+
const dbColType = this.resolveDBColType(dbColumn);
|
|
22
|
+
return {
|
|
23
|
+
name: dbColumn.column_name,
|
|
24
|
+
nullable: dbColumn.is_nullable === "YES",
|
|
25
|
+
...dbColType,
|
|
26
|
+
...(()=>{
|
|
27
|
+
if (dbColumn.column_default !== null) {
|
|
28
|
+
// PostgreSQL default 값 정리 (nextval, CURRENT_TIMESTAMP 등)
|
|
29
|
+
let defaultValue = dbColumn.column_default;
|
|
30
|
+
// nextval 제거 (SERIAL 타입)
|
|
31
|
+
if (defaultValue.startsWith("nextval(")) {
|
|
32
|
+
return {};
|
|
33
|
+
}
|
|
34
|
+
// 타입 캐스팅 제거 (예: '1'::integer → 1)
|
|
35
|
+
defaultValue = defaultValue.replace(/::[\w\s]+$/g, "");
|
|
36
|
+
// 따옴표 제거가 필요한 경우
|
|
37
|
+
if (defaultValue.startsWith("'") && defaultValue.endsWith("'")) {
|
|
38
|
+
defaultValue = defaultValue.slice(1, -1);
|
|
39
|
+
}
|
|
40
|
+
return {
|
|
41
|
+
defaultTo: defaultValue
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
return {};
|
|
45
|
+
})()
|
|
46
|
+
};
|
|
47
|
+
});
|
|
48
|
+
// PRIMARY KEY와 foreign key용 인덱스 제외
|
|
49
|
+
const dbIndexesGroup = group(dbIndexes.filter((dbIndex)=>!dbIndex.is_primary && !dbForeigns.find((dbForeign)=>dbIndex.index_name.includes(dbForeign.constraint_name))), (dbIndex)=>dbIndex.index_name);
|
|
50
|
+
// indexes 처리
|
|
51
|
+
const indexes = Object.keys(dbIndexesGroup).map((indexName)=>{
|
|
52
|
+
const currentIndexes = dbIndexesGroup[indexName];
|
|
53
|
+
assert(currentIndexes);
|
|
54
|
+
const firstIndex = currentIndexes[0];
|
|
55
|
+
const type = firstIndex.is_unique ? "unique" : "index";
|
|
56
|
+
return {
|
|
57
|
+
type,
|
|
58
|
+
columns: currentIndexes.map((idx)=>idx.column_name)
|
|
59
|
+
};
|
|
60
|
+
});
|
|
61
|
+
// foreigns 처리
|
|
62
|
+
const foreigns = dbForeigns.map((dbForeign)=>{
|
|
63
|
+
return {
|
|
64
|
+
columns: [
|
|
65
|
+
dbForeign.column_name
|
|
66
|
+
],
|
|
67
|
+
to: `${dbForeign.foreign_table_name}.${dbForeign.foreign_column_name}`,
|
|
68
|
+
onUpdate: this.mapConstraintAction(dbForeign.update_rule),
|
|
69
|
+
onDelete: this.mapConstraintAction(dbForeign.delete_rule)
|
|
70
|
+
};
|
|
71
|
+
});
|
|
72
|
+
return {
|
|
73
|
+
table,
|
|
74
|
+
columns,
|
|
75
|
+
indexes,
|
|
76
|
+
foreigns
|
|
77
|
+
};
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* PostgreSQL의 constraint action을 Knex 형식으로 변환
|
|
81
|
+
*/ mapConstraintAction(action) {
|
|
82
|
+
const actionMap = {
|
|
83
|
+
"NO ACTION": "NO ACTION",
|
|
84
|
+
RESTRICT: "RESTRICT",
|
|
85
|
+
CASCADE: "CASCADE",
|
|
86
|
+
"SET NULL": "SET NULL",
|
|
87
|
+
"SET DEFAULT": "SET DEFAULT"
|
|
88
|
+
};
|
|
89
|
+
return actionMap[action] ?? "NO ACTION";
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* 기존 테이블 읽어서 cols, indexes, foreigns 반환
|
|
93
|
+
*/ async readTable(compareDB, tableName) {
|
|
94
|
+
// Columns 조회
|
|
95
|
+
const columns = await compareDB.select("column_name", "data_type", "udt_name", "character_maximum_length", "numeric_precision", "numeric_scale", "is_nullable", "column_default").from("information_schema.columns").where({
|
|
96
|
+
table_name: tableName
|
|
97
|
+
}).orderBy("ordinal_position");
|
|
98
|
+
if (columns.length === 0) {
|
|
99
|
+
throw new Error(`Table not found: ${tableName}`);
|
|
100
|
+
}
|
|
101
|
+
// Indexes 조회
|
|
102
|
+
const indexesQuery = `
|
|
103
|
+
SELECT
|
|
104
|
+
i.relname as index_name,
|
|
105
|
+
a.attname as column_name,
|
|
106
|
+
ix.indisunique as is_unique,
|
|
107
|
+
ix.indisprimary as is_primary,
|
|
108
|
+
am.amname as index_type
|
|
109
|
+
FROM pg_class t
|
|
110
|
+
JOIN pg_index ix ON t.oid = ix.indrelid
|
|
111
|
+
JOIN pg_class i ON i.oid = ix.indexrelid
|
|
112
|
+
JOIN pg_attribute a ON a.attrelid = t.oid
|
|
113
|
+
JOIN pg_am am ON i.relam = am.oid
|
|
114
|
+
WHERE t.relname = ?
|
|
115
|
+
AND a.attnum = ANY(ix.indkey)
|
|
116
|
+
ORDER BY i.relname, array_position(ix.indkey, a.attnum)
|
|
117
|
+
`;
|
|
118
|
+
const indexes = (await compareDB.raw(indexesQuery, [
|
|
119
|
+
tableName
|
|
120
|
+
])).rows;
|
|
121
|
+
// Foreign Keys 조회
|
|
122
|
+
const foreignsQuery = `
|
|
123
|
+
SELECT
|
|
124
|
+
tc.constraint_name,
|
|
125
|
+
kcu.column_name,
|
|
126
|
+
ccu.table_name AS foreign_table_name,
|
|
127
|
+
ccu.column_name AS foreign_column_name,
|
|
128
|
+
rc.update_rule,
|
|
129
|
+
rc.delete_rule
|
|
130
|
+
FROM information_schema.table_constraints AS tc
|
|
131
|
+
JOIN information_schema.key_column_usage AS kcu
|
|
132
|
+
ON tc.constraint_name = kcu.constraint_name
|
|
133
|
+
AND tc.table_schema = kcu.table_schema
|
|
134
|
+
JOIN information_schema.constraint_column_usage AS ccu
|
|
135
|
+
ON ccu.constraint_name = tc.constraint_name
|
|
136
|
+
AND ccu.table_schema = tc.table_schema
|
|
137
|
+
JOIN information_schema.referential_constraints AS rc
|
|
138
|
+
ON rc.constraint_name = tc.constraint_name
|
|
139
|
+
AND rc.constraint_schema = tc.table_schema
|
|
140
|
+
WHERE tc.constraint_type = 'FOREIGN KEY'
|
|
141
|
+
AND tc.table_name = ?
|
|
142
|
+
`;
|
|
143
|
+
const foreigns = (await compareDB.raw(foreignsQuery, [
|
|
144
|
+
tableName
|
|
145
|
+
])).rows;
|
|
146
|
+
return [
|
|
147
|
+
columns,
|
|
148
|
+
indexes,
|
|
149
|
+
foreigns
|
|
150
|
+
];
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* PostgreSQL 컬럼 타입을 분석하여 MigrationColumn 객체로 변환합니다.
|
|
154
|
+
*/ resolveDBColType(dbColumn) {
|
|
155
|
+
const { udt_name: _udt_name, character_maximum_length, numeric_precision, numeric_scale } = dbColumn;
|
|
156
|
+
const { udt_name, singleOrArray } = (()=>{
|
|
157
|
+
if (_udt_name.startsWith("_")) {
|
|
158
|
+
return {
|
|
159
|
+
udt_name: _udt_name.substring(1),
|
|
160
|
+
singleOrArray: "[]"
|
|
161
|
+
};
|
|
162
|
+
}
|
|
163
|
+
return {
|
|
164
|
+
udt_name: _udt_name,
|
|
165
|
+
singleOrArray: ""
|
|
166
|
+
};
|
|
167
|
+
})();
|
|
168
|
+
// UUID
|
|
169
|
+
if (udt_name === "uuid") {
|
|
170
|
+
return {
|
|
171
|
+
type: `uuid${singleOrArray}`
|
|
172
|
+
};
|
|
173
|
+
}
|
|
174
|
+
// Integer types
|
|
175
|
+
if (udt_name === "int4") {
|
|
176
|
+
return {
|
|
177
|
+
type: `integer${singleOrArray}`
|
|
178
|
+
};
|
|
179
|
+
}
|
|
180
|
+
if (udt_name === "int8") {
|
|
181
|
+
return {
|
|
182
|
+
type: `bigInteger${singleOrArray}`
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
// String types
|
|
186
|
+
if (udt_name === "varchar") {
|
|
187
|
+
return {
|
|
188
|
+
type: `string${singleOrArray}`,
|
|
189
|
+
...character_maximum_length && {
|
|
190
|
+
length: character_maximum_length
|
|
191
|
+
}
|
|
192
|
+
};
|
|
193
|
+
}
|
|
194
|
+
if (udt_name === "text") {
|
|
195
|
+
return {
|
|
196
|
+
type: `string${singleOrArray}`
|
|
197
|
+
}; // StringProp without length
|
|
198
|
+
}
|
|
199
|
+
// NumberOrNumeric types
|
|
200
|
+
if (udt_name === "numeric") {
|
|
201
|
+
return {
|
|
202
|
+
type: `numberOrNumeric${singleOrArray}`,
|
|
203
|
+
numberType: "numeric",
|
|
204
|
+
...numeric_precision !== null && numeric_scale !== null && {
|
|
205
|
+
precision: numeric_precision,
|
|
206
|
+
scale: numeric_scale
|
|
207
|
+
}
|
|
208
|
+
};
|
|
209
|
+
}
|
|
210
|
+
if (udt_name === "float4") {
|
|
211
|
+
return {
|
|
212
|
+
type: `numberOrNumeric${singleOrArray}`,
|
|
213
|
+
numberType: "real"
|
|
214
|
+
};
|
|
215
|
+
}
|
|
216
|
+
if (udt_name === "float8") {
|
|
217
|
+
return {
|
|
218
|
+
type: `numberOrNumeric${singleOrArray}`,
|
|
219
|
+
numberType: "double precision"
|
|
220
|
+
};
|
|
221
|
+
}
|
|
222
|
+
// Boolean
|
|
223
|
+
if (udt_name === "bool") {
|
|
224
|
+
return {
|
|
225
|
+
type: `boolean${singleOrArray}`
|
|
226
|
+
};
|
|
227
|
+
}
|
|
228
|
+
// Timestampz types
|
|
229
|
+
if (udt_name === "timestamptz") {
|
|
230
|
+
return {
|
|
231
|
+
type: `date${singleOrArray}`
|
|
232
|
+
}; // DateProp → timestamptz
|
|
233
|
+
}
|
|
234
|
+
// JSON
|
|
235
|
+
if (udt_name === "json" || udt_name === "jsonb") {
|
|
236
|
+
return {
|
|
237
|
+
type: "json"
|
|
238
|
+
};
|
|
239
|
+
}
|
|
240
|
+
throw new Error(`resolve 불가능한 PostgreSQL 컬럼 타입: ${udt_name}`);
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
export const PostgreSQLSchemaReader = new PostgreSQLSchemaReaderClass();
|
|
244
|
+
|
|
245
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9taWdyYXRpb24vcG9zdGdyZXNxbC1zY2hlbWEtcmVhZGVyLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBhc3NlcnQgZnJvbSBcImFzc2VydFwiO1xuaW1wb3J0IHR5cGUgeyBLbmV4IH0gZnJvbSBcImtuZXhcIjtcbmltcG9ydCB7IGdyb3VwIH0gZnJvbSBcInJhZGFzaGlcIjtcbmltcG9ydCB0eXBlIHtcbiAgTWlncmF0aW9uQ29sdW1uLFxuICBNaWdyYXRpb25Gb3JlaWduLFxuICBNaWdyYXRpb25JbmRleCxcbiAgTWlncmF0aW9uU2V0LFxuICBSZWxhdGlvbk9uLFxufSBmcm9tIFwiLi4vdHlwZXMvdHlwZXNcIjtcblxuZXhwb3J0IHR5cGUgUGdDb2x1bW4gPSB7XG4gIGNvbHVtbl9uYW1lOiBzdHJpbmc7XG4gIGRhdGFfdHlwZTogc3RyaW5nO1xuICB1ZHRfbmFtZTogc3RyaW5nO1xuICBjaGFyYWN0ZXJfbWF4aW11bV9sZW5ndGg6IG51bWJlciB8IG51bGw7XG4gIG51bWVyaWNfcHJlY2lzaW9uOiBudW1iZXIgfCBudWxsO1xuICBudW1lcmljX3NjYWxlOiBudW1iZXIgfCBudWxsO1xuICBpc19udWxsYWJsZTogc3RyaW5nO1xuICBjb2x1bW5fZGVmYXVsdDogc3RyaW5nIHwgbnVsbDtcbn07XG5cbnR5cGUgUGdJbmRleCA9IHtcbiAgaW5kZXhfbmFtZTogc3RyaW5nO1xuICBjb2x1bW5fbmFtZTogc3RyaW5nO1xuICBpc191bmlxdWU6IGJvb2xlYW47XG4gIGlzX3ByaW1hcnk6IGJvb2xlYW47XG4gIGluZGV4X3R5cGU6IHN0cmluZztcbn07XG5cbnR5cGUgUGdGb3JlaWduID0ge1xuICBjb25zdHJhaW50X25hbWU6IHN0cmluZztcbiAgY29sdW1uX25hbWU6IHN0cmluZztcbiAgZm9yZWlnbl90YWJsZV9uYW1lOiBzdHJpbmc7XG4gIGZvcmVpZ25fY29sdW1uX25hbWU6IHN0cmluZztcbiAgdXBkYXRlX3J1bGU6IHN0cmluZztcbiAgZGVsZXRlX3J1bGU6IHN0cmluZztcbn07XG5cbmNsYXNzIFBvc3RncmVTUUxTY2hlbWFSZWFkZXJDbGFzcyB7XG4gIC8qKlxuICAgKiBEQuyXkOyEnCDthYzsnbTruJQg7KCV67O066W8IOydveyWtOyEnCBNaWdyYXRpb25TZXTsnYQg66eM65Ok7Ja07Ji164uI64ukLlxuICAgKiBAcGFyYW0gY29tcGFyZURCIEtuZXgg7J247Iqk7YS07IqkXG4gICAqIEBwYXJhbSB0YWJsZSDthYzsnbTruJQg7J2066aEXG4gICAqIEByZXR1cm5zIE1pZ3JhdGlvblNldCDqsJ3ssrRcbiAgICovXG4gIGFzeW5jIGdldE1pZ3JhdGlvblNldEZyb21EQihjb21wYXJlREI6IEtuZXgsIHRhYmxlOiBzdHJpbmcpOiBQcm9taXNlPE1pZ3JhdGlvblNldCB8IG51bGw+IHtcbiAgICBsZXQgZGJDb2x1bW5zOiBQZ0NvbHVtbltdLCBkYkluZGV4ZXM6IFBnSW5kZXhbXSwgZGJGb3JlaWduczogUGdGb3JlaWduW107XG4gICAgdHJ5IHtcbiAgICAgIFtkYkNvbHVtbnMsIGRiSW5kZXhlcywgZGJGb3JlaWduc10gPSBhd2FpdCB0aGlzLnJlYWRUYWJsZShjb21wYXJlREIsIHRhYmxlKTtcbiAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICBpZiAoZSBpbnN0YW5jZW9mIEVycm9yICYmIGUubWVzc2FnZS5pbmNsdWRlcyhcIlRhYmxlIG5vdCBmb3VuZFwiKSkge1xuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgIH1cbiAgICAgIGNvbnNvbGUuZXJyb3IoZSk7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICBjb25zdCBjb2x1bW5zOiBNaWdyYXRpb25Db2x1bW5bXSA9IGRiQ29sdW1ucy5tYXAoKGRiQ29sdW1uKSA9PiB7XG4gICAgICBjb25zdCBkYkNvbFR5cGUgPSB0aGlzLnJlc29sdmVEQkNvbFR5cGUoZGJDb2x1bW4pO1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgbmFtZTogZGJDb2x1bW4uY29sdW1uX25hbWUsXG4gICAgICAgIG51bGxhYmxlOiBkYkNvbHVtbi5pc19udWxsYWJsZSA9PT0gXCJZRVNcIixcbiAgICAgICAgLi4uZGJDb2xUeXBlLFxuICAgICAgICAuLi4oKCkgPT4ge1xuICAgICAgICAgIGlmIChkYkNvbHVtbi5jb2x1bW5fZGVmYXVsdCAhPT0gbnVsbCkge1xuICAgICAgICAgICAgLy8gUG9zdGdyZVNRTCBkZWZhdWx0IOqwkiDsoJXrpqwgKG5leHR2YWwsIENVUlJFTlRfVElNRVNUQU1QIOuTsSlcbiAgICAgICAgICAgIGxldCBkZWZhdWx0VmFsdWUgPSBkYkNvbHVtbi5jb2x1bW5fZGVmYXVsdDtcblxuICAgICAgICAgICAgLy8gbmV4dHZhbCDsoJzqsbAgKFNFUklBTCDtg4DsnoUpXG4gICAgICAgICAgICBpZiAoZGVmYXVsdFZhbHVlLnN0YXJ0c1dpdGgoXCJuZXh0dmFsKFwiKSkge1xuICAgICAgICAgICAgICByZXR1cm4ge307XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIO2DgOyehSDsupDsiqTtjIUg7KCc6rGwICjsmIg6ICcxJzo6aW50ZWdlciDihpIgMSlcbiAgICAgICAgICAgIGRlZmF1bHRWYWx1ZSA9IGRlZmF1bHRWYWx1ZS5yZXBsYWNlKC86OltcXHdcXHNdKyQvZywgXCJcIik7XG5cbiAgICAgICAgICAgIC8vIOuUsOyYtO2RnCDsoJzqsbDqsIAg7ZWE7JqU7ZWcIOqyveyasFxuICAgICAgICAgICAgaWYgKGRlZmF1bHRWYWx1ZS5zdGFydHNXaXRoKFwiJ1wiKSAmJiBkZWZhdWx0VmFsdWUuZW5kc1dpdGgoXCInXCIpKSB7XG4gICAgICAgICAgICAgIGRlZmF1bHRWYWx1ZSA9IGRlZmF1bHRWYWx1ZS5zbGljZSgxLCAtMSk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIGRlZmF1bHRUbzogZGVmYXVsdFZhbHVlLFxuICAgICAgICAgICAgfTtcbiAgICAgICAgICB9XG4gICAgICAgICAgcmV0dXJuIHt9O1xuICAgICAgICB9KSgpLFxuICAgICAgfTtcbiAgICB9KTtcblxuICAgIC8vIFBSSU1BUlkgS0VZ7JmAIGZvcmVpZ24ga2V57JqpIOyduOuNseyKpCDsoJzsmbhcbiAgICBjb25zdCBkYkluZGV4ZXNHcm91cCA9IGdyb3VwKFxuICAgICAgZGJJbmRleGVzLmZpbHRlcihcbiAgICAgICAgKGRiSW5kZXgpID0+XG4gICAgICAgICAgIWRiSW5kZXguaXNfcHJpbWFyeSAmJlxuICAgICAgICAgICFkYkZvcmVpZ25zLmZpbmQoKGRiRm9yZWlnbikgPT4gZGJJbmRleC5pbmRleF9uYW1lLmluY2x1ZGVzKGRiRm9yZWlnbi5jb25zdHJhaW50X25hbWUpKSxcbiAgICAgICksXG4gICAgICAoZGJJbmRleCkgPT4gZGJJbmRleC5pbmRleF9uYW1lLFxuICAgICk7XG5cbiAgICAvLyBpbmRleGVzIOyymOumrFxuICAgIGNvbnN0IGluZGV4ZXM6IE1pZ3JhdGlvbkluZGV4W10gPSBPYmplY3Qua2V5cyhkYkluZGV4ZXNHcm91cCkubWFwKChpbmRleE5hbWUpID0+IHtcbiAgICAgIGNvbnN0IGN1cnJlbnRJbmRleGVzID0gZGJJbmRleGVzR3JvdXBbaW5kZXhOYW1lXTtcbiAgICAgIGFzc2VydChjdXJyZW50SW5kZXhlcyk7XG5cbiAgICAgIGNvbnN0IGZpcnN0SW5kZXggPSBjdXJyZW50SW5kZXhlc1swXTtcbiAgICAgIGNvbnN0IHR5cGUgPSBmaXJzdEluZGV4LmlzX3VuaXF1ZSA/IFwidW5pcXVlXCIgOiBcImluZGV4XCI7XG5cbiAgICAgIHJldHVybiB7XG4gICAgICAgIHR5cGUsXG4gICAgICAgIGNvbHVtbnM6IGN1cnJlbnRJbmRleGVzLm1hcCgoaWR4KSA9PiBpZHguY29sdW1uX25hbWUpLFxuICAgICAgfTtcbiAgICB9KTtcblxuICAgIC8vIGZvcmVpZ25zIOyymOumrFxuICAgIGNvbnN0IGZvcmVpZ25zOiBNaWdyYXRpb25Gb3JlaWduW10gPSBkYkZvcmVpZ25zLm1hcCgoZGJGb3JlaWduKSA9PiB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBjb2x1bW5zOiBbZGJGb3JlaWduLmNvbHVtbl9uYW1lXSxcbiAgICAgICAgdG86IGAke2RiRm9yZWlnbi5mb3JlaWduX3RhYmxlX25hbWV9LiR7ZGJGb3JlaWduLmZvcmVpZ25fY29sdW1uX25hbWV9YCxcbiAgICAgICAgb25VcGRhdGU6IHRoaXMubWFwQ29uc3RyYWludEFjdGlvbihkYkZvcmVpZ24udXBkYXRlX3J1bGUpLFxuICAgICAgICBvbkRlbGV0ZTogdGhpcy5tYXBDb25zdHJhaW50QWN0aW9uKGRiRm9yZWlnbi5kZWxldGVfcnVsZSksXG4gICAgICB9O1xuICAgIH0pO1xuXG4gICAgcmV0dXJuIHtcbiAgICAgIHRhYmxlLFxuICAgICAgY29sdW1ucyxcbiAgICAgIGluZGV4ZXMsXG4gICAgICBmb3JlaWducyxcbiAgICB9O1xuICB9XG5cbiAgLyoqXG4gICAqIFBvc3RncmVTUUzsnZggY29uc3RyYWludCBhY3Rpb27snYQgS25leCDtmJXsi53snLzroZwg67OA7ZmYXG4gICAqL1xuICBwcml2YXRlIG1hcENvbnN0cmFpbnRBY3Rpb24oYWN0aW9uOiBzdHJpbmcpOiBSZWxhdGlvbk9uIHtcbiAgICBjb25zdCBhY3Rpb25NYXA6IFJlY29yZDxzdHJpbmcsIFJlbGF0aW9uT24+ID0ge1xuICAgICAgXCJOTyBBQ1RJT05cIjogXCJOTyBBQ1RJT05cIixcbiAgICAgIFJFU1RSSUNUOiBcIlJFU1RSSUNUXCIsXG4gICAgICBDQVNDQURFOiBcIkNBU0NBREVcIixcbiAgICAgIFwiU0VUIE5VTExcIjogXCJTRVQgTlVMTFwiLFxuICAgICAgXCJTRVQgREVGQVVMVFwiOiBcIlNFVCBERUZBVUxUXCIsXG4gICAgfTtcbiAgICByZXR1cm4gYWN0aW9uTWFwW2FjdGlvbl0gPz8gXCJOTyBBQ1RJT05cIjtcbiAgfVxuXG4gIC8qKlxuICAgKiDquLDsobQg7YWM7J2067iUIOydveyWtOyEnCBjb2xzLCBpbmRleGVzLCBmb3JlaWducyDrsJjtmZhcbiAgICovXG4gIGFzeW5jIHJlYWRUYWJsZShcbiAgICBjb21wYXJlREI6IEtuZXgsXG4gICAgdGFibGVOYW1lOiBzdHJpbmcsXG4gICk6IFByb21pc2U8W1BnQ29sdW1uW10sIFBnSW5kZXhbXSwgUGdGb3JlaWduW11dPiB7XG4gICAgLy8gQ29sdW1ucyDsobDtmoxcbiAgICBjb25zdCBjb2x1bW5zID0gYXdhaXQgY29tcGFyZURCXG4gICAgICAuc2VsZWN0KFxuICAgICAgICBcImNvbHVtbl9uYW1lXCIsXG4gICAgICAgIFwiZGF0YV90eXBlXCIsXG4gICAgICAgIFwidWR0X25hbWVcIixcbiAgICAgICAgXCJjaGFyYWN0ZXJfbWF4aW11bV9sZW5ndGhcIixcbiAgICAgICAgXCJudW1lcmljX3ByZWNpc2lvblwiLFxuICAgICAgICBcIm51bWVyaWNfc2NhbGVcIixcbiAgICAgICAgXCJpc19udWxsYWJsZVwiLFxuICAgICAgICBcImNvbHVtbl9kZWZhdWx0XCIsXG4gICAgICApXG4gICAgICAuZnJvbShcImluZm9ybWF0aW9uX3NjaGVtYS5jb2x1bW5zXCIpXG4gICAgICAud2hlcmUoeyB0YWJsZV9uYW1lOiB0YWJsZU5hbWUgfSlcbiAgICAgIC5vcmRlckJ5KFwib3JkaW5hbF9wb3NpdGlvblwiKTtcbiAgICBpZiAoY29sdW1ucy5sZW5ndGggPT09IDApIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgVGFibGUgbm90IGZvdW5kOiAke3RhYmxlTmFtZX1gKTtcbiAgICB9XG5cbiAgICAvLyBJbmRleGVzIOyhsO2ajFxuICAgIGNvbnN0IGluZGV4ZXNRdWVyeSA9IGBcbiAgICAgIFNFTEVDVFxuICAgICAgICBpLnJlbG5hbWUgYXMgaW5kZXhfbmFtZSxcbiAgICAgICAgYS5hdHRuYW1lIGFzIGNvbHVtbl9uYW1lLFxuICAgICAgICBpeC5pbmRpc3VuaXF1ZSBhcyBpc191bmlxdWUsXG4gICAgICAgIGl4LmluZGlzcHJpbWFyeSBhcyBpc19wcmltYXJ5LFxuICAgICAgICBhbS5hbW5hbWUgYXMgaW5kZXhfdHlwZVxuICAgICAgRlJPTSBwZ19jbGFzcyB0XG4gICAgICBKT0lOIHBnX2luZGV4IGl4IE9OIHQub2lkID0gaXguaW5kcmVsaWRcbiAgICAgIEpPSU4gcGdfY2xhc3MgaSBPTiBpLm9pZCA9IGl4LmluZGV4cmVsaWRcbiAgICAgIEpPSU4gcGdfYXR0cmlidXRlIGEgT04gYS5hdHRyZWxpZCA9IHQub2lkXG4gICAgICBKT0lOIHBnX2FtIGFtIE9OIGkucmVsYW0gPSBhbS5vaWRcbiAgICAgIFdIRVJFIHQucmVsbmFtZSA9ID9cbiAgICAgICAgQU5EIGEuYXR0bnVtID0gQU5ZKGl4LmluZGtleSlcbiAgICAgIE9SREVSIEJZIGkucmVsbmFtZSwgYXJyYXlfcG9zaXRpb24oaXguaW5ka2V5LCBhLmF0dG51bSlcbiAgICBgO1xuICAgIGNvbnN0IGluZGV4ZXMgPSAoYXdhaXQgY29tcGFyZURCLnJhdyhpbmRleGVzUXVlcnksIFt0YWJsZU5hbWVdKSkucm93cztcblxuICAgIC8vIEZvcmVpZ24gS2V5cyDsobDtmoxcbiAgICBjb25zdCBmb3JlaWduc1F1ZXJ5ID0gYFxuICAgICAgU0VMRUNUXG4gICAgICAgIHRjLmNvbnN0cmFpbnRfbmFtZSxcbiAgICAgICAga2N1LmNvbHVtbl9uYW1lLFxuICAgICAgICBjY3UudGFibGVfbmFtZSBBUyBmb3JlaWduX3RhYmxlX25hbWUsXG4gICAgICAgIGNjdS5jb2x1bW5fbmFtZSBBUyBmb3JlaWduX2NvbHVtbl9uYW1lLFxuICAgICAgICByYy51cGRhdGVfcnVsZSxcbiAgICAgICAgcmMuZGVsZXRlX3J1bGVcbiAgICAgIEZST00gaW5mb3JtYXRpb25fc2NoZW1hLnRhYmxlX2NvbnN0cmFpbnRzIEFTIHRjXG4gICAgICBKT0lOIGluZm9ybWF0aW9uX3NjaGVtYS5rZXlfY29sdW1uX3VzYWdlIEFTIGtjdVxuICAgICAgICBPTiB0Yy5jb25zdHJhaW50X25hbWUgPSBrY3UuY29uc3RyYWludF9uYW1lXG4gICAgICAgIEFORCB0Yy50YWJsZV9zY2hlbWEgPSBrY3UudGFibGVfc2NoZW1hXG4gICAgICBKT0lOIGluZm9ybWF0aW9uX3NjaGVtYS5jb25zdHJhaW50X2NvbHVtbl91c2FnZSBBUyBjY3VcbiAgICAgICAgT04gY2N1LmNvbnN0cmFpbnRfbmFtZSA9IHRjLmNvbnN0cmFpbnRfbmFtZVxuICAgICAgICBBTkQgY2N1LnRhYmxlX3NjaGVtYSA9IHRjLnRhYmxlX3NjaGVtYVxuICAgICAgSk9JTiBpbmZvcm1hdGlvbl9zY2hlbWEucmVmZXJlbnRpYWxfY29uc3RyYWludHMgQVMgcmNcbiAgICAgICAgT04gcmMuY29uc3RyYWludF9uYW1lID0gdGMuY29uc3RyYWludF9uYW1lXG4gICAgICAgIEFORCByYy5jb25zdHJhaW50X3NjaGVtYSA9IHRjLnRhYmxlX3NjaGVtYVxuICAgICAgV0hFUkUgdGMuY29uc3RyYWludF90eXBlID0gJ0ZPUkVJR04gS0VZJ1xuICAgICAgICBBTkQgdGMudGFibGVfbmFtZSA9ID9cbiAgICBgO1xuICAgIGNvbnN0IGZvcmVpZ25zID0gKGF3YWl0IGNvbXBhcmVEQi5yYXcoZm9yZWlnbnNRdWVyeSwgW3RhYmxlTmFtZV0pKS5yb3dzO1xuXG4gICAgcmV0dXJuIFtjb2x1bW5zLCBpbmRleGVzLCBmb3JlaWduc107XG4gIH1cblxuICAvKipcbiAgICogUG9zdGdyZVNRTCDsu6zrn7wg7YOA7J6F7J2EIOu2hOyEne2VmOyXrCBNaWdyYXRpb25Db2x1bW4g6rCd7LK066GcIOuzgO2ZmO2VqeuLiOuLpC5cbiAgICovXG4gIHJlc29sdmVEQkNvbFR5cGUoXG4gICAgZGJDb2x1bW46IFBnQ29sdW1uLFxuICApOiBQaWNrPE1pZ3JhdGlvbkNvbHVtbiwgXCJ0eXBlXCIgfCBcImxlbmd0aFwiIHwgXCJwcmVjaXNpb25cIiB8IFwic2NhbGVcIiB8IFwibnVtYmVyVHlwZVwiPiB7XG4gICAgY29uc3Qge1xuICAgICAgdWR0X25hbWU6IF91ZHRfbmFtZSxcbiAgICAgIGNoYXJhY3Rlcl9tYXhpbXVtX2xlbmd0aCxcbiAgICAgIG51bWVyaWNfcHJlY2lzaW9uLFxuICAgICAgbnVtZXJpY19zY2FsZSxcbiAgICB9ID0gZGJDb2x1bW47XG5cbiAgICBjb25zdCB7IHVkdF9uYW1lLCBzaW5nbGVPckFycmF5IH0gPSAoKCkgPT4ge1xuICAgICAgaWYgKF91ZHRfbmFtZS5zdGFydHNXaXRoKFwiX1wiKSkge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIHVkdF9uYW1lOiBfdWR0X25hbWUuc3Vic3RyaW5nKDEpLFxuICAgICAgICAgIHNpbmdsZU9yQXJyYXk6IFwiW11cIiBhcyBjb25zdCxcbiAgICAgICAgfTtcbiAgICAgIH1cbiAgICAgIHJldHVybiB7XG4gICAgICAgIHVkdF9uYW1lOiBfdWR0X25hbWUsXG4gICAgICAgIHNpbmdsZU9yQXJyYXk6IFwiXCIgYXMgY29uc3QsXG4gICAgICB9O1xuICAgIH0pKCk7XG5cbiAgICAvLyBVVUlEXG4gICAgaWYgKHVkdF9uYW1lID09PSBcInV1aWRcIikge1xuICAgICAgcmV0dXJuIHsgdHlwZTogYHV1aWQke3NpbmdsZU9yQXJyYXl9YCB9O1xuICAgIH1cblxuICAgIC8vIEludGVnZXIgdHlwZXNcbiAgICBpZiAodWR0X25hbWUgPT09IFwiaW50NFwiKSB7XG4gICAgICByZXR1cm4geyB0eXBlOiBgaW50ZWdlciR7c2luZ2xlT3JBcnJheX1gIH07XG4gICAgfVxuICAgIGlmICh1ZHRfbmFtZSA9PT0gXCJpbnQ4XCIpIHtcbiAgICAgIHJldHVybiB7IHR5cGU6IGBiaWdJbnRlZ2VyJHtzaW5nbGVPckFycmF5fWAgfTtcbiAgICB9XG5cbiAgICAvLyBTdHJpbmcgdHlwZXNcbiAgICBpZiAodWR0X25hbWUgPT09IFwidmFyY2hhclwiKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICB0eXBlOiBgc3RyaW5nJHtzaW5nbGVPckFycmF5fWAsXG4gICAgICAgIC4uLihjaGFyYWN0ZXJfbWF4aW11bV9sZW5ndGggJiYge1xuICAgICAgICAgIGxlbmd0aDogY2hhcmFjdGVyX21heGltdW1fbGVuZ3RoLFxuICAgICAgICB9KSxcbiAgICAgIH07XG4gICAgfVxuICAgIGlmICh1ZHRfbmFtZSA9PT0gXCJ0ZXh0XCIpIHtcbiAgICAgIHJldHVybiB7IHR5cGU6IGBzdHJpbmcke3NpbmdsZU9yQXJyYXl9YCB9OyAvLyBTdHJpbmdQcm9wIHdpdGhvdXQgbGVuZ3RoXG4gICAgfVxuXG4gICAgLy8gTnVtYmVyT3JOdW1lcmljIHR5cGVzXG4gICAgaWYgKHVkdF9uYW1lID09PSBcIm51bWVyaWNcIikge1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgdHlwZTogYG51bWJlck9yTnVtZXJpYyR7c2luZ2xlT3JBcnJheX1gLFxuICAgICAgICBudW1iZXJUeXBlOiBcIm51bWVyaWNcIixcbiAgICAgICAgLi4uKG51bWVyaWNfcHJlY2lzaW9uICE9PSBudWxsICYmXG4gICAgICAgICAgbnVtZXJpY19zY2FsZSAhPT0gbnVsbCAmJiB7XG4gICAgICAgICAgICBwcmVjaXNpb246IG51bWVyaWNfcHJlY2lzaW9uLFxuICAgICAgICAgICAgc2NhbGU6IG51bWVyaWNfc2NhbGUsXG4gICAgICAgICAgfSksXG4gICAgICB9O1xuICAgIH1cbiAgICBpZiAodWR0X25hbWUgPT09IFwiZmxvYXQ0XCIpIHtcbiAgICAgIHJldHVybiB7IHR5cGU6IGBudW1iZXJPck51bWVyaWMke3NpbmdsZU9yQXJyYXl9YCwgbnVtYmVyVHlwZTogXCJyZWFsXCIgfTtcbiAgICB9XG4gICAgaWYgKHVkdF9uYW1lID09PSBcImZsb2F0OFwiKSB7XG4gICAgICByZXR1cm4geyB0eXBlOiBgbnVtYmVyT3JOdW1lcmljJHtzaW5nbGVPckFycmF5fWAsIG51bWJlclR5cGU6IFwiZG91YmxlIHByZWNpc2lvblwiIH07XG4gICAgfVxuXG4gICAgLy8gQm9vbGVhblxuICAgIGlmICh1ZHRfbmFtZSA9PT0gXCJib29sXCIpIHtcbiAgICAgIHJldHVybiB7IHR5cGU6IGBib29sZWFuJHtzaW5nbGVPckFycmF5fWAgfTtcbiAgICB9XG5cbiAgICAvLyBUaW1lc3RhbXB6IHR5cGVzXG4gICAgaWYgKHVkdF9uYW1lID09PSBcInRpbWVzdGFtcHR6XCIpIHtcbiAgICAgIHJldHVybiB7IHR5cGU6IGBkYXRlJHtzaW5nbGVPckFycmF5fWAgfTsgLy8gRGF0ZVByb3Ag4oaSIHRpbWVzdGFtcHR6XG4gICAgfVxuXG4gICAgLy8gSlNPTlxuICAgIGlmICh1ZHRfbmFtZSA9PT0gXCJqc29uXCIgfHwgdWR0X25hbWUgPT09IFwianNvbmJcIikge1xuICAgICAgcmV0dXJuIHsgdHlwZTogXCJqc29uXCIgfTtcbiAgICB9XG5cbiAgICB0aHJvdyBuZXcgRXJyb3IoYHJlc29sdmUg67aI6rCA64ql7ZWcIFBvc3RncmVTUUwg7Lus65+8IO2DgOyehTogJHt1ZHRfbmFtZX1gKTtcbiAgfVxufVxuXG5leHBvcnQgY29uc3QgUG9zdGdyZVNRTFNjaGVtYVJlYWRlciA9IG5ldyBQb3N0Z3JlU1FMU2NoZW1hUmVhZGVyQ2xhc3MoKTtcbiJdLCJuYW1lcyI6WyJhc3NlcnQiLCJncm91cCIsIlBvc3RncmVTUUxTY2hlbWFSZWFkZXJDbGFzcyIsImdldE1pZ3JhdGlvblNldEZyb21EQiIsImNvbXBhcmVEQiIsInRhYmxlIiwiZGJDb2x1bW5zIiwiZGJJbmRleGVzIiwiZGJGb3JlaWducyIsInJlYWRUYWJsZSIsImUiLCJFcnJvciIsIm1lc3NhZ2UiLCJpbmNsdWRlcyIsImNvbnNvbGUiLCJlcnJvciIsImNvbHVtbnMiLCJtYXAiLCJkYkNvbHVtbiIsImRiQ29sVHlwZSIsInJlc29sdmVEQkNvbFR5cGUiLCJuYW1lIiwiY29sdW1uX25hbWUiLCJudWxsYWJsZSIsImlzX251bGxhYmxlIiwiY29sdW1uX2RlZmF1bHQiLCJkZWZhdWx0VmFsdWUiLCJzdGFydHNXaXRoIiwicmVwbGFjZSIsImVuZHNXaXRoIiwic2xpY2UiLCJkZWZhdWx0VG8iLCJkYkluZGV4ZXNHcm91cCIsImZpbHRlciIsImRiSW5kZXgiLCJpc19wcmltYXJ5IiwiZmluZCIsImRiRm9yZWlnbiIsImluZGV4X25hbWUiLCJjb25zdHJhaW50X25hbWUiLCJpbmRleGVzIiwiT2JqZWN0Iiwia2V5cyIsImluZGV4TmFtZSIsImN1cnJlbnRJbmRleGVzIiwiZmlyc3RJbmRleCIsInR5cGUiLCJpc191bmlxdWUiLCJpZHgiLCJmb3JlaWducyIsInRvIiwiZm9yZWlnbl90YWJsZV9uYW1lIiwiZm9yZWlnbl9jb2x1bW5fbmFtZSIsIm9uVXBkYXRlIiwibWFwQ29uc3RyYWludEFjdGlvbiIsInVwZGF0ZV9ydWxlIiwib25EZWxldGUiLCJkZWxldGVfcnVsZSIsImFjdGlvbiIsImFjdGlvbk1hcCIsIlJFU1RSSUNUIiwiQ0FTQ0FERSIsInRhYmxlTmFtZSIsInNlbGVjdCIsImZyb20iLCJ3aGVyZSIsInRhYmxlX25hbWUiLCJvcmRlckJ5IiwibGVuZ3RoIiwiaW5kZXhlc1F1ZXJ5IiwicmF3Iiwicm93cyIsImZvcmVpZ25zUXVlcnkiLCJ1ZHRfbmFtZSIsIl91ZHRfbmFtZSIsImNoYXJhY3Rlcl9tYXhpbXVtX2xlbmd0aCIsIm51bWVyaWNfcHJlY2lzaW9uIiwibnVtZXJpY19zY2FsZSIsInNpbmdsZU9yQXJyYXkiLCJzdWJzdHJpbmciLCJudW1iZXJUeXBlIiwicHJlY2lzaW9uIiwic2NhbGUiLCJQb3N0Z3JlU1FMU2NoZW1hUmVhZGVyIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPQSxZQUFZLFNBQVM7QUFFNUIsU0FBU0MsS0FBSyxRQUFRLFVBQVU7QUFxQ2hDLE1BQU1DO0lBQ0o7Ozs7O0dBS0MsR0FDRCxNQUFNQyxzQkFBc0JDLFNBQWUsRUFBRUMsS0FBYSxFQUFnQztRQUN4RixJQUFJQyxXQUF1QkMsV0FBc0JDO1FBQ2pELElBQUk7WUFDRixDQUFDRixXQUFXQyxXQUFXQyxXQUFXLEdBQUcsTUFBTSxJQUFJLENBQUNDLFNBQVMsQ0FBQ0wsV0FBV0M7UUFDdkUsRUFBRSxPQUFPSyxHQUFZO1lBQ25CLElBQUlBLGFBQWFDLFNBQVNELEVBQUVFLE9BQU8sQ0FBQ0MsUUFBUSxDQUFDLG9CQUFvQjtnQkFDL0QsT0FBTztZQUNUO1lBQ0FDLFFBQVFDLEtB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
|
|
@@ -1,53 +1,21 @@
|
|
|
1
|
-
import { SonamuDBConfig } from "../database/db";
|
|
2
|
-
import { GenMigrationCode } from "../types/types";
|
|
1
|
+
import type { SonamuDBConfig } from "../database/db";
|
|
2
|
+
import type { GenMigrationCode } from "../types/types";
|
|
3
3
|
export type MigrationCode = {
|
|
4
4
|
name: string;
|
|
5
5
|
path: string;
|
|
6
6
|
};
|
|
7
|
-
export type ConnString = `${"
|
|
7
|
+
export type ConnString = `${"pg"}://${string}@${string}:${number}/${string}`;
|
|
8
8
|
export type MigrationStatus = {
|
|
9
9
|
codes: MigrationCode[];
|
|
10
10
|
conns: {
|
|
11
11
|
name: string;
|
|
12
12
|
connKey: keyof SonamuDBConfig;
|
|
13
13
|
connString: ConnString;
|
|
14
|
-
currentVersion: string;
|
|
15
|
-
status:
|
|
14
|
+
currentVersion: string | "error";
|
|
15
|
+
status: number | "error";
|
|
16
16
|
pending: string[];
|
|
17
17
|
}[];
|
|
18
18
|
preparedCodes: GenMigrationCode[];
|
|
19
|
-
|
|
20
|
-
export type DBColumn = {
|
|
21
|
-
Field: string;
|
|
22
|
-
Type: string;
|
|
23
|
-
Null: string;
|
|
24
|
-
Key: string;
|
|
25
|
-
Default: string | null;
|
|
26
|
-
Extra: string;
|
|
27
|
-
};
|
|
28
|
-
export type DBIndex = {
|
|
29
|
-
Table: string;
|
|
30
|
-
Non_unique: number;
|
|
31
|
-
Key_name: string;
|
|
32
|
-
Seq_in_index: number;
|
|
33
|
-
Column_name: string;
|
|
34
|
-
Collation: string | null;
|
|
35
|
-
Cardinality: number | null;
|
|
36
|
-
Sub_part: number | null;
|
|
37
|
-
Packed: string | null;
|
|
38
|
-
Null: string;
|
|
39
|
-
Index_type: string;
|
|
40
|
-
Comment: string;
|
|
41
|
-
Index_comment: string;
|
|
42
|
-
Visible: string;
|
|
43
|
-
Expression: string | null;
|
|
44
|
-
};
|
|
45
|
-
export type DBForeign = {
|
|
46
|
-
keyName: string;
|
|
47
|
-
from: string;
|
|
48
|
-
referencesTable: string;
|
|
49
|
-
referencesField: string;
|
|
50
|
-
onDelete: string;
|
|
51
|
-
onUpdate: string;
|
|
19
|
+
error?: string;
|
|
52
20
|
};
|
|
53
21
|
//# sourceMappingURL=types.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/migration/types.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/migration/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AACrD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAEvD,MAAM,MAAM,aAAa,GAAG;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;CACd,CAAC;AACF,MAAM,MAAM,UAAU,GAAG,GAAG,IAAI,MAAM,MAAM,IAAI,MAAM,IAAI,MAAM,IAAI,MAAM,EAAE,CAAC;AAC7E,MAAM,MAAM,eAAe,GAAG;IAC5B,KAAK,EAAE,aAAa,EAAE,CAAC;IACvB,KAAK,EAAE;QACL,IAAI,EAAE,MAAM,CAAC;QACb,OAAO,EAAE,MAAM,cAAc,CAAC;QAC9B,UAAU,EAAE,UAAU,CAAC;QACvB,cAAc,EAAE,MAAM,GAAG,OAAO,CAAC;QACjC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC;QACzB,OAAO,EAAE,MAAM,EAAE,CAAC;KACnB,EAAE,CAAC;IACJ,aAAa,EAAE,gBAAgB,EAAE,CAAC;IAClC,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB,CAAC"}
|
package/dist/migration/types.js
CHANGED
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
export { };
|
|
2
2
|
|
|
3
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9taWdyYXRpb24vdHlwZXMudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBTb25hbXVEQkNvbmZpZyB9IGZyb20gXCIuLi9kYXRhYmFzZS9kYlwiO1xuaW1wb3J0IHR5cGUgeyBHZW5NaWdyYXRpb25Db2RlIH0gZnJvbSBcIi4uL3R5cGVzL3R5cGVzXCI7XG5cbmV4cG9ydCB0eXBlIE1pZ3JhdGlvbkNvZGUgPSB7XG4gIG5hbWU6IHN0cmluZztcbiAgcGF0aDogc3RyaW5nO1xufTtcbmV4cG9ydCB0eXBlIENvbm5TdHJpbmcgPSBgJHtcInBnXCJ9Oi8vJHtzdHJpbmd9QCR7c3RyaW5nfToke251bWJlcn0vJHtzdHJpbmd9YDsgLy8gcGc6Ly9hY2NvdW50QGhvc3Q6cG9ydC9kYXRhYmFzZVxuZXhwb3J0IHR5cGUgTWlncmF0aW9uU3RhdHVzID0ge1xuICBjb2RlczogTWlncmF0aW9uQ29kZVtdO1xuICBjb25uczoge1xuICAgIG5hbWU6IHN0cmluZztcbiAgICBjb25uS2V5OiBrZXlvZiBTb25hbXVEQkNvbmZpZztcbiAgICBjb25uU3RyaW5nOiBDb25uU3RyaW5nO1xuICAgIGN1cnJlbnRWZXJzaW9uOiBzdHJpbmcgfCBcImVycm9yXCI7XG4gICAgc3RhdHVzOiBudW1iZXIgfCBcImVycm9yXCI7XG4gICAgcGVuZGluZzogc3RyaW5nW107XG4gIH1bXTtcbiAgcHJlcGFyZWRDb2RlczogR2VuTWlncmF0aW9uQ29kZVtdO1xuICBlcnJvcj86IHN0cmluZztcbn07XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBUUEsV0FZRSJ9
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Sonamu extension과 공유하는 Naite 메시징 관련 타입 정의들입니다.
|
|
3
|
+
* 이 파일은 cartanova-ai/sonamu와 cartanova-ai/vscode-sonamu에서 공통으로 사용됩니다.
|
|
4
|
+
*/
|
|
5
|
+
export declare namespace NaiteMessagingTypes {
|
|
6
|
+
type NaiteRunStartMessage = {
|
|
7
|
+
type: "run/start";
|
|
8
|
+
startedAt: string;
|
|
9
|
+
};
|
|
10
|
+
type NaiteTestResultMessage = {
|
|
11
|
+
type: "test/result";
|
|
12
|
+
receivedAt: string;
|
|
13
|
+
} & TestResult;
|
|
14
|
+
type NaiteRunEndMessage = {
|
|
15
|
+
type: "run/end";
|
|
16
|
+
endedAt: string;
|
|
17
|
+
};
|
|
18
|
+
type NaiteMessage = NaiteRunStartMessage | NaiteTestResultMessage | NaiteRunEndMessage;
|
|
19
|
+
type NaiteTrace = {
|
|
20
|
+
key: string;
|
|
21
|
+
value: any;
|
|
22
|
+
filePath: string;
|
|
23
|
+
lineNumber: number;
|
|
24
|
+
at: string;
|
|
25
|
+
};
|
|
26
|
+
type TestError = {
|
|
27
|
+
message: string;
|
|
28
|
+
stack?: string;
|
|
29
|
+
};
|
|
30
|
+
type TestResult = {
|
|
31
|
+
suiteName: string;
|
|
32
|
+
suiteFilePath?: string;
|
|
33
|
+
testName: string;
|
|
34
|
+
testFilePath: string;
|
|
35
|
+
testLine: number;
|
|
36
|
+
status: string;
|
|
37
|
+
duration: number;
|
|
38
|
+
error?: TestError;
|
|
39
|
+
traces: NaiteTrace[];
|
|
40
|
+
receivedAt: string;
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
//# sourceMappingURL=messaging-types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"messaging-types.d.ts","sourceRoot":"","sources":["../../src/naite/messaging-types.ts"],"names":[],"mappings":"AAEA;;;GAGG;AACH,yBAAiB,mBAAmB,CAAC;IACjC,KAAY,oBAAoB,GAAG;QACjC,IAAI,EAAE,WAAW,CAAC;QAClB,SAAS,EAAE,MAAM,CAAC;KACnB,CAAC;IAEF,KAAY,sBAAsB,GAAG;QACnC,IAAI,EAAE,aAAa,CAAC;QACpB,UAAU,EAAE,MAAM,CAAC;KACpB,GAAG,UAAU,CAAC;IAEf,KAAY,kBAAkB,GAAG;QAC/B,IAAI,EAAE,SAAS,CAAC;QAChB,OAAO,EAAE,MAAM,CAAC;KACjB,CAAC;IAEF,KAAY,YAAY,GAAG,oBAAoB,GAAG,sBAAsB,GAAG,kBAAkB,CAAC;IAE9F,KAAY,UAAU,GAAG;QACvB,GAAG,EAAE,MAAM,CAAC;QACZ,KAAK,EAAE,GAAG,CAAC;QACX,QAAQ,EAAE,MAAM,CAAC;QACjB,UAAU,EAAE,MAAM,CAAC;QACnB,EAAE,EAAE,MAAM,CAAC;KACZ,CAAC;IAEF,KAAY,SAAS,GAAG;QACtB,OAAO,EAAE,MAAM,CAAC;QAChB,KAAK,CAAC,EAAE,MAAM,CAAC;KAChB,CAAC;IAEF,KAAY,UAAU,GAAG;QACvB,SAAS,EAAE,MAAM,CAAC;QAClB,aAAa,CAAC,EAAE,MAAM,CAAC;QACvB,QAAQ,EAAE,MAAM,CAAC;QACjB,YAAY,EAAE,MAAM,CAAC;QACrB,QAAQ,EAAE,MAAM,CAAC;QACjB,MAAM,EAAE,MAAM,CAAC;QACf,QAAQ,EAAE,MAAM,CAAC;QACjB,KAAK,CAAC,EAAE,SAAS,CAAC;QAClB,MAAM,EAAE,UAAU,EAAE,CAAC;QACrB,UAAU,EAAE,MAAM,CAAC;KACpB,CAAC;CACH"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
// biome-ignore-all lint/suspicious/noExplicitAny: Naite는 expect와 호응하도록 any를 허용함
|
|
2
|
+
/**
|
|
3
|
+
* Sonamu extension과 공유하는 Naite 메시징 관련 타입 정의들입니다.
|
|
4
|
+
* 이 파일은 cartanova-ai/sonamu와 cartanova-ai/vscode-sonamu에서 공통으로 사용됩니다.
|
|
5
|
+
*/ export { };
|
|
6
|
+
|
|
7
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9uYWl0ZS9tZXNzYWdpbmctdHlwZXMudHMiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gYmlvbWUtaWdub3JlLWFsbCBsaW50L3N1c3BpY2lvdXMvbm9FeHBsaWNpdEFueTogTmFpdGXripQgZXhwZWN07JmAIO2YuOydke2VmOuPhOuhnSBhbnnrpbwg7ZeI7Jqp7ZWoXG5cbi8qKlxuICogU29uYW11IGV4dGVuc2lvbuqzvCDqs7XsnKDtlZjripQgTmFpdGUg66mU7Iuc7KeVIOq0gOugqCDtg4DsnoUg7KCV7J2Y65Ok7J6F64uI64ukLlxuICog7J20IO2MjOydvOydgCBjYXJ0YW5vdmEtYWkvc29uYW117JmAIGNhcnRhbm92YS1haS92c2NvZGUtc29uYW117JeQ7IScIOqzte2GteycvOuhnCDsgqzsmqnrkKnri4jri6QuXG4gKi9cbmV4cG9ydCBuYW1lc3BhY2UgTmFpdGVNZXNzYWdpbmdUeXBlcyB7XG4gICAgZXhwb3J0IHR5cGUgTmFpdGVSdW5TdGFydE1lc3NhZ2UgPSB7XG4gICAgICB0eXBlOiBcInJ1bi9zdGFydFwiO1xuICAgICAgc3RhcnRlZEF0OiBzdHJpbmc7XG4gICAgfTtcbiAgXG4gICAgZXhwb3J0IHR5cGUgTmFpdGVUZXN0UmVzdWx0TWVzc2FnZSA9IHtcbiAgICAgIHR5cGU6IFwidGVzdC9yZXN1bHRcIjtcbiAgICAgIHJlY2VpdmVkQXQ6IHN0cmluZztcbiAgICB9ICYgVGVzdFJlc3VsdDtcbiAgXG4gICAgZXhwb3J0IHR5cGUgTmFpdGVSdW5FbmRNZXNzYWdlID0ge1xuICAgICAgdHlwZTogXCJydW4vZW5kXCI7XG4gICAgICBlbmRlZEF0OiBzdHJpbmc7XG4gICAgfTtcbiAgXG4gICAgZXhwb3J0IHR5cGUgTmFpdGVNZXNzYWdlID0gTmFpdGVSdW5TdGFydE1lc3NhZ2UgfCBOYWl0ZVRlc3RSZXN1bHRNZXNzYWdlIHwgTmFpdGVSdW5FbmRNZXNzYWdlO1xuICBcbiAgICBleHBvcnQgdHlwZSBOYWl0ZVRyYWNlID0ge1xuICAgICAga2V5OiBzdHJpbmc7XG4gICAgICB2YWx1ZTogYW55O1xuICAgICAgZmlsZVBhdGg6IHN0cmluZztcbiAgICAgIGxpbmVOdW1iZXI6IG51bWJlcjtcbiAgICAgIGF0OiBzdHJpbmc7XG4gICAgfTtcbiAgXG4gICAgZXhwb3J0IHR5cGUgVGVzdEVycm9yID0ge1xuICAgICAgbWVzc2FnZTogc3RyaW5nO1xuICAgICAgc3RhY2s/OiBzdHJpbmc7XG4gICAgfTtcbiAgXG4gICAgZXhwb3J0IHR5cGUgVGVzdFJlc3VsdCA9IHtcbiAgICAgIHN1aXRlTmFtZTogc3RyaW5nO1xuICAgICAgc3VpdGVGaWxlUGF0aD86IHN0cmluZztcbiAgICAgIHRlc3ROYW1lOiBzdHJpbmc7XG4gICAgICB0ZXN0RmlsZVBhdGg6IHN0cmluZztcbiAgICAgIHRlc3RMaW5lOiBudW1iZXI7XG4gICAgICBzdGF0dXM6IHN0cmluZztcbiAgICAgIGR1cmF0aW9uOiBudW1iZXI7XG4gICAgICBlcnJvcj86IFRlc3RFcnJvcjtcbiAgICAgIHRyYWNlczogTmFpdGVUcmFjZVtdO1xuICAgICAgcmVjZWl2ZWRBdDogc3RyaW5nO1xuICAgIH07XG4gIH1cbiAgIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGdGQUFnRjtBQUVoRjs7O0NBR0MsR0FDRCxXQTJDRyJ9
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* NaiteReporter
|
|
3
|
+
*
|
|
4
|
+
* 테스트 결과와 Trace 정보를 Unix Socket으로 VS Code extension에 전달합니다.
|
|
5
|
+
* extension이 ~/.sonamu/naite.sock에 소켓 서버를 열어둡니다.
|
|
6
|
+
*
|
|
7
|
+
* fs mock 충돌을 피하기 위해 net 모듈만 사용합니다.
|
|
8
|
+
*/
|
|
9
|
+
/** biome-ignore-all lint/suspicious/noExplicitAny: Naite는 expect와 호응하도록 any를 허용함 */
|
|
10
|
+
import type { NaiteMessagingTypes } from "./messaging-types";
|
|
11
|
+
declare class NaiteReporterClass {
|
|
12
|
+
private socket;
|
|
13
|
+
private connected;
|
|
14
|
+
private buffer;
|
|
15
|
+
/**
|
|
16
|
+
* 소켓 연결 시도
|
|
17
|
+
*/
|
|
18
|
+
private ensureConnection;
|
|
19
|
+
/**
|
|
20
|
+
* 메시지 전송 (줄바꿈으로 구분)
|
|
21
|
+
*/
|
|
22
|
+
private send;
|
|
23
|
+
/**
|
|
24
|
+
* beforeAll에서 호출합니다.
|
|
25
|
+
* 테스트 run 시작을 알립니다 (데이터 클리어 신호).
|
|
26
|
+
*/
|
|
27
|
+
startTestRun(): void;
|
|
28
|
+
/**
|
|
29
|
+
* afterEach에서 호출합니다.
|
|
30
|
+
* 테스트 케이스 결과를 traces와 함께 전송합니다.
|
|
31
|
+
*/
|
|
32
|
+
reportTestResult(result: Omit<NaiteMessagingTypes.TestResult, "receivedAt">): void;
|
|
33
|
+
/**
|
|
34
|
+
* afterAll에서 호출합니다.
|
|
35
|
+
* 테스트 run 종료를 알립니다.
|
|
36
|
+
*/
|
|
37
|
+
endTestRun(): void;
|
|
38
|
+
}
|
|
39
|
+
export declare const NaiteReporter: NaiteReporterClass;
|
|
40
|
+
export {};
|
|
41
|
+
//# sourceMappingURL=naite-reporter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"naite-reporter.d.ts","sourceRoot":"","sources":["../../src/naite/naite-reporter.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AACH,oFAAoF;AAKpF,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAM7D,cAAM,kBAAkB;IACtB,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,MAAM,CAAgB;IAE9B;;OAEG;IACH,OAAO,CAAC,gBAAgB;IA8BxB;;OAEG;IACH,OAAO,CAAC,IAAI;IAaZ;;;OAGG;IACH,YAAY,IAAI,IAAI;IAWpB;;;OAGG;IACH,gBAAgB,CAAC,MAAM,EAAE,IAAI,CAAC,mBAAmB,CAAC,UAAU,EAAE,YAAY,CAAC,GAAG,IAAI;IAYlF;;;OAGG;IACH,UAAU,IAAI,IAAI;CAiBnB;AAED,eAAO,MAAM,aAAa,oBAA2B,CAAC"}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* NaiteReporter
|
|
3
|
+
*
|
|
4
|
+
* 테스트 결과와 Trace 정보를 Unix Socket으로 VS Code extension에 전달합니다.
|
|
5
|
+
* extension이 ~/.sonamu/naite.sock에 소켓 서버를 열어둡니다.
|
|
6
|
+
*
|
|
7
|
+
* fs mock 충돌을 피하기 위해 net 모듈만 사용합니다.
|
|
8
|
+
*/ /** biome-ignore-all lint/suspicious/noExplicitAny: Naite는 expect와 호응하도록 any를 허용함 */ import { connect } from "net";
|
|
9
|
+
import { homedir } from "os";
|
|
10
|
+
import { join } from "path";
|
|
11
|
+
// 소켓 경로
|
|
12
|
+
const SOCKET_PATH = process.platform === "win32" ? "\\\\.\\pipe\\naite" : join(homedir(), ".sonamu", "naite.sock");
|
|
13
|
+
class NaiteReporterClass {
|
|
14
|
+
socket = null;
|
|
15
|
+
connected = false;
|
|
16
|
+
buffer = [];
|
|
17
|
+
/**
|
|
18
|
+
* 소켓 연결 시도
|
|
19
|
+
*/ ensureConnection() {
|
|
20
|
+
if (this.connected || this.socket) return;
|
|
21
|
+
try {
|
|
22
|
+
this.socket = connect(SOCKET_PATH);
|
|
23
|
+
this.socket.on("connect", ()=>{
|
|
24
|
+
this.connected = true;
|
|
25
|
+
// 버퍼에 쌓인 메시지 전송
|
|
26
|
+
for (const msg of this.buffer){
|
|
27
|
+
this.socket?.write(msg);
|
|
28
|
+
}
|
|
29
|
+
this.buffer = [];
|
|
30
|
+
});
|
|
31
|
+
this.socket.on("error", ()=>{
|
|
32
|
+
// 연결 실패 무시 (extension이 꺼져있을 수 있음)
|
|
33
|
+
this.connected = false;
|
|
34
|
+
this.socket = null;
|
|
35
|
+
});
|
|
36
|
+
this.socket.on("close", ()=>{
|
|
37
|
+
this.connected = false;
|
|
38
|
+
this.socket = null;
|
|
39
|
+
});
|
|
40
|
+
} catch {
|
|
41
|
+
// 연결 실패 무시
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* 메시지 전송 (줄바꿈으로 구분)
|
|
46
|
+
*/ send(data) {
|
|
47
|
+
const msg = `${JSON.stringify(data)}\n`;
|
|
48
|
+
this.ensureConnection();
|
|
49
|
+
if (this.connected && this.socket) {
|
|
50
|
+
this.socket.write(msg);
|
|
51
|
+
} else {
|
|
52
|
+
// 연결 대기 중이면 버퍼에 저장
|
|
53
|
+
this.buffer.push(msg);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* beforeAll에서 호출합니다.
|
|
58
|
+
* 테스트 run 시작을 알립니다 (데이터 클리어 신호).
|
|
59
|
+
*/ startTestRun() {
|
|
60
|
+
if (process.env.CI) {
|
|
61
|
+
return;
|
|
62
|
+
}
|
|
63
|
+
this.send({
|
|
64
|
+
type: "run/start",
|
|
65
|
+
startedAt: new Date().toISOString()
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* afterEach에서 호출합니다.
|
|
70
|
+
* 테스트 케이스 결과를 traces와 함께 전송합니다.
|
|
71
|
+
*/ reportTestResult(result) {
|
|
72
|
+
if (process.env.CI) {
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
this.send({
|
|
76
|
+
type: "test/result",
|
|
77
|
+
receivedAt: new Date().toISOString(),
|
|
78
|
+
...result
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* afterAll에서 호출합니다.
|
|
83
|
+
* 테스트 run 종료를 알립니다.
|
|
84
|
+
*/ endTestRun() {
|
|
85
|
+
if (process.env.CI) {
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
this.send({
|
|
89
|
+
type: "run/end",
|
|
90
|
+
endedAt: new Date().toISOString()
|
|
91
|
+
});
|
|
92
|
+
// 연결 종료
|
|
93
|
+
if (this.socket) {
|
|
94
|
+
this.socket.end();
|
|
95
|
+
this.socket = null;
|
|
96
|
+
this.connected = false;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
export const NaiteReporter = new NaiteReporterClass();
|
|
101
|
+
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9uYWl0ZS9uYWl0ZS1yZXBvcnRlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE5haXRlUmVwb3J0ZXJcbiAqXG4gKiDthYzsiqTtirgg6rKw6rO87JmAIFRyYWNlIOygleuztOulvCBVbml4IFNvY2tldOycvOuhnCBWUyBDb2RlIGV4dGVuc2lvbuyXkCDsoITri6ztlanri4jri6QuXG4gKiBleHRlbnNpb27snbQgfi8uc29uYW11L25haXRlLnNvY2vsl5Ag7IaM7LyTIOyEnOuyhOulvCDsl7TslrTrkaHri4jri6QuXG4gKlxuICogZnMgbW9jayDstqnrj4zsnYQg7ZS87ZWY6riwIOychO2VtCBuZXQg66qo65OI66eMIOyCrOyaqe2VqeuLiOuLpC5cbiAqL1xuLyoqIGJpb21lLWlnbm9yZS1hbGwgbGludC9zdXNwaWNpb3VzL25vRXhwbGljaXRBbnk6IE5haXRl64qUIGV4cGVjdOyZgCDtmLjsnZHtlZjrj4TroZ0gYW5566W8IO2XiOyaqe2VqCAqL1xuXG5pbXBvcnQgeyBjb25uZWN0LCB0eXBlIFNvY2tldCB9IGZyb20gXCJuZXRcIjtcbmltcG9ydCB7IGhvbWVkaXIgfSBmcm9tIFwib3NcIjtcbmltcG9ydCB7IGpvaW4gfSBmcm9tIFwicGF0aFwiO1xuaW1wb3J0IHR5cGUgeyBOYWl0ZU1lc3NhZ2luZ1R5cGVzIH0gZnJvbSBcIi4vbWVzc2FnaW5nLXR5cGVzXCI7XG5cbi8vIOyGjOy8kyDqsr3roZxcbmNvbnN0IFNPQ0tFVF9QQVRIID1cbiAgcHJvY2Vzcy5wbGF0Zm9ybSA9PT0gXCJ3aW4zMlwiID8gXCJcXFxcXFxcXC5cXFxccGlwZVxcXFxuYWl0ZVwiIDogam9pbihob21lZGlyKCksIFwiLnNvbmFtdVwiLCBcIm5haXRlLnNvY2tcIik7XG5cbmNsYXNzIE5haXRlUmVwb3J0ZXJDbGFzcyB7XG4gIHByaXZhdGUgc29ja2V0OiBTb2NrZXQgfCBudWxsID0gbnVsbDtcbiAgcHJpdmF0ZSBjb25uZWN0ZWQgPSBmYWxzZTtcbiAgcHJpdmF0ZSBidWZmZXI6IHN0cmluZ1tdID0gW107XG5cbiAgLyoqXG4gICAqIOyGjOy8kyDsl7DqsrAg7Iuc64+EXG4gICAqL1xuICBwcml2YXRlIGVuc3VyZUNvbm5lY3Rpb24oKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuY29ubmVjdGVkIHx8IHRoaXMuc29ja2V0KSByZXR1cm47XG5cbiAgICB0cnkge1xuICAgICAgdGhpcy5zb2NrZXQgPSBjb25uZWN0KFNPQ0tFVF9QQVRIKTtcblxuICAgICAgdGhpcy5zb2NrZXQub24oXCJjb25uZWN0XCIsICgpID0+IHtcbiAgICAgICAgdGhpcy5jb25uZWN0ZWQgPSB0cnVlO1xuICAgICAgICAvLyDrsoTtjbzsl5Ag7IyT7J24IOuplOyLnOyngCDsoITshqFcbiAgICAgICAgZm9yIChjb25zdCBtc2cgb2YgdGhpcy5idWZmZXIpIHtcbiAgICAgICAgICB0aGlzLnNvY2tldD8ud3JpdGUobXNnKTtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLmJ1ZmZlciA9IFtdO1xuICAgICAgfSk7XG5cbiAgICAgIHRoaXMuc29ja2V0Lm9uKFwiZXJyb3JcIiwgKCkgPT4ge1xuICAgICAgICAvLyDsl7DqsrAg7Iuk7YyoIOustOyLnCAoZXh0ZW5zaW9u7J20IOq6vOyguOyeiOydhCDsiJgg7J6I7J2MKVxuICAgICAgICB0aGlzLmNvbm5lY3RlZCA9IGZhbHNlO1xuICAgICAgICB0aGlzLnNvY2tldCA9IG51bGw7XG4gICAgICB9KTtcblxuICAgICAgdGhpcy5zb2NrZXQub24oXCJjbG9zZVwiLCAoKSA9PiB7XG4gICAgICAgIHRoaXMuY29ubmVjdGVkID0gZmFsc2U7XG4gICAgICAgIHRoaXMuc29ja2V0ID0gbnVsbDtcbiAgICAgIH0pO1xuICAgIH0gY2F0Y2gge1xuICAgICAgLy8g7Jew6rKwIOyLpO2MqCDrrLTsi5xcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICog66mU7Iuc7KeAIOyghOyGoSAo7KSE67CU6r+I7Jy866GcIOq1rOu2hClcbiAgICovXG4gIHByaXZhdGUgc2VuZChkYXRhOiBOYWl0ZU1lc3NhZ2luZ1R5cGVzLk5haXRlTWVzc2FnZSk6IHZvaWQge1xuICAgIGNvbnN0IG1zZyA9IGAke0pTT04uc3RyaW5naWZ5KGRhdGEpfVxcbmA7XG5cbiAgICB0aGlzLmVuc3VyZUNvbm5lY3Rpb24oKTtcblxuICAgIGlmICh0aGlzLmNvbm5lY3RlZCAmJiB0aGlzLnNvY2tldCkge1xuICAgICAgdGhpcy5zb2NrZXQud3JpdGUobXNnKTtcbiAgICB9IGVsc2Uge1xuICAgICAgLy8g7Jew6rKwIOuMgOq4sCDspJHsnbTrqbQg67KE7Y287JeQIOyggOyepVxuICAgICAgdGhpcy5idWZmZXIucHVzaChtc2cpO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBiZWZvcmVBbGzsl5DshJwg7Zi47Lac7ZWp64uI64ukLlxuICAgKiDthYzsiqTtirggcnVuIOyLnOyekeydhCDslYzrpr3ri4jri6QgKOuNsOydtO2EsCDtgbTrpqzslrQg7Iug7Zi4KS5cbiAgICovXG4gIHN0YXJ0VGVzdFJ1bigpOiB2b2lkIHtcbiAgICBpZiAocHJvY2Vzcy5lbnYuQ0kpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICB0aGlzLnNlbmQoe1xuICAgICAgdHlwZTogXCJydW4vc3RhcnRcIixcbiAgICAgIHN0YXJ0ZWRBdDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIGFmdGVyRWFjaOyXkOyEnCDtmLjstpztlanri4jri6QuXG4gICAqIO2FjOyKpO2KuCDsvIDsnbTsiqQg6rKw6rO866W8IHRyYWNlc+yZgCDtlajqu5gg7KCE7Iah7ZWp64uI64ukLlxuICAgKi9cbiAgcmVwb3J0VGVzdFJlc3VsdChyZXN1bHQ6IE9taXQ8TmFpdGVNZXNzYWdpbmdUeXBlcy5UZXN0UmVzdWx0LCBcInJlY2VpdmVkQXRcIj4pOiB2b2lkIHtcbiAgICBpZiAocHJvY2Vzcy5lbnYuQ0kpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICB0aGlzLnNlbmQoe1xuICAgICAgdHlwZTogXCJ0ZXN0L3Jlc3VsdFwiLFxuICAgICAgcmVjZWl2ZWRBdDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgLi4ucmVzdWx0LFxuICAgIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIGFmdGVyQWxs7JeQ7IScIO2YuOy2nO2VqeuLiOuLpC5cbiAgICog7YWM7Iqk7Yq4IHJ1biDsooXro4zrpbwg7JWM66a964uI64ukLlxuICAgKi9cbiAgZW5kVGVzdFJ1bigpOiB2b2lkIHtcbiAgICBpZiAocHJvY2Vzcy5lbnYuQ0kpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICB0aGlzLnNlbmQoe1xuICAgICAgdHlwZTogXCJydW4vZW5kXCIsXG4gICAgICBlbmRlZEF0OiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgfSk7XG5cbiAgICAvLyDsl7DqsrAg7KKF66OMXG4gICAgaWYgKHRoaXMuc29ja2V0KSB7XG4gICAgICB0aGlzLnNvY2tldC5lbmQoKTtcbiAgICAgIHRoaXMuc29ja2V0ID0gbnVsbDtcbiAgICAgIHRoaXMuY29ubmVjdGVkID0gZmFsc2U7XG4gICAgfVxuICB9XG59XG5cbmV4cG9ydCBjb25zdCBOYWl0ZVJlcG9ydGVyID0gbmV3IE5haXRlUmVwb3J0ZXJDbGFzcygpO1xuIl0sIm5hbWVzIjpbImNvbm5lY3QiLCJob21lZGlyIiwiam9pbiIsIlNPQ0tFVF9QQVRIIiwicHJvY2VzcyIsInBsYXRmb3JtIiwiTmFpdGVSZXBvcnRlckNsYXNzIiwic29ja2V0IiwiY29ubmVjdGVkIiwiYnVmZmVyIiwiZW5zdXJlQ29ubmVjdGlvbiIsIm9uIiwibXNnIiwid3JpdGUiLCJzZW5kIiwiZGF0YSIsIkpTT04iLCJzdHJpbmdpZnkiLCJwdXNoIiwic3RhcnRUZXN0UnVuIiwiZW52IiwiQ0kiLCJ0eXBlIiwic3RhcnRlZEF0IiwiRGF0ZSIsInRvSVNPU3RyaW5nIiwicmVwb3J0VGVzdFJlc3VsdCIsInJlc3VsdCIsInJlY2VpdmVkQXQi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
|