sonamu 0.6.0 → 0.7.1
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 +227 -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 +386 -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 +55 -30
- 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 +261 -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 +459 -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
|
@@ -1,57 +1,20 @@
|
|
|
1
|
-
import
|
|
2
|
-
import knex from "knex";
|
|
1
|
+
import assert from "assert";
|
|
3
2
|
import chalk from "chalk";
|
|
4
|
-
import { DateTime } from "luxon";
|
|
5
3
|
import { mkdir, readdir, unlink, writeFile } from "node:fs/promises";
|
|
6
|
-
import
|
|
7
|
-
import prompts from "prompts";
|
|
8
|
-
import { execSync } from "child_process";
|
|
4
|
+
import knex from "knex";
|
|
9
5
|
import path from "path";
|
|
10
|
-
import {
|
|
6
|
+
import { group, sum, unique } from "radashi";
|
|
11
7
|
import { Sonamu } from "../api/index.js";
|
|
8
|
+
import { DB } from "../database/db.js";
|
|
9
|
+
import { EntityManager } from "../entity/entity-manager.js";
|
|
12
10
|
import { ServiceUnavailableException } from "../exceptions/so-exceptions.js";
|
|
13
|
-
import {
|
|
14
|
-
import {
|
|
11
|
+
import { Naite } from "../naite/naite.js";
|
|
12
|
+
import { isTest } from "../utils/controller.js";
|
|
13
|
+
import { exists } from "../utils/fs-utils.js";
|
|
14
|
+
import { generateAlterCode, generateCreateCode } from "./code-generation.js";
|
|
15
15
|
import { getMigrationSetFromEntity } from "./migration-set.js";
|
|
16
|
+
import { PostgreSQLSchemaReader } from "./postgresql-schema-reader.js";
|
|
16
17
|
export class Migrator {
|
|
17
|
-
options;
|
|
18
|
-
targets;
|
|
19
|
-
constructor(options){
|
|
20
|
-
this.options = options;
|
|
21
|
-
const { dbConfig } = Sonamu;
|
|
22
|
-
if (this.options.mode === "dev") {
|
|
23
|
-
const devDB = knex(dbConfig.development_master);
|
|
24
|
-
const testDB = knex(dbConfig.test);
|
|
25
|
-
const fixtureLocalDB = knex(dbConfig.fixture_local);
|
|
26
|
-
const applyDBs = [
|
|
27
|
-
devDB,
|
|
28
|
-
testDB,
|
|
29
|
-
fixtureLocalDB
|
|
30
|
-
];
|
|
31
|
-
if (dbConfig.fixture_local.connection.host !== dbConfig.fixture_remote.connection.host || dbConfig.fixture_local.connection.database !== dbConfig.fixture_remote.connection.database) {
|
|
32
|
-
const fixtureRemoteDB = knex(dbConfig.fixture_remote);
|
|
33
|
-
applyDBs.push(fixtureRemoteDB);
|
|
34
|
-
}
|
|
35
|
-
this.targets = {
|
|
36
|
-
compare: devDB,
|
|
37
|
-
pending: devDB,
|
|
38
|
-
shadow: testDB,
|
|
39
|
-
apply: applyDBs
|
|
40
|
-
};
|
|
41
|
-
} else if (this.options.mode === "deploy") {
|
|
42
|
-
const productionDB = knex(dbConfig.production_master);
|
|
43
|
-
const testDB = knex(dbConfig.test);
|
|
44
|
-
this.targets = {
|
|
45
|
-
pending: productionDB,
|
|
46
|
-
shadow: testDB,
|
|
47
|
-
apply: [
|
|
48
|
-
productionDB
|
|
49
|
-
]
|
|
50
|
-
};
|
|
51
|
-
} else {
|
|
52
|
-
throw new Error(`잘못된 모드 ${this.options.mode} 입력`);
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
18
|
async getMigrationCodes() {
|
|
56
19
|
const srcMigrationsDir = path.join(Sonamu.apiRootPath, "src", "migrations"); // 이건 환경에 관계없이 항상 src에서 찾아야 해요.
|
|
57
20
|
if (!await exists(srcMigrationsDir)) {
|
|
@@ -63,6 +26,7 @@ export class Migrator {
|
|
|
63
26
|
name: f.replace(".ts", ""),
|
|
64
27
|
path: path.join(srcMigrationsDir, f)
|
|
65
28
|
})).sort((a, b)=>a.name < b.name ? 1 : -1); // 이름 내림차순 정렬(최신순)
|
|
29
|
+
Naite.t("migrator:getMigrationCodes:results", codes);
|
|
66
30
|
return codes;
|
|
67
31
|
}
|
|
68
32
|
/**
|
|
@@ -75,7 +39,9 @@ export class Migrator {
|
|
|
75
39
|
* @returns
|
|
76
40
|
*/ async getStatus() {
|
|
77
41
|
const codes = await this.getMigrationCodes();
|
|
42
|
+
Naite.t("migrator:getStatus:codes", codes);
|
|
78
43
|
const connKeys = Object.keys(Sonamu.dbConfig).filter((key)=>key.endsWith("_slave") === false);
|
|
44
|
+
let migrationStatusError;
|
|
79
45
|
const statuses = await Promise.all(connKeys.map(async (connKey)=>{
|
|
80
46
|
const knexOptions = Sonamu.dbConfig[connKey];
|
|
81
47
|
const tConn = knex(knexOptions);
|
|
@@ -84,7 +50,8 @@ export class Migrator {
|
|
|
84
50
|
return await tConn.migrate.status();
|
|
85
51
|
} catch (err) {
|
|
86
52
|
console.warn(chalk.yellow(`${connKey}의 마이그레이션 상태를 가져오는 데에 실패하였습니다. 데이터베이스가 올바르게 구성되지 않은 것 같습니다. 확인하시고 다시 시도해주세요.\n시도한 연결 설정:\n${JSON.stringify(knexOptions.connection, null, 2)}\n발생한 에러:\n${err}\n`));
|
|
87
|
-
|
|
53
|
+
migrationStatusError = err instanceof Error ? err.message : String(err);
|
|
54
|
+
return "error";
|
|
88
55
|
}
|
|
89
56
|
})();
|
|
90
57
|
const pending = await (async ()=>{
|
|
@@ -92,27 +59,31 @@ export class Migrator {
|
|
|
92
59
|
const [, fdList] = await tConn.migrate.list();
|
|
93
60
|
return fdList.map((fd)=>fd.file.replace(".ts", ""));
|
|
94
61
|
} catch (err) {
|
|
62
|
+
migrationStatusError = err instanceof Error ? err.message : String(err);
|
|
95
63
|
return [];
|
|
96
64
|
}
|
|
97
65
|
})();
|
|
98
66
|
const currentVersion = await (async ()=>{
|
|
99
67
|
try {
|
|
100
68
|
return await tConn.migrate.currentVersion();
|
|
101
|
-
} catch (
|
|
69
|
+
} catch (_err) {
|
|
70
|
+
migrationStatusError = _err instanceof Error ? _err.message : String(_err);
|
|
102
71
|
return "error";
|
|
103
72
|
}
|
|
104
73
|
})();
|
|
74
|
+
Naite.t("migrator:getStatus:status", status);
|
|
105
75
|
const connection = knexOptions.connection;
|
|
106
76
|
await tConn.destroy();
|
|
107
77
|
return {
|
|
108
78
|
name: connKey.replace("_master", ""),
|
|
109
79
|
connKey,
|
|
110
|
-
connString: `
|
|
80
|
+
connString: `pg://${connection.user ?? ""}@${connection.host}:${connection.port}/${connection.database}`,
|
|
111
81
|
currentVersion,
|
|
112
|
-
status,
|
|
82
|
+
status: status,
|
|
113
83
|
pending
|
|
114
84
|
};
|
|
115
85
|
}));
|
|
86
|
+
Naite.t("migrator:getStatus:conns", statuses);
|
|
116
87
|
const preparedCodes = await (async ()=>{
|
|
117
88
|
const status0conn = statuses.find((status)=>status.status === 0);
|
|
118
89
|
if (status0conn === undefined) {
|
|
@@ -124,20 +95,14 @@ export class Migrator {
|
|
|
124
95
|
await compareDBconn.destroy();
|
|
125
96
|
return genCodes;
|
|
126
97
|
})();
|
|
98
|
+
Naite.t("migrator:getStatus:preparedCodes", preparedCodes);
|
|
127
99
|
return {
|
|
128
100
|
conns: statuses,
|
|
129
101
|
codes,
|
|
130
|
-
preparedCodes
|
|
102
|
+
preparedCodes,
|
|
103
|
+
error: migrationStatusError
|
|
131
104
|
};
|
|
132
|
-
|
|
133
|
-
DB 마이그레이션 상태 확인
|
|
134
|
-
1. 전체 DB설정에 대해서 현재 마이그레이션 상태 확인
|
|
135
|
-
- connKey: string
|
|
136
|
-
- status: number
|
|
137
|
-
- currentVersion: string
|
|
138
|
-
- list: { file: string; directory: string }[]
|
|
139
|
-
|
|
140
|
-
*/ }
|
|
105
|
+
}
|
|
141
106
|
/**
|
|
142
107
|
* 마이그레이션을 적용하거나 롤백합니다.
|
|
143
108
|
* Sonamu UI에서 마이그레이션 작업을 수행할 때 사용됩니다.
|
|
@@ -148,11 +113,13 @@ export class Migrator {
|
|
|
148
113
|
* @param targets 작업 대상 DB 설정 키 (keyof SonamuDBConfig)
|
|
149
114
|
* @returns 작업 결과
|
|
150
115
|
*/ async runAction(action, targets) {
|
|
116
|
+
Naite.t("migrator:runAction:action", action);
|
|
117
|
+
Naite.t("migrator:runAction:targets", targets);
|
|
151
118
|
// get uniq knex configs
|
|
152
|
-
const configs =
|
|
119
|
+
const configs = unique(targets.map((target)=>({
|
|
153
120
|
connKey: target,
|
|
154
121
|
options: Sonamu.dbConfig[target]
|
|
155
|
-
})).filter((c)=>c.options !== undefined), ({ options })=>`${options.connection.host}:${options.connection.port ??
|
|
122
|
+
})).filter((c)=>c.options !== undefined), ({ options })=>`${options.connection.host}:${options.connection.port ?? 5432}/${options.connection.database}`);
|
|
156
123
|
// get connections
|
|
157
124
|
const conns = await Promise.all(configs.map(async (config)=>({
|
|
158
125
|
connKey: config.connKey,
|
|
@@ -185,9 +152,23 @@ export class Migrator {
|
|
|
185
152
|
await Promise.all(conns.map(({ knex })=>{
|
|
186
153
|
return knex.destroy();
|
|
187
154
|
}));
|
|
155
|
+
Naite.t("migrator:runAction:result", result);
|
|
188
156
|
return result;
|
|
189
157
|
}
|
|
190
158
|
/**
|
|
159
|
+
* 삭제 가능한 마이그레이션 코드 파일을 검증합니다.
|
|
160
|
+
*
|
|
161
|
+
* @param conns 마이그레이션 상태 배열
|
|
162
|
+
* @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열
|
|
163
|
+
* @returns 삭제 가능 여부 및 적용된 마이그레이션 코드 파일 이름
|
|
164
|
+
*/ validateDeletable(conns, codeNames) {
|
|
165
|
+
const appliedCodes = codeNames.filter((codeName)=>conns.some((conn)=>conn.pending.includes(codeName) === false));
|
|
166
|
+
return {
|
|
167
|
+
canDelete: appliedCodes.length === 0,
|
|
168
|
+
appliedCodes
|
|
169
|
+
};
|
|
170
|
+
}
|
|
171
|
+
/**
|
|
191
172
|
* 마이그레이션 코드 파일을 삭제합니다.
|
|
192
173
|
*
|
|
193
174
|
* Sonamu UI에서 사용됩니다.
|
|
@@ -196,21 +177,23 @@ export class Migrator {
|
|
|
196
177
|
* @returns 삭제된 마이그레이션 코드 파일 개수
|
|
197
178
|
*/ async delCodes(codeNames) {
|
|
198
179
|
const { conns } = await this.getStatus();
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
throw new Error("You cannot delete a migration file if there is already applied.");
|
|
180
|
+
const { canDelete, appliedCodes } = this.validateDeletable(conns, codeNames);
|
|
181
|
+
if (!canDelete) {
|
|
182
|
+
throw new Error(`You cannot delete a migration file if there is already applied. Applied codes: ${appliedCodes.join(", ")}`);
|
|
203
183
|
}
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
if (await exists(
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
return delFiles.includes(".ts") ? 1 : 0;
|
|
184
|
+
return sum(await Promise.all(codeNames.map(async (codeName)=>{
|
|
185
|
+
const filePath = `${Sonamu.apiRootPath}/src/migrations/${codeName}.ts`;
|
|
186
|
+
if (await exists(filePath)) {
|
|
187
|
+
await unlink(filePath);
|
|
188
|
+
return 1;
|
|
210
189
|
}
|
|
211
190
|
return 0;
|
|
212
|
-
}));
|
|
213
|
-
|
|
191
|
+
})));
|
|
192
|
+
}
|
|
193
|
+
genDateTag(index, baseDate = new Date()) {
|
|
194
|
+
const date = new Date(baseDate.getTime() + index * 1000);
|
|
195
|
+
const pad = (num, size = 2)=>num.toString().padStart(size, "0");
|
|
196
|
+
return date.getFullYear().toString() + pad(date.getMonth() + 1) + pad(date.getDate()) + pad(date.getHours()) + pad(date.getMinutes()) + pad(date.getSeconds());
|
|
214
197
|
}
|
|
215
198
|
/**
|
|
216
199
|
* 마이그레이션 코드 파일을 생성합니다.
|
|
@@ -220,6 +203,7 @@ export class Migrator {
|
|
|
220
203
|
* @returns 생성된 마이그레이션 코드 파일 개수
|
|
221
204
|
*/ async generatePreparedCodes() {
|
|
222
205
|
const { preparedCodes } = await this.getStatus();
|
|
206
|
+
Naite.t("migrator:generatePreparedCodes:preparedCodes", preparedCodes);
|
|
223
207
|
if (preparedCodes.length === 0) {
|
|
224
208
|
console.log(chalk.green("\n현재 모두 싱크된 상태입니다."));
|
|
225
209
|
return 0;
|
|
@@ -228,240 +212,30 @@ export class Migrator {
|
|
|
228
212
|
const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;
|
|
229
213
|
for (const [index, pcode] of preparedCodes.entries()){
|
|
230
214
|
if (pcode.formatted) {
|
|
231
|
-
const dateTag =
|
|
232
|
-
seconds: index
|
|
233
|
-
}).toFormat("yyyyMMddHHmmss");
|
|
215
|
+
const dateTag = this.genDateTag(index);
|
|
234
216
|
const filePath = `${migrationsDir}/${dateTag}_${pcode.title}.ts`;
|
|
235
217
|
await writeFile(filePath, pcode.formatted);
|
|
236
|
-
console.log(chalk.green(`MIGRTAION CREATED ${filePath}`));
|
|
218
|
+
!isTest() && console.log(chalk.green(`MIGRTAION CREATED ${filePath}`));
|
|
237
219
|
}
|
|
238
220
|
}
|
|
239
221
|
return preparedCodes.length;
|
|
240
222
|
}
|
|
241
|
-
/**
|
|
242
|
-
* pending 마이그레이션 목록을 삭제합니다.
|
|
243
|
-
*
|
|
244
|
-
* CLI에서 사용됩니다.
|
|
245
|
-
*/ async clearPendingList() {
|
|
246
|
-
const [, pendingList] = await this.targets.pending.migrate.list();
|
|
247
|
-
const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;
|
|
248
|
-
const delList = pendingList.map((df)=>{
|
|
249
|
-
return path.join(migrationsDir, df.file);
|
|
250
|
-
});
|
|
251
|
-
for (let p of delList){
|
|
252
|
-
if (await exists(p)) {
|
|
253
|
-
await unlink(p);
|
|
254
|
-
}
|
|
255
|
-
}
|
|
256
|
-
}
|
|
257
|
-
/**
|
|
258
|
-
* 마이그레이션 코드 파일을 확인합니다.
|
|
259
|
-
*
|
|
260
|
-
* CLI에서 사용됩니다.
|
|
261
|
-
*/ async check() {
|
|
262
|
-
const codes = await this.compareMigrations(this.targets.compare);
|
|
263
|
-
if (codes.length === 0) {
|
|
264
|
-
console.log(chalk.green("\n현재 모두 싱크된 상태입니다."));
|
|
265
|
-
return;
|
|
266
|
-
}
|
|
267
|
-
// 현재 생성된 코드 표기
|
|
268
|
-
console.table(codes, [
|
|
269
|
-
"type",
|
|
270
|
-
"title"
|
|
271
|
-
]);
|
|
272
|
-
console.log(codes[0]);
|
|
273
|
-
}
|
|
274
|
-
/**
|
|
275
|
-
* 마이그레이션을 수행합니다.
|
|
276
|
-
*
|
|
277
|
-
* runAction이 인자로 들어온 타겟들에 대해 주어진 동작(apply/rollback)을 수행한다면,
|
|
278
|
-
* 이 함수는 생성자로 들어온 connection(knex)들에 대해 마이그레이션을 수행합니다.
|
|
279
|
-
*
|
|
280
|
-
* CLI에서 사용됩니다.
|
|
281
|
-
*/ async run() {
|
|
282
|
-
// pending 마이그레이션 확인
|
|
283
|
-
const [, pendingList] = await this.targets.pending.migrate.list();
|
|
284
|
-
if (pendingList.length > 0) {
|
|
285
|
-
console.log(chalk.red("pending 된 마이그레이션이 존재합니다."), pendingList.map((pending)=>pending.file));
|
|
286
|
-
// pending이 있는 경우 Shadow DB 테스트 진행 여부 컨펌
|
|
287
|
-
const answer = await prompts({
|
|
288
|
-
type: "confirm",
|
|
289
|
-
name: "value",
|
|
290
|
-
message: "Shadow DB 테스트를 진행하시겠습니까?",
|
|
291
|
-
initial: true
|
|
292
|
-
});
|
|
293
|
-
if (answer.value === false) {
|
|
294
|
-
return;
|
|
295
|
-
}
|
|
296
|
-
console.time(chalk.blue("Migrator - runShadowTest"));
|
|
297
|
-
await this.runShadowTest();
|
|
298
|
-
console.timeEnd(chalk.blue("Migrator - runShadowTest"));
|
|
299
|
-
await Promise.all(this.targets.apply.map(async (applyDb)=>{
|
|
300
|
-
const label = chalk.green(`APPLIED ${applyDb.client.connectionSettings.host} ${applyDb.client.database()}`);
|
|
301
|
-
console.time(label);
|
|
302
|
-
const [, ] = await applyDb.migrate.latest();
|
|
303
|
-
console.timeEnd(label);
|
|
304
|
-
}));
|
|
305
|
-
}
|
|
306
|
-
// Entity-DB간 비교하여 코드 생성 리턴
|
|
307
|
-
const codes = await this.compareMigrations(this.targets.compare);
|
|
308
|
-
if (codes.length === 0) {
|
|
309
|
-
console.log(chalk.green("\n현재 모두 싱크된 상태입니다."));
|
|
310
|
-
return;
|
|
311
|
-
}
|
|
312
|
-
// 현재 생성된 코드 표기
|
|
313
|
-
console.table(codes, [
|
|
314
|
-
"type",
|
|
315
|
-
"title"
|
|
316
|
-
]);
|
|
317
|
-
/* DEBUG: 디버깅용 코드
|
|
318
|
-
codes.map((code) => console.log(code.formatted));
|
|
319
|
-
process.exit();
|
|
320
|
-
*/ // 실제 파일 생성 프롬프트
|
|
321
|
-
const answer = await prompts({
|
|
322
|
-
type: "confirm",
|
|
323
|
-
name: "value",
|
|
324
|
-
message: "마이그레이션 코드를 생성하시겠습니까?",
|
|
325
|
-
initial: false
|
|
326
|
-
});
|
|
327
|
-
if (answer.value === false) {
|
|
328
|
-
return;
|
|
329
|
-
}
|
|
330
|
-
// 실제 코드 생성
|
|
331
|
-
const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;
|
|
332
|
-
for (const [index, code] of codes.entries()){
|
|
333
|
-
if (code.formatted) {
|
|
334
|
-
const dateTag = DateTime.local().plus({
|
|
335
|
-
seconds: index
|
|
336
|
-
}).toFormat("yyyyMMddHHmmss");
|
|
337
|
-
const filePath = `${migrationsDir}/${dateTag}_${code.title}.ts`;
|
|
338
|
-
await writeFile(filePath, code.formatted);
|
|
339
|
-
console.log(chalk.green(`MIGRTAION CREATED ${filePath}`));
|
|
340
|
-
}
|
|
341
|
-
}
|
|
342
|
-
}
|
|
343
|
-
/**
|
|
344
|
-
* 타겟으로 지정된 DB를 롤백합니다.
|
|
345
|
-
*
|
|
346
|
-
* runAction이 인자로 들어온 타겟들에 대해 주어진 동작(apply/rollback)을 수행한다면,
|
|
347
|
-
* 이 함수는 생성자로 들어온 connection(knex)들에 대해 롤백을 수행합니다.
|
|
348
|
-
*
|
|
349
|
-
* CLI에서 사용됩니다.
|
|
350
|
-
*/ async rollback() {
|
|
351
|
-
console.time(chalk.red("rollback:"));
|
|
352
|
-
const rollbackAllResult = await Promise.all(this.targets.apply.map(async (db)=>{
|
|
353
|
-
await db.migrate.forceFreeMigrationsLock();
|
|
354
|
-
return db.migrate.rollback(undefined, false);
|
|
355
|
-
}));
|
|
356
|
-
console.dir({
|
|
357
|
-
rollbackAllResult
|
|
358
|
-
}, {
|
|
359
|
-
depth: null
|
|
360
|
-
});
|
|
361
|
-
console.timeEnd(chalk.red("rollback:"));
|
|
362
|
-
}
|
|
363
|
-
/**
|
|
364
|
-
* Shadow DB 테스트를 진행합니다.
|
|
365
|
-
*
|
|
366
|
-
* Sonamu UI에서 사용됩니다.
|
|
367
|
-
*
|
|
368
|
-
* @returns Shadow DB 테스트 결과
|
|
369
|
-
*/ async runShadowTest() {
|
|
370
|
-
// ShadowDB 생성 후 테스트 진행
|
|
371
|
-
const tdb = knex(Sonamu.dbConfig.test);
|
|
372
|
-
const tdbConn = Sonamu.dbConfig.test.connection;
|
|
373
|
-
const shadowDatabase = tdbConn.database + "__migration_shadow";
|
|
374
|
-
const tmpSqlPath = `/tmp/${shadowDatabase}.sql`;
|
|
375
|
-
// 테스트DB 덤프 후 Database명 치환
|
|
376
|
-
console.log(chalk.magenta(`${tdbConn.database}의 데이터 ${tmpSqlPath}로 덤프`));
|
|
377
|
-
execSync(`mysqldump -h${tdbConn.host} -P${tdbConn.port ?? 3306} -u${tdbConn.user} -p'${tdbConn.password}' ${tdbConn.database} --single-transaction --no-create-db --triggers > ${tmpSqlPath};`);
|
|
378
|
-
execSync(`sed -i'' -e 's/\`${tdbConn.database}\`/\`${shadowDatabase}\`/g' ${tmpSqlPath};`);
|
|
379
|
-
// 기존 ShadowDB 리셋
|
|
380
|
-
console.log(chalk.magenta(`${shadowDatabase} 리셋`));
|
|
381
|
-
await tdb.raw(`DROP DATABASE IF EXISTS \`${shadowDatabase}\`;`);
|
|
382
|
-
await tdb.raw(`CREATE DATABASE \`${shadowDatabase}\`;`);
|
|
383
|
-
// ShadowDB 테이블 + 데이터 생성
|
|
384
|
-
console.log(chalk.magenta(`${shadowDatabase} 데이터베이스 생성`));
|
|
385
|
-
execSync(`mysql -h${tdbConn.host} -P${tdbConn.port ?? 3306} -u${tdbConn.user} -p'${tdbConn.password}' ${shadowDatabase} < ${tmpSqlPath};`);
|
|
386
|
-
// shadow db 테스트 진행
|
|
387
|
-
const sdb = knex({
|
|
388
|
-
...Sonamu.dbConfig.test,
|
|
389
|
-
connection: {
|
|
390
|
-
...tdbConn,
|
|
391
|
-
database: shadowDatabase,
|
|
392
|
-
password: tdbConn.password
|
|
393
|
-
}
|
|
394
|
-
});
|
|
395
|
-
// shadow db 테스트 진행
|
|
396
|
-
try {
|
|
397
|
-
const [batchNo, applied] = await sdb.migrate.latest();
|
|
398
|
-
console.log(chalk.green("Shadow DB 테스트에 성공했습니다!"), {
|
|
399
|
-
batchNo,
|
|
400
|
-
applied
|
|
401
|
-
});
|
|
402
|
-
// 생성한 Shadow DB 삭제
|
|
403
|
-
console.log(chalk.magenta(`${shadowDatabase} 삭제`));
|
|
404
|
-
await tdb.raw(`DROP DATABASE IF EXISTS \`${shadowDatabase}\`;`);
|
|
405
|
-
return [
|
|
406
|
-
{
|
|
407
|
-
connKey: "shadow",
|
|
408
|
-
batchNo,
|
|
409
|
-
applied
|
|
410
|
-
}
|
|
411
|
-
];
|
|
412
|
-
} catch (e) {
|
|
413
|
-
console.error(e);
|
|
414
|
-
throw new ServiceUnavailableException("Shadow DB 테스트 진행 중 에러");
|
|
415
|
-
} finally{
|
|
416
|
-
await tdb.destroy();
|
|
417
|
-
}
|
|
418
|
-
}
|
|
419
|
-
/**
|
|
420
|
-
* 모든 DB를 롤백하고 전체 마이그레이션 파일을 삭제합니다.
|
|
421
|
-
*
|
|
422
|
-
* CLI에서 사용됩니다.
|
|
423
|
-
*
|
|
424
|
-
* @returns
|
|
425
|
-
*/ async resetAll() {
|
|
426
|
-
const answer = await prompts({
|
|
427
|
-
type: "confirm",
|
|
428
|
-
name: "value",
|
|
429
|
-
message: "모든 DB를 롤백하고 전체 마이그레이션 파일을 삭제하시겠습니까?",
|
|
430
|
-
initial: false
|
|
431
|
-
});
|
|
432
|
-
if (answer.value === false) {
|
|
433
|
-
return;
|
|
434
|
-
}
|
|
435
|
-
console.time(chalk.red("rollback-all:"));
|
|
436
|
-
const rollbackAllResult = await Promise.all(this.targets.apply.map(async (db)=>{
|
|
437
|
-
await db.migrate.forceFreeMigrationsLock();
|
|
438
|
-
return db.migrate.rollback(undefined, true);
|
|
439
|
-
}));
|
|
440
|
-
console.log({
|
|
441
|
-
rollbackAllResult
|
|
442
|
-
});
|
|
443
|
-
console.timeEnd(chalk.red("rollback-all:"));
|
|
444
|
-
const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;
|
|
445
|
-
console.time(chalk.red("delete migration files"));
|
|
446
|
-
execSync(`rm -f ${migrationsDir}/*`);
|
|
447
|
-
execSync(`rm -f ${migrationsDir.replace("/src/", "/dist/")}/*`);
|
|
448
|
-
console.timeEnd(chalk.red("delete migration files"));
|
|
449
|
-
}
|
|
450
223
|
async compareMigrations(compareDB) {
|
|
451
224
|
// Entity 순회하여 싱크
|
|
452
225
|
const entityIds = EntityManager.getAllIds();
|
|
453
226
|
// 조인테이블 포함하여 Entity에서 MigrationSet 추출
|
|
454
227
|
const entitySetsWithJoinTable = entityIds.filter((entityId)=>EntityManager.get(entityId).props.length > 0).map((entityId)=>getMigrationSetFromEntity(EntityManager.get(entityId)));
|
|
455
228
|
// 조인테이블만 추출
|
|
456
|
-
const joinTablesWithDup = entitySetsWithJoinTable.
|
|
229
|
+
const joinTablesWithDup = entitySetsWithJoinTable.flatMap((entitySet)=>entitySet.joinTables);
|
|
457
230
|
// 중복 제거 (중복인 경우 indexes를 병합)
|
|
458
|
-
const joinTables = Object.values(
|
|
231
|
+
const joinTables = Object.values(group(joinTablesWithDup, (jt)=>jt.table)).map((tables)=>{
|
|
232
|
+
assert(tables !== undefined, "tables is undefined");
|
|
459
233
|
if (tables.length === 1) {
|
|
460
234
|
return tables[0];
|
|
461
235
|
}
|
|
462
236
|
return {
|
|
463
237
|
...tables[0],
|
|
464
|
-
indexes:
|
|
238
|
+
indexes: unique(tables.flatMap((t)=>t.indexes), (index)=>[
|
|
465
239
|
index.type,
|
|
466
240
|
...index.columns.sort()
|
|
467
241
|
].join("-"))
|
|
@@ -473,7 +247,9 @@ export class Migrator {
|
|
|
473
247
|
...joinTables
|
|
474
248
|
];
|
|
475
249
|
const codes = (await Promise.all(entitySets.map(async (entitySet)=>{
|
|
476
|
-
const dbSet = await getMigrationSetFromDB(compareDB, entitySet.table);
|
|
250
|
+
const dbSet = await PostgreSQLSchemaReader.getMigrationSetFromDB(compareDB, entitySet.table);
|
|
251
|
+
Naite.t(`migrator:compareMigrations:entitySet:${entitySet.table}`, entitySet);
|
|
252
|
+
Naite.t(`migrator:compareMigrations:dbSet:${entitySet.table}`, dbSet);
|
|
477
253
|
if (dbSet === null) {
|
|
478
254
|
// 기존 테이블 없음, 새로 테이블 생성
|
|
479
255
|
return await generateCreateCode(entitySet);
|
|
@@ -484,7 +260,7 @@ export class Migrator {
|
|
|
484
260
|
}))).flat();
|
|
485
261
|
// normal 타입이 앞으로, foreign이 뒤로
|
|
486
262
|
codes.sort((codeA, codeB)=>{
|
|
487
|
-
if (codeA.type === "foreign" && codeB.type
|
|
263
|
+
if (codeA.type === "foreign" && codeB.type === "normal") {
|
|
488
264
|
return 1;
|
|
489
265
|
} else if (codeA.type === "normal" && codeB.type === "foreign") {
|
|
490
266
|
return -1;
|
|
@@ -495,16 +271,60 @@ export class Migrator {
|
|
|
495
271
|
return codes;
|
|
496
272
|
}
|
|
497
273
|
/**
|
|
498
|
-
*
|
|
274
|
+
* Shadow DB 테스트를 진행합니다.
|
|
499
275
|
*
|
|
500
|
-
*
|
|
276
|
+
* Sonamu UI에서 사용됩니다.
|
|
501
277
|
*
|
|
502
|
-
* @returns
|
|
503
|
-
*/ async
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
|
|
278
|
+
* @returns Shadow DB 테스트 결과
|
|
279
|
+
*/ async runShadowTest() {
|
|
280
|
+
const tdbConn = Sonamu.dbConfig.test.connection;
|
|
281
|
+
const shadowDatabase = `${tdbConn.database}__migration_shadow`;
|
|
282
|
+
// 테스트 상황에서는 트랜잭션을 초기화하고, 새 데이터베이스 커넥션을 가져와야 함
|
|
283
|
+
if (isTest()) {
|
|
284
|
+
await DB.clearTestTransaction();
|
|
285
|
+
await DB.destroy();
|
|
286
|
+
}
|
|
287
|
+
// 기존 Shadow DB 삭제 후 Shadow DB 생성
|
|
288
|
+
const tdb = knex(Sonamu.dbConfig.test);
|
|
289
|
+
!isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));
|
|
290
|
+
await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);
|
|
291
|
+
await tdb.raw(`CREATE DATABASE ${shadowDatabase} TEMPLATE ${tdbConn.database}`);
|
|
292
|
+
// Shadow DB에 연결
|
|
293
|
+
const sdb = knex({
|
|
294
|
+
...Sonamu.dbConfig.test,
|
|
295
|
+
connection: {
|
|
296
|
+
...tdbConn,
|
|
297
|
+
database: shadowDatabase,
|
|
298
|
+
password: tdbConn.password
|
|
299
|
+
}
|
|
300
|
+
});
|
|
301
|
+
// shadow DB 테스트 진행
|
|
302
|
+
try {
|
|
303
|
+
const [batchNo, applied] = await sdb.migrate.latest();
|
|
304
|
+
!isTest() && console.log(chalk.green("Shadow DB 테스트에 성공했습니다!"), {
|
|
305
|
+
batchNo,
|
|
306
|
+
applied
|
|
307
|
+
});
|
|
308
|
+
return [
|
|
309
|
+
{
|
|
310
|
+
connKey: "shadow",
|
|
311
|
+
batchNo,
|
|
312
|
+
applied
|
|
313
|
+
}
|
|
314
|
+
];
|
|
315
|
+
} catch (e) {
|
|
316
|
+
console.error(e);
|
|
317
|
+
throw new ServiceUnavailableException("Shadow DB 테스트 진행 중 에러");
|
|
318
|
+
} finally{
|
|
319
|
+
// Shadow DB 연결 종료
|
|
320
|
+
await sdb.destroy();
|
|
321
|
+
// Shadow DB 삭제
|
|
322
|
+
!isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));
|
|
323
|
+
await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);
|
|
324
|
+
// Test DB 연결 종료
|
|
325
|
+
await tdb.destroy();
|
|
326
|
+
}
|
|
507
327
|
}
|
|
508
328
|
}
|
|
509
329
|
|
|
510
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/migration/migrator.ts"],"sourcesContent":["import * as _ from \"lodash-es\";\nimport knex, { Knex } from \"knex\";\nimport chalk from \"chalk\";\nimport { DateTime } from \"luxon\";\nimport { mkdir, readdir, unlink, writeFile } from \"fs/promises\";\nimport { exists } from \"../utils/fs-utils\";\nimport prompts from \"prompts\";\nimport { execSync } from \"child_process\";\nimport path from \"path\";\nimport { GenMigrationCode, MigrationSet } from \"../types/types\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { Sonamu } from \"../api\";\nimport { ServiceUnavailableException } from \"../exceptions/so-exceptions\";\nimport { SonamuDBConfig } from \"../database/db\";\nimport { generateCreateCode, generateAlterCode } from \"./code-generation\";\nimport { MigrationStatus, MigrationCode, ConnString } from \"./types\";\nimport { getMigrationSetFromDB } from \"./migration-set\";\nimport { getMigrationSetFromEntity } from \"./migration-set\";\n\ntype MigratorMode = \"dev\" | \"deploy\";\nexport type MigratorOptions = {\n  readonly mode: MigratorMode;\n};\n\nexport class Migrator {\n  targets: {\n    compare?: Knex;\n    pending: Knex;\n    shadow: Knex;\n    apply: Knex[];\n  };\n\n  constructor(private readonly options: MigratorOptions) {\n    const { dbConfig } = Sonamu;\n\n    if (this.options.mode === \"dev\") {\n      const devDB = knex(dbConfig.development_master);\n      const testDB = knex(dbConfig.test);\n      const fixtureLocalDB = knex(dbConfig.fixture_local);\n\n      const applyDBs = [devDB, testDB, fixtureLocalDB];\n      if (\n        (dbConfig.fixture_local.connection as Knex.MySql2ConnectionConfig)\n          .host !==\n          (dbConfig.fixture_remote.connection as Knex.MySql2ConnectionConfig)\n            .host ||\n        (dbConfig.fixture_local.connection as Knex.MySql2ConnectionConfig)\n          .database !==\n          (dbConfig.fixture_remote.connection as Knex.MySql2ConnectionConfig)\n            .database\n      ) {\n        const fixtureRemoteDB = knex(dbConfig.fixture_remote);\n        applyDBs.push(fixtureRemoteDB);\n      }\n\n      this.targets = {\n        compare: devDB,\n        pending: devDB,\n        shadow: testDB,\n        apply: applyDBs,\n      };\n    } else if (this.options.mode === \"deploy\") {\n      const productionDB = knex(dbConfig.production_master);\n      const testDB = knex(dbConfig.test);\n\n      this.targets = {\n        pending: productionDB,\n        shadow: testDB,\n        apply: [productionDB],\n      };\n    } else {\n      throw new Error(`잘못된 모드 ${this.options.mode} 입력`);\n    }\n  }\n\n  private async getMigrationCodes(): Promise<MigrationCode[]> {\n    const srcMigrationsDir = path.join(Sonamu.apiRootPath, \"src\", \"migrations\"); // 이건 환경에 관계없이 항상 src에서 찾아야 해요.\n\n    if (!(await exists(srcMigrationsDir))) {\n      await mkdir(srcMigrationsDir, {\n        recursive: true,\n      });\n    }\n\n    const codes = (await readdir(srcMigrationsDir))\n      .filter((f) => f.endsWith(\".ts\"))\n      .map((f) => ({\n        name: f.replace(\".ts\", \"\"),\n        path: path.join(srcMigrationsDir, f),\n      }))\n      .sort((a, b) => (a.name < b.name ? 1 : -1)); // 이름 내림차순 정렬(최신순)\n\n    return codes;\n  }\n\n  /**\n   * 타겟별 마이그레이션 상태와 코드 생성/준비 상태를 구해옵니다.\n   * 실제로 DB에 접근도 하고 마이그레이션 코드 파일도 확인하고,\n   * 필요하다면 적용할 수 있는 코드를 생성까지 해옵니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @returns\n   */\n  async getStatus(): Promise<MigrationStatus> {\n    const codes = await this.getMigrationCodes();\n\n    const connKeys = Object.keys(Sonamu.dbConfig).filter(\n      (key) => key.endsWith(\"_slave\") === false\n    ) as (keyof typeof Sonamu.dbConfig)[];\n\n    const statuses = await Promise.all(\n      connKeys.map(async (connKey) => {\n        const knexOptions = Sonamu.dbConfig[connKey];\n        const tConn = knex(knexOptions);\n\n        const status = await (async () => {\n          try {\n            return await tConn.migrate.status();\n          } catch (err) {\n            console.warn(\n              chalk.yellow(\n                `${connKey}의 마이그레이션 상태를 가져오는 데에 실패하였습니다. 데이터베이스가 올바르게 구성되지 않은 것 같습니다. 확인하시고 다시 시도해주세요.\\n시도한 연결 설정:\\n${JSON.stringify(knexOptions.connection, null, 2)}\\n발생한 에러:\\n${err}\\n`\n              )\n            );\n            return \"error\" /*클라이언트에서 에러 체크에 사용하는 리터럴입니다.*/;\n          }\n        })();\n        const pending = await (async () => {\n          try {\n            const [, fdList] = await tConn.migrate.list();\n            return fdList.map((fd: { file: string }) =>\n              fd.file.replace(\".ts\", \"\")\n            );\n          } catch (err) {\n            return [];\n          }\n        })();\n        const currentVersion = await (async () => {\n          try {\n            return await tConn.migrate.currentVersion();\n          } catch (err) {\n            return \"error\";\n          }\n        })();\n\n        const connection =\n          knexOptions.connection as Knex.MySql2ConnectionConfig;\n\n        await tConn.destroy();\n\n        return {\n          name: connKey.replace(\"_master\", \"\"),\n          connKey,\n          connString: `mysql2://${connection.user ?? \"\"}@${connection.host}:${\n            connection.port\n          }/${connection.database}` as ConnString,\n          currentVersion,\n          status,\n          pending,\n        };\n      })\n    );\n\n    const preparedCodes: GenMigrationCode[] = await (async () => {\n      const status0conn = statuses.find((status) => status.status === 0);\n      if (status0conn === undefined) {\n        console.warn(\n          chalk.yellow(\n            `While trying to prepare migration codes, we found that there is no database to compare migrations. We need at least one database where every migration is applied(status === 0). You might want to apply your existing migrations to one of the databases.`\n          )\n        );\n        return [];\n      }\n\n      const compareDBconn = knex(Sonamu.dbConfig[status0conn.connKey]);\n      const genCodes = await this.compareMigrations(compareDBconn);\n\n      await compareDBconn.destroy();\n\n      return genCodes;\n    })();\n\n    return {\n      conns: statuses,\n      codes,\n      preparedCodes,\n    };\n    /*\n    DB 마이그레이션 상태 확인\n    1. 전체 DB설정에 대해서 현재 마이그레이션 상태 확인\n    - connKey: string\n    - status: number\n    - currentVersion: string\n    - list: { file: string; directory: string }[]\n    \n    */\n  }\n\n  /**\n   * 마이그레이션을 적용하거나 롤백합니다.\n   * Sonamu UI에서 마이그레이션 작업을 수행할 때 사용됩니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @param action 작업 유형 (apply/rollback)\n   * @param targets 작업 대상 DB 설정 키 (keyof SonamuDBConfig)\n   * @returns 작업 결과\n   */\n  async runAction(\n    action: \"apply\" | \"rollback\",\n    targets: (keyof SonamuDBConfig)[]\n  ): Promise<\n    {\n      connKey: string;\n      batchNo: number;\n      applied: string[];\n    }[]\n  > {\n    // get uniq knex configs\n    const configs = _.uniqBy(\n      targets\n        .map((target) => ({\n          connKey: target,\n          options: Sonamu.dbConfig[target as keyof typeof Sonamu.dbConfig],\n        }))\n        .filter((c) => c.options !== undefined),\n      ({ options }) =>\n        `${(options.connection as Knex.MySql2ConnectionConfig).host}:${\n          (options.connection as Knex.MySql2ConnectionConfig).port ?? 3306\n        }/${(options.connection as Knex.MySql2ConnectionConfig).database}`\n    );\n\n    // get connections\n    const conns = await Promise.all(\n      configs.map(async (config) => ({\n        connKey: config.connKey,\n        knex: knex(config.options),\n      }))\n    );\n\n    // action\n    const result = await (async () => {\n      switch (action) {\n        case \"apply\":\n          return Promise.all(\n            conns.map(async ({ connKey, knex }) => {\n              const [batchNo, applied] = await knex.migrate.latest();\n              return {\n                connKey,\n                batchNo,\n                applied,\n              };\n            })\n          );\n        case \"rollback\":\n          return Promise.all(\n            conns.map(async ({ connKey, knex }) => {\n              const [batchNo, applied] = await knex.migrate.rollback();\n              return {\n                connKey,\n                batchNo,\n                applied,\n              };\n            })\n          );\n      }\n    })();\n\n    // destroy\n    await Promise.all(\n      conns.map(({ knex }) => {\n        return knex.destroy();\n      })\n    );\n\n    return result;\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 삭제합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제된 마이그레이션 코드 파일 개수\n   */\n  async delCodes(codeNames: string[]): Promise<number> {\n    const { conns } = await this.getStatus();\n    if (\n      conns.some((conn) => {\n        return codeNames.some(\n          (codeName) => conn.pending.includes(codeName) === false\n        );\n      })\n    ) {\n      throw new Error(\n        \"You cannot delete a migration file if there is already applied.\"\n      );\n    }\n\n    const delFiles = codeNames.map(\n      (codeName) => `${Sonamu.apiRootPath}/src/migrations/${codeName}.ts`\n    );\n\n    const res = await Promise.all(\n      delFiles.map(async (delFile) => {\n        if (await exists(delFile)) {\n          console.log(chalk.red(`DELETE: ${delFile}`));\n          await unlink(delFile);\n          return delFiles.includes(\".ts\") ? 1 : 0;\n        }\n        return 0;\n      })\n    );\n    return _.sum(res);\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 생성합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns 생성된 마이그레이션 코드 파일 개수\n   */\n  async generatePreparedCodes(): Promise<number> {\n    const { preparedCodes } = await this.getStatus();\n    if (preparedCodes.length === 0) {\n      console.log(chalk.green(\"\\n현재 모두 싱크된 상태입니다.\"));\n      return 0;\n    }\n\n    // 실제 코드 생성\n    const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;\n\n    for (const [index, pcode] of preparedCodes.entries()) {\n      if (pcode.formatted) {\n        const dateTag = DateTime.local()\n          .plus({ seconds: index })\n          .toFormat(\"yyyyMMddHHmmss\");\n        const filePath = `${migrationsDir}/${dateTag}_${pcode.title}.ts`;\n        await writeFile(filePath, pcode.formatted!);\n        console.log(chalk.green(`MIGRTAION CREATED ${filePath}`));\n      }\n    }\n\n    return preparedCodes.length;\n  }\n\n  /**\n   * pending 마이그레이션 목록을 삭제합니다.\n   *\n   * CLI에서 사용됩니다.\n   */\n  async clearPendingList(): Promise<void> {\n    const [, pendingList] = (await this.targets.pending.migrate.list()) as [\n      unknown,\n      {\n        file: string;\n        directory: string;\n      }[],\n    ];\n    const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;\n    const delList = pendingList.map((df) => {\n      return path.join(migrationsDir, df.file);\n    });\n    for (let p of delList) {\n      if (await exists(p)) {\n        await unlink(p);\n      }\n    }\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 확인합니다.\n   *\n   * CLI에서 사용됩니다.\n   */\n  async check(): Promise<void> {\n    const codes = await this.compareMigrations(this.targets.compare!);\n    if (codes.length === 0) {\n      console.log(chalk.green(\"\\n현재 모두 싱크된 상태입니다.\"));\n      return;\n    }\n\n    // 현재 생성된 코드 표기\n    console.table(codes, [\"type\", \"title\"]);\n    console.log(codes[0]);\n  }\n\n  /**\n   * 마이그레이션을 수행합니다.\n   *\n   * runAction이 인자로 들어온 타겟들에 대해 주어진 동작(apply/rollback)을 수행한다면,\n   * 이 함수는 생성자로 들어온 connection(knex)들에 대해 마이그레이션을 수행합니다.\n   *\n   * CLI에서 사용됩니다.\n   */\n  async run(): Promise<void> {\n    // pending 마이그레이션 확인\n    const [, pendingList] = await this.targets.pending.migrate.list();\n    if (pendingList.length > 0) {\n      console.log(\n        chalk.red(\"pending 된 마이그레이션이 존재합니다.\"),\n        pendingList.map((pending: any) => pending.file)\n      );\n\n      // pending이 있는 경우 Shadow DB 테스트 진행 여부 컨펌\n      const answer = await prompts({\n        type: \"confirm\",\n        name: \"value\",\n        message: \"Shadow DB 테스트를 진행하시겠습니까?\",\n        initial: true,\n      });\n      if (answer.value === false) {\n        return;\n      }\n\n      console.time(chalk.blue(\"Migrator - runShadowTest\"));\n      await this.runShadowTest();\n      console.timeEnd(chalk.blue(\"Migrator - runShadowTest\"));\n      await Promise.all(\n        this.targets.apply.map(async (applyDb) => {\n          const label = chalk.green(\n            `APPLIED ${\n              applyDb.client.connectionSettings.host\n            } ${applyDb.client.database()}`\n          );\n          console.time(label);\n          const [,] = await applyDb.migrate.latest();\n          console.timeEnd(label);\n        })\n      );\n    }\n\n    // Entity-DB간 비교하여 코드 생성 리턴\n    const codes = await this.compareMigrations(this.targets.compare!);\n    if (codes.length === 0) {\n      console.log(chalk.green(\"\\n현재 모두 싱크된 상태입니다.\"));\n      return;\n    }\n\n    // 현재 생성된 코드 표기\n    console.table(codes, [\"type\", \"title\"]);\n\n    /* DEBUG: 디버깅용 코드\n    codes.map((code) => console.log(code.formatted));\n    process.exit();\n     */\n\n    // 실제 파일 생성 프롬프트\n    const answer = await prompts({\n      type: \"confirm\",\n      name: \"value\",\n      message: \"마이그레이션 코드를 생성하시겠습니까?\",\n      initial: false,\n    });\n    if (answer.value === false) {\n      return;\n    }\n\n    // 실제 코드 생성\n    const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;\n\n    for (const [index, code] of codes.entries()) {\n      if (code.formatted) {\n        const dateTag = DateTime.local()\n          .plus({ seconds: index })\n          .toFormat(\"yyyyMMddHHmmss\");\n        const filePath = `${migrationsDir}/${dateTag}_${code.title}.ts`;\n        await writeFile(filePath, code.formatted!);\n        console.log(chalk.green(`MIGRTAION CREATED ${filePath}`));\n      }\n    }\n  }\n\n  /**\n   * 타겟으로 지정된 DB를 롤백합니다.\n   *\n   * runAction이 인자로 들어온 타겟들에 대해 주어진 동작(apply/rollback)을 수행한다면,\n   * 이 함수는 생성자로 들어온 connection(knex)들에 대해 롤백을 수행합니다.\n   *\n   * CLI에서 사용됩니다.\n   */\n  async rollback() {\n    console.time(chalk.red(\"rollback:\"));\n    const rollbackAllResult = await Promise.all(\n      this.targets.apply.map(async (db) => {\n        await db.migrate.forceFreeMigrationsLock();\n        return db.migrate.rollback(undefined, false);\n      })\n    );\n    console.dir({ rollbackAllResult }, { depth: null });\n    console.timeEnd(chalk.red(\"rollback:\"));\n  }\n  /**\n   * Shadow DB 테스트를 진행합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns Shadow DB 테스트 결과\n   */\n  async runShadowTest(): Promise<\n    {\n      connKey: string;\n      batchNo: number;\n      applied: string[];\n    }[]\n  > {\n    // ShadowDB 생성 후 테스트 진행\n    const tdb = knex(Sonamu.dbConfig.test);\n    const tdbConn = Sonamu.dbConfig.test\n      .connection as Knex.MySql2ConnectionConfig;\n    const shadowDatabase = tdbConn.database + \"__migration_shadow\";\n    const tmpSqlPath = `/tmp/${shadowDatabase}.sql`;\n\n    // 테스트DB 덤프 후 Database명 치환\n    console.log(\n      chalk.magenta(`${tdbConn.database}의 데이터 ${tmpSqlPath}로 덤프`)\n    );\n    execSync(\n      `mysqldump -h${tdbConn.host} -P${tdbConn.port ?? 3306} -u${tdbConn.user} -p'${tdbConn.password}' ${tdbConn.database} --single-transaction --no-create-db --triggers > ${tmpSqlPath};`\n    );\n    execSync(\n      `sed -i'' -e 's/\\`${tdbConn.database}\\`/\\`${shadowDatabase}\\`/g' ${tmpSqlPath};`\n    );\n\n    // 기존 ShadowDB 리셋\n    console.log(chalk.magenta(`${shadowDatabase} 리셋`));\n    await tdb.raw(`DROP DATABASE IF EXISTS \\`${shadowDatabase}\\`;`);\n    await tdb.raw(`CREATE DATABASE \\`${shadowDatabase}\\`;`);\n\n    // ShadowDB 테이블 + 데이터 생성\n    console.log(chalk.magenta(`${shadowDatabase} 데이터베이스 생성`));\n    execSync(\n      `mysql -h${tdbConn.host} -P${tdbConn.port ?? 3306} -u${tdbConn.user} -p'${tdbConn.password}' ${shadowDatabase} < ${tmpSqlPath};`\n    );\n\n    // shadow db 테스트 진행\n    const sdb = knex({\n      ...Sonamu.dbConfig.test,\n      connection: {\n        ...tdbConn,\n        database: shadowDatabase,\n        password: tdbConn.password,\n      },\n    });\n\n    // shadow db 테스트 진행\n    try {\n      const [batchNo, applied] = await sdb.migrate.latest();\n      console.log(chalk.green(\"Shadow DB 테스트에 성공했습니다!\"), {\n        batchNo,\n        applied,\n      });\n\n      // 생성한 Shadow DB 삭제\n      console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n      await tdb.raw(`DROP DATABASE IF EXISTS \\`${shadowDatabase}\\`;`);\n\n      return [\n        {\n          connKey: \"shadow\",\n          batchNo,\n          applied,\n        },\n      ];\n    } catch (e) {\n      console.error(e);\n      throw new ServiceUnavailableException(\"Shadow DB 테스트 진행 중 에러\");\n    } finally {\n      await tdb.destroy();\n    }\n  }\n\n  /**\n   * 모든 DB를 롤백하고 전체 마이그레이션 파일을 삭제합니다.\n   *\n   * CLI에서 사용됩니다.\n   *\n   * @returns\n   */\n  async resetAll() {\n    const answer = await prompts({\n      type: \"confirm\",\n      name: \"value\",\n      message: \"모든 DB를 롤백하고 전체 마이그레이션 파일을 삭제하시겠습니까?\",\n      initial: false,\n    });\n    if (answer.value === false) {\n      return;\n    }\n\n    console.time(chalk.red(\"rollback-all:\"));\n    const rollbackAllResult = await Promise.all(\n      this.targets.apply.map(async (db) => {\n        await db.migrate.forceFreeMigrationsLock();\n        return db.migrate.rollback(undefined, true);\n      })\n    );\n    console.log({ rollbackAllResult });\n    console.timeEnd(chalk.red(\"rollback-all:\"));\n\n    const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;\n    console.time(chalk.red(\"delete migration files\"));\n    execSync(`rm -f ${migrationsDir}/*`);\n    execSync(`rm -f ${migrationsDir.replace(\"/src/\", \"/dist/\")}/*`);\n    console.timeEnd(chalk.red(\"delete migration files\"));\n  }\n\n  private async compareMigrations(\n    compareDB: Knex\n  ): Promise<GenMigrationCode[]> {\n    // Entity 순회하여 싱크\n    const entityIds = EntityManager.getAllIds();\n\n    // 조인테이블 포함하여 Entity에서 MigrationSet 추출\n    const entitySetsWithJoinTable = entityIds\n      .filter((entityId) => EntityManager.get(entityId).props.length > 0)\n      .map((entityId) =>\n        getMigrationSetFromEntity(EntityManager.get(entityId))\n      );\n\n    // 조인테이블만 추출\n    const joinTablesWithDup = entitySetsWithJoinTable\n      .map((entitySet) => entitySet.joinTables)\n      .flat();\n    // 중복 제거 (중복인 경우 indexes를 병합)\n    const joinTables = Object.values(\n      _.groupBy(joinTablesWithDup, (jt) => jt.table)\n    ).map((tables) => {\n      if (tables.length === 1) {\n        return tables[0];\n      }\n      return {\n        ...tables[0],\n        indexes: _.uniqBy(\n          tables.flatMap((t) => t.indexes),\n          (index) => [index.type, ...index.columns.sort()].join(\"-\")\n        ),\n      };\n    });\n\n    // 조인테이블 포함하여 MigrationSet 배열\n    const entitySets: MigrationSet[] = [\n      ...entitySetsWithJoinTable,\n      ...joinTables,\n    ];\n\n    const codes: GenMigrationCode[] = (\n      await Promise.all(\n        entitySets.map(async (entitySet) => {\n          const dbSet = await getMigrationSetFromDB(compareDB, entitySet.table);\n\n          if (dbSet === null) {\n            // 기존 테이블 없음, 새로 테이블 생성\n            return await generateCreateCode(entitySet);\n          } else {\n            // 기존 테이블 존재하는 케이스\n            return await generateAlterCode(entitySet, dbSet);\n          }\n        })\n      )\n    ).flat();\n\n    // normal 타입이 앞으로, foreign이 뒤로\n    codes.sort((codeA, codeB) => {\n      if (codeA.type === \"foreign\" && codeB.type == \"normal\") {\n        return 1;\n      } else if (codeA.type === \"normal\" && codeB.type === \"foreign\") {\n        return -1;\n      } else {\n        return 0;\n      }\n    });\n\n    return codes;\n  }\n\n  /**\n   * 마이그레이션 대상 커넥션을 종료합니다.\n   *\n   * CLI에서 사용됩니다.\n   *\n   * @returns {Promise<void>} 종료 결과\n   */\n  async destroy(): Promise<void> {\n    await Promise.all(\n      this.targets.apply.map((db) => {\n        return db.destroy();\n      })\n    );\n  }\n}\n"],"names":["_","knex","chalk","DateTime","mkdir","readdir","unlink","writeFile","exists","prompts","execSync","path","EntityManager","Sonamu","ServiceUnavailableException","generateCreateCode","generateAlterCode","getMigrationSetFromDB","getMigrationSetFromEntity","Migrator","targets","options","dbConfig","mode","devDB","development_master","testDB","test","fixtureLocalDB","fixture_local","applyDBs","connection","host","fixture_remote","database","fixtureRemoteDB","push","compare","pending","shadow","apply","productionDB","production_master","Error","getMigrationCodes","srcMigrationsDir","join","apiRootPath","recursive","codes","filter","f","endsWith","map","name","replace","sort","a","b","getStatus","connKeys","Object","keys","key","statuses","Promise","all","connKey","knexOptions","tConn","status","migrate","err","console","warn","yellow","JSON","stringify","fdList","list","fd","file","currentVersion","destroy","connString","user","port","preparedCodes","status0conn","find","undefined","compareDBconn","genCodes","compareMigrations","conns","runAction","action","configs","uniqBy","target","c","config","result","batchNo","applied","latest","rollback","delCodes","codeNames","some","conn","codeName","includes","delFiles","res","delFile","log","red","sum","generatePreparedCodes","length","green","migrationsDir","index","pcode","entries","formatted","dateTag","local","plus","seconds","toFormat","filePath","title","clearPendingList","pendingList","delList","df","p","check","table","run","answer","type","message","initial","value","time","blue","runShadowTest","timeEnd","applyDb","label","client","connectionSettings","code","rollbackAllResult","db","forceFreeMigrationsLock","dir","depth","tdb","tdbConn","shadowDatabase","tmpSqlPath","magenta","password","raw","sdb","e","error","resetAll","compareDB","entityIds","getAllIds","entitySetsWithJoinTable","entityId","get","props","joinTablesWithDup","entitySet","joinTables","flat","values","groupBy","jt","tables","indexes","flatMap","t","columns","entitySets","dbSet","codeA","codeB"],"mappings":"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"}
|
|
330
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/migration/migrator.ts"],"sourcesContent":["import assert from \"assert\";\nimport chalk from \"chalk\";\nimport { mkdir, readdir, unlink, writeFile } from \"fs/promises\";\nimport knex, { type Knex } from \"knex\";\nimport path from \"path\";\nimport { group, sum, unique } from \"radashi\";\nimport { Sonamu } from \"../api\";\nimport { DB, type SonamuDBConfig } from \"../database/db\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { ServiceUnavailableException } from \"../exceptions/so-exceptions\";\nimport { Naite } from \"../naite/naite\";\nimport type { GenMigrationCode, MigrationSet } from \"../types/types\";\nimport { isTest } from \"../utils/controller\";\nimport { exists } from \"../utils/fs-utils\";\nimport { generateAlterCode, generateCreateCode } from \"./code-generation\";\nimport { getMigrationSetFromEntity } from \"./migration-set\";\nimport { PostgreSQLSchemaReader } from \"./postgresql-schema-reader\";\nimport type { ConnString, MigrationCode, MigrationStatus } from \"./types\";\n\nexport type MigrationResult = {\n  connKey: string;\n  batchNo: number;\n  applied: string[];\n}[];\n\nexport class Migrator {\n  private async getMigrationCodes(): Promise<MigrationCode[]> {\n    const srcMigrationsDir = path.join(Sonamu.apiRootPath, \"src\", \"migrations\"); // 이건 환경에 관계없이 항상 src에서 찾아야 해요.\n\n    if (!(await exists(srcMigrationsDir))) {\n      await mkdir(srcMigrationsDir, {\n        recursive: true,\n      });\n    }\n\n    const codes = (await readdir(srcMigrationsDir))\n      .filter((f) => f.endsWith(\".ts\"))\n      .map((f) => ({\n        name: f.replace(\".ts\", \"\"),\n        path: path.join(srcMigrationsDir, f),\n      }))\n      .sort((a, b) => (a.name < b.name ? 1 : -1)); // 이름 내림차순 정렬(최신순)\n\n    Naite.t(\"migrator:getMigrationCodes:results\", codes);\n    return codes;\n  }\n\n  /**\n   * 타겟별 마이그레이션 상태와 코드 생성/준비 상태를 구해옵니다.\n   * 실제로 DB에 접근도 하고 마이그레이션 코드 파일도 확인하고,\n   * 필요하다면 적용할 수 있는 코드를 생성까지 해옵니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @returns\n   */\n  async getStatus(): Promise<MigrationStatus> {\n    const codes = await this.getMigrationCodes();\n    Naite.t(\"migrator:getStatus:codes\", codes);\n\n    const connKeys = Object.keys(Sonamu.dbConfig).filter(\n      (key) => key.endsWith(\"_slave\") === false,\n    ) as (keyof typeof Sonamu.dbConfig)[];\n\n    let migrationStatusError: string | undefined;\n\n    const statuses = await Promise.all(\n      connKeys.map(async (connKey) => {\n        const knexOptions = Sonamu.dbConfig[connKey];\n        const tConn = knex(knexOptions);\n\n        const status = await (async () => {\n          try {\n            return await tConn.migrate.status();\n          } catch (err) {\n            console.warn(\n              chalk.yellow(\n                `${connKey}의 마이그레이션 상태를 가져오는 데에 실패하였습니다. 데이터베이스가 올바르게 구성되지 않은 것 같습니다. 확인하시고 다시 시도해주세요.\\n시도한 연결 설정:\\n${JSON.stringify(knexOptions.connection, null, 2)}\\n발생한 에러:\\n${err}\\n`,\n              ),\n            );\n            migrationStatusError = err instanceof Error ? err.message : String(err);\n            return \"error\";\n          }\n        })();\n        const pending: string[] = await (async () => {\n          try {\n            const [, fdList] = await tConn.migrate.list();\n            return fdList.map((fd: { file: string }) => fd.file.replace(\".ts\", \"\"));\n          } catch (err) {\n            migrationStatusError = err instanceof Error ? err.message : String(err);\n            return [];\n          }\n        })();\n        const currentVersion = await (async () => {\n          try {\n            return await tConn.migrate.currentVersion();\n          } catch (_err) {\n            migrationStatusError = _err instanceof Error ? _err.message : String(_err);\n            return \"error\";\n          }\n        })();\n        Naite.t(\"migrator:getStatus:status\", status);\n\n        const connection = knexOptions.connection as Knex.PgConnectionConfig;\n\n        await tConn.destroy();\n\n        return {\n          name: connKey.replace(\"_master\", \"\"),\n          connKey,\n          connString: `pg://${connection.user ?? \"\"}@${connection.host}:${\n            connection.port\n          }/${connection.database}` as ConnString,\n          currentVersion,\n          status: status as number | \"error\",\n          pending,\n        };\n      }),\n    );\n\n    Naite.t(\"migrator:getStatus:conns\", statuses);\n\n    const preparedCodes: GenMigrationCode[] = await (async () => {\n      const status0conn = statuses.find((status) => status.status === 0);\n      if (status0conn === undefined) {\n        console.warn(\n          chalk.yellow(\n            `While trying to prepare migration codes, we found that there is no database to compare migrations. We need at least one database where every migration is applied(status === 0). You might want to apply your existing migrations to one of the databases.`,\n          ),\n        );\n        return [];\n      }\n\n      const compareDBconn = knex(Sonamu.dbConfig[status0conn.connKey]);\n      const genCodes = await this.compareMigrations(compareDBconn);\n\n      await compareDBconn.destroy();\n\n      return genCodes;\n    })();\n\n    Naite.t(\"migrator:getStatus:preparedCodes\", preparedCodes);\n\n    return {\n      conns: statuses,\n      codes,\n      preparedCodes,\n      error: migrationStatusError,\n    };\n  }\n\n  /**\n   * 마이그레이션을 적용하거나 롤백합니다.\n   * Sonamu UI에서 마이그레이션 작업을 수행할 때 사용됩니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @param action 작업 유형 (apply/rollback)\n   * @param targets 작업 대상 DB 설정 키 (keyof SonamuDBConfig)\n   * @returns 작업 결과\n   */\n  async runAction(\n    action: \"apply\" | \"rollback\",\n    targets: (keyof SonamuDBConfig)[],\n  ): Promise<MigrationResult> {\n    Naite.t(\"migrator:runAction:action\", action);\n    Naite.t(\"migrator:runAction:targets\", targets);\n\n    // get uniq knex configs\n    const configs = unique(\n      targets\n        .map((target) => ({\n          connKey: target,\n          options: Sonamu.dbConfig[target as keyof typeof Sonamu.dbConfig],\n        }))\n        .filter((c) => c.options !== undefined),\n      ({ options }) =>\n        `${(options.connection as Knex.PgConnectionConfig).host}:${\n          (options.connection as Knex.PgConnectionConfig).port ?? 5432\n        }/${(options.connection as Knex.PgConnectionConfig).database}`,\n    );\n\n    // get connections\n    const conns = await Promise.all(\n      configs.map(async (config) => ({\n        connKey: config.connKey,\n        knex: knex(config.options),\n      })),\n    );\n\n    // action\n    const result = await (async () => {\n      switch (action) {\n        case \"apply\":\n          return Promise.all(\n            conns.map(async ({ connKey, knex }) => {\n              const [batchNo, applied] = await knex.migrate.latest();\n              return {\n                connKey,\n                batchNo,\n                applied, // 이번 latest 호출로 인해 \"up\"이 적용된 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L560\n              };\n            }),\n          );\n        case \"rollback\":\n          return Promise.all(\n            conns.map(async ({ connKey, knex }) => {\n              const [batchNo, applied] = await knex.migrate.rollback();\n              return {\n                connKey,\n                batchNo,\n                applied, // 이번 rollback 호출로 인해 \"down\"이 적용된(=롤백된) 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L611\n              };\n            }),\n          );\n      }\n    })();\n\n    // destroy\n    await Promise.all(\n      conns.map(({ knex }) => {\n        return knex.destroy();\n      }),\n    );\n\n    Naite.t(\"migrator:runAction:result\", result);\n\n    return result;\n  }\n\n  /**\n   * 삭제 가능한 마이그레이션 코드 파일을 검증합니다.\n   *\n   * @param conns 마이그레이션 상태 배열\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제 가능 여부 및 적용된 마이그레이션 코드 파일 이름\n   */\n  validateDeletable(conns: MigrationStatus[\"conns\"], codeNames: string[]) {\n    const appliedCodes = codeNames.filter((codeName) =>\n      conns.some((conn) => conn.pending.includes(codeName) === false),\n    );\n\n    return {\n      canDelete: appliedCodes.length === 0,\n      appliedCodes,\n    };\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 삭제합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제된 마이그레이션 코드 파일 개수\n   */\n  async delCodes(codeNames: string[]): Promise<number> {\n    const { conns } = await this.getStatus();\n    const { canDelete, appliedCodes } = this.validateDeletable(conns, codeNames);\n    if (!canDelete) {\n      throw new Error(\n        `You cannot delete a migration file if there is already applied. Applied codes: ${appliedCodes.join(\", \")}`,\n      );\n    }\n\n    return sum(\n      await Promise.all(\n        codeNames.map(async (codeName) => {\n          const filePath = `${Sonamu.apiRootPath}/src/migrations/${codeName}.ts`;\n          if (await exists(filePath)) {\n            await unlink(filePath);\n            return 1;\n          }\n          return 0;\n        }),\n      ),\n    );\n  }\n\n  private genDateTag(index: number, baseDate: Date = new Date()): string {\n    const date = new Date(baseDate.getTime() + index * 1000);\n    const pad = (num: number, size: number = 2) => num.toString().padStart(size, \"0\");\n    return (\n      date.getFullYear().toString() +\n      pad(date.getMonth() + 1) +\n      pad(date.getDate()) +\n      pad(date.getHours()) +\n      pad(date.getMinutes()) +\n      pad(date.getSeconds())\n    );\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 생성합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns 생성된 마이그레이션 코드 파일 개수\n   */\n  async generatePreparedCodes(): Promise<number> {\n    const { preparedCodes } = await this.getStatus();\n    Naite.t(\"migrator:generatePreparedCodes:preparedCodes\", preparedCodes);\n    if (preparedCodes.length === 0) {\n      console.log(chalk.green(\"\\n현재 모두 싱크된 상태입니다.\"));\n      return 0;\n    }\n\n    // 실제 코드 생성\n    const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;\n\n    for (const [index, pcode] of preparedCodes.entries()) {\n      if (pcode.formatted) {\n        const dateTag = this.genDateTag(index);\n        const filePath = `${migrationsDir}/${dateTag}_${pcode.title}.ts`;\n        await writeFile(filePath, pcode.formatted);\n        !isTest() && console.log(chalk.green(`MIGRTAION CREATED ${filePath}`));\n      }\n    }\n\n    return preparedCodes.length;\n  }\n\n  async compareMigrations(compareDB: Knex): Promise<GenMigrationCode[]> {\n    // Entity 순회하여 싱크\n    const entityIds = EntityManager.getAllIds();\n\n    // 조인테이블 포함하여 Entity에서 MigrationSet 추출\n    const entitySetsWithJoinTable = entityIds\n      .filter((entityId) => EntityManager.get(entityId).props.length > 0)\n      .map((entityId) => getMigrationSetFromEntity(EntityManager.get(entityId)));\n\n    // 조인테이블만 추출\n    const joinTablesWithDup = entitySetsWithJoinTable.flatMap((entitySet) => entitySet.joinTables);\n    // 중복 제거 (중복인 경우 indexes를 병합)\n    const joinTables = Object.values(group(joinTablesWithDup, (jt) => jt.table)).map((tables) => {\n      assert(tables !== undefined, \"tables is undefined\");\n      if (tables.length === 1) {\n        return tables[0];\n      }\n      return {\n        ...tables[0],\n        indexes: unique(\n          tables.flatMap((t) => t.indexes),\n          (index) => [index.type, ...index.columns.sort()].join(\"-\"),\n        ),\n      };\n    });\n\n    // 조인테이블 포함하여 MigrationSet 배열\n    const entitySets: MigrationSet[] = [...entitySetsWithJoinTable, ...joinTables];\n\n    const codes: GenMigrationCode[] = (\n      await Promise.all(\n        entitySets.map(async (entitySet) => {\n          const dbSet = await PostgreSQLSchemaReader.getMigrationSetFromDB(\n            compareDB,\n            entitySet.table,\n          );\n          Naite.t(`migrator:compareMigrations:entitySet:${entitySet.table}`, entitySet);\n          Naite.t(`migrator:compareMigrations:dbSet:${entitySet.table}`, dbSet);\n\n          if (dbSet === null) {\n            // 기존 테이블 없음, 새로 테이블 생성\n            return await generateCreateCode(entitySet);\n          } else {\n            // 기존 테이블 존재하는 케이스\n            return await generateAlterCode(entitySet, dbSet);\n          }\n        }),\n      )\n    ).flat();\n\n    // normal 타입이 앞으로, foreign이 뒤로\n    codes.sort((codeA, codeB) => {\n      if (codeA.type === \"foreign\" && codeB.type === \"normal\") {\n        return 1;\n      } else if (codeA.type === \"normal\" && codeB.type === \"foreign\") {\n        return -1;\n      } else {\n        return 0;\n      }\n    });\n\n    return codes;\n  }\n\n  /**\n   * Shadow DB 테스트를 진행합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns Shadow DB 테스트 결과\n   */\n  async runShadowTest(): Promise<MigrationResult> {\n    const tdbConn = Sonamu.dbConfig.test.connection as Knex.PgConnectionConfig;\n    const shadowDatabase = `${tdbConn.database}__migration_shadow`;\n\n    // 테스트 상황에서는 트랜잭션을 초기화하고, 새 데이터베이스 커넥션을 가져와야 함\n    if (isTest()) {\n      await DB.clearTestTransaction();\n      await DB.destroy();\n    }\n\n    // 기존 Shadow DB 삭제 후 Shadow DB 생성\n    const tdb = knex(Sonamu.dbConfig.test);\n    !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n    await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n    await tdb.raw(`CREATE DATABASE ${shadowDatabase} TEMPLATE ${tdbConn.database}`);\n\n    // Shadow DB에 연결\n    const sdb = knex({\n      ...Sonamu.dbConfig.test,\n      connection: {\n        ...tdbConn,\n        database: shadowDatabase,\n        password: tdbConn.password,\n      },\n    });\n\n    // shadow DB 테스트 진행\n    try {\n      const [batchNo, applied] = await sdb.migrate.latest();\n      !isTest() &&\n        console.log(chalk.green(\"Shadow DB 테스트에 성공했습니다!\"), {\n          batchNo,\n          applied,\n        });\n\n      return [\n        {\n          connKey: \"shadow\",\n          batchNo,\n          applied,\n        },\n      ];\n    } catch (e) {\n      console.error(e);\n      throw new ServiceUnavailableException(\"Shadow DB 테스트 진행 중 에러\");\n    } finally {\n      // Shadow DB 연결 종료\n      await sdb.destroy();\n\n      // Shadow DB 삭제\n      !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n      await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n\n      // Test DB 연결 종료\n      await tdb.destroy();\n    }\n  }\n}\n"],"names":["assert","chalk","mkdir","readdir","unlink","writeFile","knex","path","group","sum","unique","Sonamu","DB","EntityManager","ServiceUnavailableException","Naite","isTest","exists","generateAlterCode","generateCreateCode","getMigrationSetFromEntity","PostgreSQLSchemaReader","Migrator","getMigrationCodes","srcMigrationsDir","join","apiRootPath","recursive","codes","filter","f","endsWith","map","name","replace","sort","a","b","t","getStatus","connKeys","Object","keys","dbConfig","key","migrationStatusError","statuses","Promise","all","connKey","knexOptions","tConn","status","migrate","err","console","warn","yellow","JSON","stringify","connection","Error","message","String","pending","fdList","list","fd","file","currentVersion","_err","destroy","connString","user","host","port","database","preparedCodes","status0conn","find","undefined","compareDBconn","genCodes","compareMigrations","conns","error","runAction","action","targets","configs","target","options","c","config","result","batchNo","applied","latest","rollback","validateDeletable","codeNames","appliedCodes","codeName","some","conn","includes","canDelete","length","delCodes","filePath","genDateTag","index","baseDate","Date","date","getTime","pad","num","size","toString","padStart","getFullYear","getMonth","getDate","getHours","getMinutes","getSeconds","generatePreparedCodes","log","green","migrationsDir","pcode","entries","formatted","dateTag","title","compareDB","entityIds","getAllIds","entitySetsWithJoinTable","entityId","get","props","joinTablesWithDup","flatMap","entitySet","joinTables","values","jt","table","tables","indexes","type","columns","entitySets","dbSet","getMigrationSetFromDB","flat","codeA","codeB","runShadowTest","tdbConn","test","shadowDatabase","clearTestTransaction","tdb","magenta","raw","sdb","password","e"],"mappings":"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"}
|