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,7 +1,7 @@
|
|
|
1
|
-
import { Knex } from "knex";
|
|
1
|
+
import type { Knex } from "knex";
|
|
2
2
|
type TableData = {
|
|
3
3
|
references: Set<string>;
|
|
4
|
-
rows:
|
|
4
|
+
rows: Record<string, unknown>[];
|
|
5
5
|
uniqueIndexes: {
|
|
6
6
|
name?: string;
|
|
7
7
|
columns: string[];
|
|
@@ -13,7 +13,7 @@ export type UBRef = {
|
|
|
13
13
|
of: string;
|
|
14
14
|
use?: string;
|
|
15
15
|
};
|
|
16
|
-
export declare function isRefField(field:
|
|
16
|
+
export declare function isRefField(field: unknown): field is UBRef;
|
|
17
17
|
export declare class UpsertBuilder {
|
|
18
18
|
tables: Map<string, TableData>;
|
|
19
19
|
constructor();
|
|
@@ -29,6 +29,12 @@ export declare class UpsertBuilder {
|
|
|
29
29
|
chunkSize?: number;
|
|
30
30
|
where?: string | string[];
|
|
31
31
|
}): Promise<void>;
|
|
32
|
+
/**
|
|
33
|
+
* rows를 의존성 순서에 따라 레벨별로 그룹화
|
|
34
|
+
* - 자기 참조 없는 경우 : 모든 rows가 Level 0
|
|
35
|
+
* - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화
|
|
36
|
+
*/
|
|
37
|
+
private buildInsertLevels;
|
|
32
38
|
}
|
|
33
39
|
export {};
|
|
34
40
|
//# sourceMappingURL=upsert-builder.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"upsert-builder.d.ts","sourceRoot":"","sources":["../../src/database/upsert-builder.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"upsert-builder.d.ts","sourceRoot":"","sources":["../../src/database/upsert-builder.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAOjC,KAAK,SAAS,GAAG;IACf,UAAU,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;IACxB,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,CAAC;IAChC,aAAa,EAAE;QAAE,IAAI,CAAC,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,EAAE,CAAA;KAAE,EAAE,CAAC;IACtD,UAAU,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CACjC,CAAC;AACF,MAAM,MAAM,KAAK,GAAG;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,EAAE,EAAE,MAAM,CAAC;IACX,GAAG,CAAC,EAAE,MAAM,CAAC;CACd,CAAC;AACF,wBAAgB,UAAU,CAAC,KAAK,EAAE,OAAO,GAAG,KAAK,IAAI,KAAK,CAOzD;AAED,qBAAa,aAAa;IACxB,MAAM,EAAE,GAAG,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;;IAK/B,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,SAAS;IAwBtC,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO;IAIpC,QAAQ,CAAC,CAAC,SAAS,MAAM,EACvB,SAAS,EAAE,MAAM,EACjB,GAAG,EAAE;SACF,GAAG,IAAI,CAAC,CAAC,CAAC,EAAE,KAAK,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,MAAM,GAAG,IAAI,GAAG,MAAM,GAAG,OAAO;KAClF,GACA,KAAK;IAqFF,MAAM,CAAC,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAG3E,UAAU,CAAC,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAI/E,cAAc,CAClB,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,IAAI,EAAE,QAAQ,GAAG,QAAQ,EACzB,SAAS,CAAC,EAAE,MAAM,GACjB,OAAO,CAAC,MAAM,EAAE,CAAC;IAoKd,WAAW,CACf,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,OAAO,CAAC,EAAE;QACR,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,KAAK,CAAC,EAAE,MAAM,GAAG,MAAM,EAAE,CAAC;KAC3B,GACA,OAAO,CAAC,IAAI,CAAC;IAyChB;;;;OAIG;IACH,OAAO,CAAC,iBAAiB;CA8D1B"}
|
|
@@ -1,10 +1,11 @@
|
|
|
1
1
|
import { randomUUID } from "crypto";
|
|
2
|
-
import
|
|
2
|
+
import { unique } from "radashi";
|
|
3
3
|
import { EntityManager } from "../entity/entity-manager.js";
|
|
4
|
-
import {
|
|
4
|
+
import { Naite } from "../naite/naite.js";
|
|
5
|
+
import { assertDefined, chunk, nonNullable } from "../utils/utils.js";
|
|
5
6
|
import { batchUpdate } from "./_batch_update.js";
|
|
6
7
|
export function isRefField(field) {
|
|
7
|
-
return field !== undefined && field !== null && field
|
|
8
|
+
return field !== undefined && field !== null && field?.of !== undefined && field?.uuid !== undefined;
|
|
8
9
|
}
|
|
9
10
|
export class UpsertBuilder {
|
|
10
11
|
tables;
|
|
@@ -13,22 +14,24 @@ export class UpsertBuilder {
|
|
|
13
14
|
}
|
|
14
15
|
getTable(tableName) {
|
|
15
16
|
const table = this.tables.get(tableName);
|
|
16
|
-
if (table
|
|
17
|
-
|
|
18
|
-
try {
|
|
19
|
-
return EntityManager.getTableSpec(tableName);
|
|
20
|
-
} catch {
|
|
21
|
-
return null;
|
|
22
|
-
}
|
|
23
|
-
})();
|
|
24
|
-
this.tables.set(tableName, {
|
|
25
|
-
references: new Set(),
|
|
26
|
-
rows: [],
|
|
27
|
-
uniqueIndexes: tableSpec?.uniqueIndexes ?? [],
|
|
28
|
-
uniquesMap: new Map()
|
|
29
|
-
});
|
|
17
|
+
if (table) {
|
|
18
|
+
return table;
|
|
30
19
|
}
|
|
31
|
-
|
|
20
|
+
const tableSpec = (()=>{
|
|
21
|
+
try {
|
|
22
|
+
return EntityManager.getTableSpec(tableName);
|
|
23
|
+
} catch {
|
|
24
|
+
return null;
|
|
25
|
+
}
|
|
26
|
+
})();
|
|
27
|
+
const tableData = {
|
|
28
|
+
references: new Set(),
|
|
29
|
+
rows: [],
|
|
30
|
+
uniqueIndexes: tableSpec?.uniqueIndexes ?? [],
|
|
31
|
+
uniquesMap: new Map()
|
|
32
|
+
};
|
|
33
|
+
this.tables.set(tableName, tableData);
|
|
34
|
+
return tableData;
|
|
32
35
|
}
|
|
33
36
|
hasTable(tableName) {
|
|
34
37
|
return this.tables.has(tableName);
|
|
@@ -52,17 +55,23 @@ export class UpsertBuilder {
|
|
|
52
55
|
return uniqueKeyArray.join("---delimiter--");
|
|
53
56
|
}).filter(nonNullable);
|
|
54
57
|
// uuid 생성 로직
|
|
55
|
-
const uuid = (()=>{
|
|
58
|
+
const { uuid, isReused } = (()=>{
|
|
56
59
|
// 키를 순회하여 이미 존재하는 키가 있는지 확인
|
|
57
60
|
if (uniqueKeys.length > 0) {
|
|
58
61
|
for (const uniqueKey of uniqueKeys){
|
|
59
62
|
if (table.uniquesMap.has(uniqueKey)) {
|
|
60
|
-
return
|
|
63
|
+
return {
|
|
64
|
+
uuid: assertDefined(table.uniquesMap.get(uniqueKey), "Unique key not found"),
|
|
65
|
+
isReused: true
|
|
66
|
+
};
|
|
61
67
|
}
|
|
62
68
|
}
|
|
63
69
|
}
|
|
64
70
|
// 찾을 수 없는 경우 생성
|
|
65
|
-
return
|
|
71
|
+
return {
|
|
72
|
+
uuid: randomUUID(),
|
|
73
|
+
isReused: false
|
|
74
|
+
};
|
|
66
75
|
})();
|
|
67
76
|
// 모든 유니크키에 대해 유니크맵에 uuid 저장
|
|
68
77
|
if (uniqueKeys.length > 0) {
|
|
@@ -72,28 +81,42 @@ export class UpsertBuilder {
|
|
|
72
81
|
}
|
|
73
82
|
// 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가
|
|
74
83
|
// 이 정보를 나중에 치환할 때 사용
|
|
75
|
-
row = Object.
|
|
76
|
-
const rowValue = row[rowKey];
|
|
84
|
+
row = Object.fromEntries(Object.entries(row).map(([rowKey, rowValue])=>{
|
|
77
85
|
if (isRefField(rowValue)) {
|
|
78
86
|
rowValue.use ??= "id";
|
|
79
|
-
table.references.add(rowValue.of
|
|
80
|
-
|
|
87
|
+
table.references.add(`${rowValue.of}.${rowValue.use}`);
|
|
88
|
+
return [
|
|
89
|
+
rowKey,
|
|
90
|
+
rowValue
|
|
91
|
+
];
|
|
81
92
|
} else if (typeof rowValue === "object" && !(rowValue instanceof Date)) {
|
|
82
93
|
// object인 경우 JSON으로 변환
|
|
83
|
-
|
|
94
|
+
return [
|
|
95
|
+
rowKey,
|
|
96
|
+
rowValue === null ? null : JSON.stringify(rowValue)
|
|
97
|
+
];
|
|
84
98
|
} else {
|
|
85
|
-
|
|
99
|
+
return [
|
|
100
|
+
rowKey,
|
|
101
|
+
rowValue
|
|
102
|
+
];
|
|
86
103
|
}
|
|
87
|
-
|
|
88
|
-
}, {});
|
|
104
|
+
}));
|
|
89
105
|
table.rows.push({
|
|
90
106
|
uuid,
|
|
91
107
|
...row
|
|
92
108
|
});
|
|
93
|
-
|
|
109
|
+
const result = {
|
|
94
110
|
of: tableName,
|
|
95
111
|
uuid: row.uuid ?? uuid
|
|
96
112
|
};
|
|
113
|
+
Naite.t("puri:ub-register", {
|
|
114
|
+
tableName,
|
|
115
|
+
uuid: result.uuid,
|
|
116
|
+
isUuidReused: isReused,
|
|
117
|
+
row
|
|
118
|
+
});
|
|
119
|
+
return result;
|
|
97
120
|
}
|
|
98
121
|
async upsert(wdb, tableName, chunkSize) {
|
|
99
122
|
return this.upsertOrInsert(wdb, tableName, "upsert", chunkSize);
|
|
@@ -116,7 +139,7 @@ export class UpsertBuilder {
|
|
|
116
139
|
}
|
|
117
140
|
// 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출
|
|
118
141
|
const { references, refTables } = Array.from(this.tables).reduce((r, [, table])=>{
|
|
119
|
-
const reference = Array.from(table.references.values()).find((ref)=>ref.includes(tableName
|
|
142
|
+
const reference = Array.from(table.references.values()).find((ref)=>ref.includes(`${tableName}.`));
|
|
120
143
|
if (reference) {
|
|
121
144
|
r.references.push(reference);
|
|
122
145
|
r.refTables.push(table);
|
|
@@ -126,90 +149,216 @@ export class UpsertBuilder {
|
|
|
126
149
|
references: [],
|
|
127
150
|
refTables: []
|
|
128
151
|
});
|
|
129
|
-
const extractFields =
|
|
130
|
-
//
|
|
131
|
-
const
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
152
|
+
const extractFields = unique(references).map((reference)=>reference.split(".")[1]).filter((field)=>field !== undefined);
|
|
153
|
+
// 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)
|
|
154
|
+
const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);
|
|
155
|
+
if (hasCircular) {
|
|
156
|
+
throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);
|
|
157
|
+
}
|
|
158
|
+
// upsert 모드일 때 유니크 인덱스가 없으면 에러
|
|
159
|
+
if (mode === "upsert" && table.uniqueIndexes.length === 0) {
|
|
160
|
+
throw new Error(`${tableName}에 unique index가 정의되지 않아 upsert를 할 수 없습니다.`);
|
|
161
|
+
}
|
|
137
162
|
const uuidMap = new Map();
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
163
|
+
const allIds = [];
|
|
164
|
+
// 레벨별로 순차 처리
|
|
165
|
+
for (const levelRows of levels){
|
|
166
|
+
// 이전 레벨에서 얻은 ID로 자기 참조 해결
|
|
167
|
+
const resolvedRows = levelRows.map((row)=>{
|
|
168
|
+
const resolved = {
|
|
169
|
+
...row
|
|
170
|
+
};
|
|
171
|
+
for (const [key, value] of Object.entries(row)){
|
|
172
|
+
if (isRefField(value) && value.of === tableName) {
|
|
173
|
+
const parent = uuidMap.get(value.uuid);
|
|
174
|
+
if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);
|
|
175
|
+
resolved[key] = parent[value.use ?? "id"];
|
|
176
|
+
Naite.t("puri:ub-ref-resolved", {
|
|
177
|
+
tableName,
|
|
178
|
+
field: key,
|
|
179
|
+
from: {
|
|
180
|
+
of: value.of,
|
|
181
|
+
uuid: value.uuid,
|
|
182
|
+
use: value.use ?? "id"
|
|
183
|
+
},
|
|
184
|
+
to: resolved[key]
|
|
185
|
+
});
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
return resolved;
|
|
189
|
+
});
|
|
190
|
+
// 현재 레벨 upsert
|
|
191
|
+
const levelChunks = chunkSize ? chunk(resolvedRows, chunkSize) : [
|
|
192
|
+
resolvedRows
|
|
193
|
+
];
|
|
194
|
+
const selectFields = unique([
|
|
148
195
|
"uuid",
|
|
149
196
|
"id",
|
|
150
197
|
...extractFields
|
|
151
|
-
])
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
198
|
+
]);
|
|
199
|
+
for (const dataChunk of levelChunks){
|
|
200
|
+
if (dataChunk.length === 0) continue;
|
|
201
|
+
let resultRows;
|
|
202
|
+
if (mode === "insert") {
|
|
203
|
+
// INSERT 모드
|
|
204
|
+
await wdb.insert(dataChunk).into(tableName);
|
|
205
|
+
const uuids = dataChunk.map((r)=>r.uuid);
|
|
206
|
+
resultRows = await wdb(tableName).select(selectFields).whereIn("uuid", uuids);
|
|
207
|
+
} else {
|
|
208
|
+
// UPSERT 모드: onConflict로 중복 처리
|
|
209
|
+
const conflictColumns = table.uniqueIndexes[0].columns;
|
|
210
|
+
const updateColumns = Object.keys(dataChunk[0]).filter((col)=>col !== "uuid" && !conflictColumns.includes(col));
|
|
211
|
+
const query = wdb.insert(dataChunk).into(tableName).onConflict(conflictColumns);
|
|
212
|
+
// updateColumns 유무에 따라 ignore/merge 선택하고 RETURNING으로 결과 받기
|
|
213
|
+
if (updateColumns.length === 0) {
|
|
214
|
+
resultRows = await query.ignore().returning(selectFields);
|
|
215
|
+
} else {
|
|
216
|
+
resultRows = await query.merge(updateColumns).returning(selectFields);
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
// 양쪽 모드 공통 처리
|
|
220
|
+
for (const row of resultRows){
|
|
221
|
+
uuidMap.set(row.uuid, row);
|
|
222
|
+
allIds.push(row.id);
|
|
223
|
+
}
|
|
224
|
+
}
|
|
155
225
|
}
|
|
156
226
|
// 해당 테이블 참조를 실제 밸류로 변경
|
|
157
|
-
|
|
227
|
+
for (const table of refTables){
|
|
158
228
|
table.rows = table.rows.map((row)=>{
|
|
159
|
-
Object.keys(row)
|
|
229
|
+
for (const key of Object.keys(row)){
|
|
160
230
|
const prop = row[key];
|
|
161
231
|
if (isRefField(prop) && prop.of === tableName) {
|
|
162
232
|
const parent = uuidMap.get(prop.uuid);
|
|
163
|
-
if (parent
|
|
233
|
+
if (!parent) {
|
|
164
234
|
console.error(prop);
|
|
165
235
|
throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);
|
|
166
236
|
}
|
|
167
|
-
|
|
237
|
+
const resolvedValue = parent[prop.use ?? "id"];
|
|
238
|
+
row[key] = resolvedValue;
|
|
239
|
+
Naite.t("puri:ub-ref-resolved", {
|
|
240
|
+
tableName,
|
|
241
|
+
field: key,
|
|
242
|
+
from: {
|
|
243
|
+
of: prop.of,
|
|
244
|
+
uuid: prop.uuid,
|
|
245
|
+
use: prop.use ?? "id"
|
|
246
|
+
},
|
|
247
|
+
to: resolvedValue
|
|
248
|
+
});
|
|
168
249
|
}
|
|
169
|
-
}
|
|
250
|
+
}
|
|
170
251
|
return row;
|
|
171
252
|
});
|
|
172
|
-
});
|
|
173
|
-
const allIds = Array.from(uuidMap.values()).map((row)=>row.id);
|
|
174
|
-
// 자기 참조가 있는 경우 재귀적으로 upsert
|
|
175
|
-
if (selfRefRows.length > 0) {
|
|
176
|
-
// 처리된 데이터를 제외하고 다시 upsert
|
|
177
|
-
table.rows = selfRefRows;
|
|
178
|
-
const selfRefIds = await this.upsert(wdb, tableName, chunkSize);
|
|
179
|
-
allIds.push(...selfRefIds);
|
|
180
|
-
} else {
|
|
181
|
-
// 자기 참조가 없으면 해당 테이블의 데이터 초기화
|
|
182
|
-
table.rows = [];
|
|
183
|
-
table.references.clear();
|
|
184
|
-
table.uniquesMap.clear();
|
|
185
253
|
}
|
|
254
|
+
// 해당 테이블의 데이터 초기화
|
|
255
|
+
table.rows = [];
|
|
256
|
+
table.references.clear();
|
|
257
|
+
table.uniquesMap.clear();
|
|
258
|
+
Naite.t("puri:ub-upserted", {
|
|
259
|
+
tableName,
|
|
260
|
+
mode,
|
|
261
|
+
rowCount: allIds.length,
|
|
262
|
+
returnedIds: allIds
|
|
263
|
+
});
|
|
186
264
|
return allIds;
|
|
187
265
|
}
|
|
188
266
|
async updateBatch(wdb, tableName, options) {
|
|
189
|
-
options =
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
267
|
+
options = {
|
|
268
|
+
...options,
|
|
269
|
+
chunkSize: options?.chunkSize ?? 500,
|
|
270
|
+
where: options?.where ?? "id"
|
|
271
|
+
};
|
|
193
272
|
if (this.hasTable(tableName) === false) {
|
|
194
273
|
return;
|
|
195
274
|
}
|
|
196
275
|
const table = this.tables.get(tableName);
|
|
197
|
-
if (table
|
|
276
|
+
if (!table) {
|
|
277
|
+
throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);
|
|
278
|
+
} else if (table.rows.length === 0) {
|
|
198
279
|
return;
|
|
199
280
|
}
|
|
200
281
|
const whereColumns = Array.isArray(options.where) ? options.where : [
|
|
201
282
|
options.where ?? "id"
|
|
202
283
|
];
|
|
203
284
|
const rows = table.rows.map((_row)=>{
|
|
204
|
-
const { uuid, ...row } = _row;
|
|
285
|
+
const { uuid: _, ...row } = _row; // uuid 제외
|
|
205
286
|
return row;
|
|
206
287
|
});
|
|
207
288
|
await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);
|
|
289
|
+
Naite.t("puri:ub-batch-updated", {
|
|
290
|
+
tableName,
|
|
291
|
+
rowCount: rows.length,
|
|
292
|
+
whereColumns
|
|
293
|
+
});
|
|
208
294
|
// updateBatch 완료 후 처리된 데이터 제거
|
|
209
295
|
table.rows = [];
|
|
210
296
|
table.references.clear();
|
|
211
297
|
table.uniquesMap.clear();
|
|
212
298
|
}
|
|
299
|
+
// ============================================================================
|
|
300
|
+
// Private Helpers
|
|
301
|
+
// ============================================================================
|
|
302
|
+
/**
|
|
303
|
+
* rows를 의존성 순서에 따라 레벨별로 그룹화
|
|
304
|
+
* - 자기 참조 없는 경우 : 모든 rows가 Level 0
|
|
305
|
+
* - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화
|
|
306
|
+
*/ buildInsertLevels(rows, tableName) {
|
|
307
|
+
// 1. 자기 참조가 없으면 한 레벨로 처리
|
|
308
|
+
const hasSelfRef = rows.flatMap((row)=>Object.values(row)).some((value)=>isRefField(value) && value.of === tableName);
|
|
309
|
+
if (!hasSelfRef) return {
|
|
310
|
+
levels: [
|
|
311
|
+
rows
|
|
312
|
+
],
|
|
313
|
+
hasCircular: false
|
|
314
|
+
};
|
|
315
|
+
// 2. uuid → row 매핑 (중복 uuid 방지)
|
|
316
|
+
const rowByUuid = new Map();
|
|
317
|
+
for (const row of rows){
|
|
318
|
+
const uuid = row.uuid;
|
|
319
|
+
if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);
|
|
320
|
+
rowByUuid.set(uuid, row);
|
|
321
|
+
}
|
|
322
|
+
let pending = Array.from(rowByUuid.values());
|
|
323
|
+
const levels = [];
|
|
324
|
+
const inserted = new Set();
|
|
325
|
+
// 3. 레벨별 분류
|
|
326
|
+
while(pending.length > 0){
|
|
327
|
+
const currentLevel = [];
|
|
328
|
+
const nextPending = [];
|
|
329
|
+
for (const row of pending){
|
|
330
|
+
// 이 row가 참조하는 자기 참조들
|
|
331
|
+
const selfRefs = Object.values(row).filter((value)=>isRefField(value) && value.of === tableName);
|
|
332
|
+
// 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함
|
|
333
|
+
const canInsert = selfRefs.every((ref)=>{
|
|
334
|
+
if (!rowByUuid.has(ref.uuid)) {
|
|
335
|
+
throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);
|
|
336
|
+
}
|
|
337
|
+
return inserted.has(ref.uuid);
|
|
338
|
+
});
|
|
339
|
+
if (canInsert) {
|
|
340
|
+
currentLevel.push(row);
|
|
341
|
+
} else {
|
|
342
|
+
nextPending.push(row);
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
// 순환 참조 감지
|
|
346
|
+
if (currentLevel.length === 0) return {
|
|
347
|
+
levels: [],
|
|
348
|
+
hasCircular: true
|
|
349
|
+
};
|
|
350
|
+
// 레벨 확정 + inserted 갱신
|
|
351
|
+
levels.push(currentLevel);
|
|
352
|
+
for (const row of currentLevel){
|
|
353
|
+
inserted.add(row.uuid);
|
|
354
|
+
}
|
|
355
|
+
pending = nextPending;
|
|
356
|
+
}
|
|
357
|
+
return {
|
|
358
|
+
levels,
|
|
359
|
+
hasCircular: false
|
|
360
|
+
};
|
|
361
|
+
}
|
|
213
362
|
}
|
|
214
363
|
|
|
215
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/upsert-builder.ts"],"sourcesContent":["import { randomUUID } from \"crypto\";\nimport * as _ from \"lodash-es\";\nimport { Knex } from \"knex\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { nonNullable } from \"../utils/utils\";\nimport { RowWithId, batchUpdate } from \"./_batch_update\";\n\ntype TableData = {\n  references: Set<string>;\n  rows: any[];\n  uniqueIndexes: { name?: string; columns: string[] }[];\n  uniquesMap: Map<string, string>;\n};\nexport type UBRef = {\n  uuid: string;\n  of: string;\n  use?: string;\n};\nexport function isRefField(field: any): field is UBRef {\n  return (\n    field !== undefined &&\n    field !== null &&\n    field.of !== undefined &&\n    field.uuid !== undefined\n  );\n}\n\nexport class UpsertBuilder {\n  tables: Map<string, TableData>;\n  constructor() {\n    this.tables = new Map();\n  }\n\n  getTable(tableName: string): TableData {\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      const tableSpec = (() => {\n        try {\n          return EntityManager.getTableSpec(tableName);\n        } catch {\n          return null;\n        }\n      })();\n\n      this.tables.set(tableName, {\n        references: new Set(),\n        rows: [],\n        uniqueIndexes: tableSpec?.uniqueIndexes ?? [],\n        uniquesMap: new Map<string, string>(),\n      });\n    }\n\n    return this.tables.get(tableName)!;\n  }\n\n  hasTable(tableName: string): boolean {\n    return this.tables.has(tableName);\n  }\n\n  register<T extends string>(\n    tableName: string,\n    row: {\n      [key in T]?:\n        | UBRef\n        | string\n        | number\n        | boolean\n        | bigint\n        | null\n        | object\n        | unknown;\n    }\n  ): UBRef {\n    const table = this.getTable(tableName);\n\n    // 해당 테이블의 unique 인덱스를 순회하며 키 생성\n    const uniqueKeys = table.uniqueIndexes\n      .map((unqIndex) => {\n        const uniqueKeyArray = unqIndex.columns.map((unqCol) => {\n          const val = row[unqCol as keyof typeof row];\n          if (isRefField(val)) {\n            return val.uuid;\n          } else {\n            return row[unqCol as keyof typeof row] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입\n          }\n        });\n\n        // 값이 모두 null인 경우 키 생성 패스\n        if (uniqueKeyArray.length === 0) {\n          return null;\n        }\n        return uniqueKeyArray.join(\"---delimiter--\");\n      })\n      .filter(nonNullable);\n\n    // uuid 생성 로직\n    const uuid: string = (() => {\n      // 키를 순회하여 이미 존재하는 키가 있는지 확인\n      if (uniqueKeys.length > 0) {\n        for (const uniqueKey of uniqueKeys) {\n          if (table.uniquesMap.has(uniqueKey)) {\n            return table.uniquesMap.get(uniqueKey)!; // 이미 has 체크를 했으므로 undefined 불가능\n          }\n        }\n      }\n\n      // 찾을 수 없는 경우 생성\n      return randomUUID();\n    })();\n\n    // 모든 유니크키에 대해 유니크맵에 uuid 저장\n    if (uniqueKeys.length > 0) {\n      for (const uniqueKey of uniqueKeys) {\n        table.uniquesMap.set(uniqueKey, uuid);\n      }\n    }\n\n    // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가\n    // 이 정보를 나중에 치환할 때 사용\n    row = Object.keys(row).reduce((r, rowKey) => {\n      const rowValue = row[rowKey as keyof typeof row];\n\n      if (isRefField(rowValue)) {\n        rowValue.use ??= \"id\";\n        table.references.add(rowValue.of + \".\" + rowValue.use);\n        r[rowKey] = rowValue;\n      } else if (typeof rowValue === \"object\" && !(rowValue instanceof Date)) {\n        // object인 경우 JSON으로 변환\n        r[rowKey] = rowValue === null ? null : JSON.stringify(rowValue);\n      } else {\n        r[rowKey] = rowValue;\n      }\n      return r;\n    }, {} as any);\n\n    table.rows.push({\n      uuid,\n      ...row,\n    });\n\n    return {\n      of: tableName,\n      uuid: (row as { uuid?: string }).uuid ?? uuid,\n    };\n  }\n\n  async upsert(\n    wdb: Knex,\n    tableName: string,\n    chunkSize?: number\n  ): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"upsert\", chunkSize);\n  }\n  async insertOnly(\n    wdb: Knex,\n    tableName: string,\n    chunkSize?: number\n  ): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"insert\", chunkSize);\n  }\n\n  async upsertOrInsert(\n    wdb: Knex,\n    tableName: string,\n    mode: \"upsert\" | \"insert\",\n    chunkSize?: number\n  ): Promise<number[]> {\n    if (this.hasTable(tableName) === false) {\n      return [];\n    }\n\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);\n    } else if (table.rows.length === 0) {\n      throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);\n    }\n\n    if (\n      table.rows.some((row) =>\n        Object.entries(row).some(\n          ([, value]) => isRefField(value) && value.of !== tableName\n        )\n      )\n    ) {\n      throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);\n    }\n\n    // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출\n    const { references, refTables } = Array.from(this.tables).reduce(\n      (r, [, table]) => {\n        const reference = Array.from(table.references.values()).find((ref) =>\n          ref.includes(tableName + \".\")\n        );\n        if (reference) {\n          r.references.push(reference);\n          r.refTables.push(table);\n        }\n\n        return r;\n      },\n      {\n        references: [] as string[],\n        refTables: [] as TableData[],\n      }\n    );\n    const extractFields = _.uniq(references).map(\n      (reference) => reference.split(\".\")[1]\n    );\n\n    // 내부 참조 있는 경우 필터하여 분리\n    const groups = _.groupBy(table.rows, (row) =>\n      Object.entries(row).some(([, value]) => isRefField(value))\n        ? \"selfRef\"\n        : \"normal\"\n    );\n    const normalRows = groups.normal ?? [];\n    const selfRefRows = groups.selfRef ?? [];\n\n    const chunks = chunkSize ? _.chunk(normalRows, chunkSize) : [normalRows];\n    const uuidMap = new Map<string, any>();\n\n    for (const chunk of chunks) {\n      const q = wdb.insert(chunk).into(tableName);\n      if (mode === \"insert\") {\n        await q;\n      } else if (mode === \"upsert\") {\n        await q.onDuplicateUpdate.apply(q, Object.keys(normalRows[0]));\n      }\n\n      // upsert된 row들을 다시 조회하여 uuidMap에 저장\n      const uuids = chunk.map((row) => row.uuid);\n      const upsertedRows = await wdb(tableName)\n        .select(_.uniq([\"uuid\", \"id\", ...extractFields]))\n        .whereIn(\"uuid\", uuids);\n      upsertedRows.forEach((row: any) => {\n        uuidMap.set(row.uuid, row);\n      });\n    }\n\n    // 해당 테이블 참조를 실제 밸류로 변경\n    refTables.map((table) => {\n      table.rows = table.rows.map((row) => {\n        Object.keys(row).map((key) => {\n          const prop = row[key];\n          if (isRefField(prop) && prop.of === tableName) {\n            const parent = uuidMap.get(prop.uuid);\n            if (parent === undefined) {\n              console.error(prop);\n              throw new Error(\n                `존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`\n              );\n            }\n            row[key] = parent[prop.use ?? \"id\"];\n          }\n        });\n        return row;\n      });\n    });\n\n    const allIds = Array.from(uuidMap.values()).map((row) => row.id);\n\n    // 자기 참조가 있는 경우 재귀적으로 upsert\n    if (selfRefRows.length > 0) {\n      // 처리된 데이터를 제외하고 다시 upsert\n      table.rows = selfRefRows;\n      const selfRefIds = await this.upsert(wdb, tableName, chunkSize);\n      allIds.push(...selfRefIds);\n    } else {\n      // 자기 참조가 없으면 해당 테이블의 데이터 초기화\n      table.rows = [];\n      table.references.clear();\n      table.uniquesMap.clear();\n    }\n\n    return allIds;\n  }\n\n  async updateBatch(\n    wdb: Knex,\n    tableName: string,\n    options?: {\n      chunkSize?: number;\n      where?: string | string[];\n    }\n  ): Promise<void> {\n    options = _.defaults(options, {\n      chunkSize: 500,\n      where: \"id\",\n    });\n\n    if (this.hasTable(tableName) === false) {\n      return;\n    }\n    const table = this.tables.get(tableName)!;\n    if (table.rows.length === 0) {\n      return;\n    }\n\n    const whereColumns = Array.isArray(options.where)\n      ? options.where\n      : [options.where ?? \"id\"];\n    const rows = table.rows.map((_row) => {\n      const { uuid, ...row } = _row;\n      return row as RowWithId<string>;\n    });\n\n    await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);\n\n    // updateBatch 완료 후 처리된 데이터 제거\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n  }\n}\n"],"names":["randomUUID","_","EntityManager","nonNullable","batchUpdate","isRefField","field","undefined","of","uuid","UpsertBuilder","tables","Map","getTable","tableName","table","get","tableSpec","getTableSpec","set","references","Set","rows","uniqueIndexes","uniquesMap","hasTable","has","register","row","uniqueKeys","map","unqIndex","uniqueKeyArray","columns","unqCol","val","length","join","filter","uniqueKey","Object","keys","reduce","r","rowKey","rowValue","use","add","Date","JSON","stringify","push","upsert","wdb","chunkSize","upsertOrInsert","insertOnly","mode","Error","some","entries","value","refTables","Array","from","reference","values","find","ref","includes","extractFields","uniq","split","groups","groupBy","normalRows","normal","selfRefRows","selfRef","chunks","chunk","uuidMap","q","insert","into","onDuplicateUpdate","apply","uuids","upsertedRows","select","whereIn","forEach","key","prop","parent","console","error","allIds","id","selfRefIds","clear","updateBatch","options","defaults","where","whereColumns","isArray","_row"],"mappings":"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"}
|
|
364
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/upsert-builder.ts"],"sourcesContent":["import { randomUUID } from \"crypto\";\nimport type { Knex } from \"knex\";\nimport { unique } from \"radashi\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport { assertDefined, chunk, nonNullable } from \"../utils/utils\";\nimport { batchUpdate, type RowWithId } from \"./_batch_update\";\n\ntype TableData = {\n  references: Set<string>;\n  rows: Record<string, unknown>[];\n  uniqueIndexes: { name?: string; columns: string[] }[];\n  uniquesMap: Map<string, string>;\n};\nexport type UBRef = {\n  uuid: string;\n  of: string;\n  use?: string;\n};\nexport function isRefField(field: unknown): field is UBRef {\n  return (\n    field !== undefined &&\n    field !== null &&\n    (field as UBRef)?.of !== undefined &&\n    (field as UBRef)?.uuid !== undefined\n  );\n}\n\nexport class UpsertBuilder {\n  tables: Map<string, TableData>;\n  constructor() {\n    this.tables = new Map();\n  }\n\n  getTable(tableName: string): TableData {\n    const table = this.tables.get(tableName);\n    if (table) {\n      return table;\n    }\n\n    const tableSpec = (() => {\n      try {\n        return EntityManager.getTableSpec(tableName);\n      } catch {\n        return null;\n      }\n    })();\n\n    const tableData = {\n      references: new Set<string>(),\n      rows: [],\n      uniqueIndexes: tableSpec?.uniqueIndexes ?? [],\n      uniquesMap: new Map<string, string>(),\n    };\n    this.tables.set(tableName, tableData);\n    return tableData;\n  }\n\n  hasTable(tableName: string): boolean {\n    return this.tables.has(tableName);\n  }\n\n  register<T extends string>(\n    tableName: string,\n    row: {\n      [key in T]?: UBRef | string | number | boolean | bigint | null | object | unknown;\n    },\n  ): UBRef {\n    const table = this.getTable(tableName);\n\n    // 해당 테이블의 unique 인덱스를 순회하며 키 생성\n    const uniqueKeys = table.uniqueIndexes\n      .map((unqIndex) => {\n        const uniqueKeyArray = unqIndex.columns.map((unqCol) => {\n          const val = row[unqCol as keyof typeof row];\n          if (isRefField(val)) {\n            return val.uuid;\n          } else {\n            return row[unqCol as keyof typeof row] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입\n          }\n        });\n\n        // 값이 모두 null인 경우 키 생성 패스\n        if (uniqueKeyArray.length === 0) {\n          return null;\n        }\n        return uniqueKeyArray.join(\"---delimiter--\");\n      })\n      .filter(nonNullable);\n\n    // uuid 생성 로직\n    const { uuid, isReused } = (() => {\n      // 키를 순회하여 이미 존재하는 키가 있는지 확인\n      if (uniqueKeys.length > 0) {\n        for (const uniqueKey of uniqueKeys) {\n          if (table.uniquesMap.has(uniqueKey)) {\n            return {\n              uuid: assertDefined(table.uniquesMap.get(uniqueKey), \"Unique key not found\"),\n              isReused: true,\n            };\n          }\n        }\n      }\n\n      // 찾을 수 없는 경우 생성\n      return { uuid: randomUUID(), isReused: false };\n    })();\n\n    // 모든 유니크키에 대해 유니크맵에 uuid 저장\n    if (uniqueKeys.length > 0) {\n      for (const uniqueKey of uniqueKeys) {\n        table.uniquesMap.set(uniqueKey, uuid);\n      }\n    }\n\n    // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가\n    // 이 정보를 나중에 치환할 때 사용\n    row = Object.fromEntries(\n      Object.entries(row).map(([rowKey, rowValue]) => {\n        if (isRefField(rowValue)) {\n          rowValue.use ??= \"id\";\n          table.references.add(`${rowValue.of}.${rowValue.use}`);\n          return [rowKey, rowValue];\n        } else if (typeof rowValue === \"object\" && !(rowValue instanceof Date)) {\n          // object인 경우 JSON으로 변환\n          return [rowKey, rowValue === null ? null : JSON.stringify(rowValue)];\n        } else {\n          return [rowKey, rowValue];\n        }\n      }),\n    ) as { [key in T]?: unknown };\n\n    table.rows.push({\n      uuid,\n      ...row,\n    });\n\n    const result: UBRef = {\n      of: tableName,\n      uuid: (row as { uuid?: string }).uuid ?? uuid,\n    };\n\n    Naite.t(\"puri:ub-register\", {\n      tableName,\n      uuid: result.uuid,\n      isUuidReused: isReused,\n      row,\n    });\n\n    return result;\n  }\n\n  async upsert(wdb: Knex, tableName: string, chunkSize?: number): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"upsert\", chunkSize);\n  }\n  async insertOnly(wdb: Knex, tableName: string, chunkSize?: number): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"insert\", chunkSize);\n  }\n\n  async upsertOrInsert(\n    wdb: Knex,\n    tableName: string,\n    mode: \"upsert\" | \"insert\",\n    chunkSize?: number,\n  ): Promise<number[]> {\n    if (this.hasTable(tableName) === false) {\n      return [];\n    }\n\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);\n    } else if (table.rows.length === 0) {\n      throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);\n    }\n\n    if (\n      table.rows.some((row) =>\n        Object.entries(row).some(([, value]) => isRefField(value) && value.of !== tableName),\n      )\n    ) {\n      throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);\n    }\n\n    // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출\n    const { references, refTables } = Array.from(this.tables).reduce(\n      (r, [, table]) => {\n        const reference = Array.from(table.references.values()).find((ref) =>\n          ref.includes(`${tableName}.`),\n        );\n        if (reference) {\n          r.references.push(reference);\n          r.refTables.push(table);\n        }\n\n        return r;\n      },\n      {\n        references: [] as string[],\n        refTables: [] as TableData[],\n      },\n    );\n    const extractFields = unique(references)\n      .map((reference) => reference.split(\".\")[1])\n      .filter((field): field is string => field !== undefined);\n\n    // 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)\n    const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);\n\n    if (hasCircular) {\n      throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);\n    }\n\n    // upsert 모드일 때 유니크 인덱스가 없으면 에러\n    if (mode === \"upsert\" && table.uniqueIndexes.length === 0) {\n      throw new Error(`${tableName}에 unique index가 정의되지 않아 upsert를 할 수 없습니다.`);\n    }\n\n    const uuidMap = new Map<string, unknown>();\n    const allIds: number[] = [];\n\n    // 레벨별로 순차 처리\n    for (const levelRows of levels) {\n      // 이전 레벨에서 얻은 ID로 자기 참조 해결\n      const resolvedRows = levelRows.map((row) => {\n        const resolved = { ...row };\n        for (const [key, value] of Object.entries(row)) {\n          if (isRefField(value) && value.of === tableName) {\n            const parent = uuidMap.get(value.uuid);\n\n            if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);\n\n            resolved[key] = (parent as Record<string, unknown>)[value.use ?? \"id\"];\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: value.of, uuid: value.uuid, use: value.use ?? \"id\" },\n              to: resolved[key],\n            });\n          }\n        }\n        return resolved;\n      });\n\n      // 현재 레벨 upsert\n      const levelChunks = chunkSize ? chunk(resolvedRows, chunkSize) : [resolvedRows];\n      const selectFields = unique([\"uuid\", \"id\", ...extractFields]);\n\n      for (const dataChunk of levelChunks) {\n        if (dataChunk.length === 0) continue;\n\n        let resultRows: { uuid: string; id: number; [key: string]: unknown }[];\n\n        if (mode === \"insert\") {\n          // INSERT 모드\n          await wdb.insert(dataChunk).into(tableName);\n\n          const uuids = dataChunk.map((r) => r.uuid);\n          resultRows = await wdb(tableName)\n            .select(selectFields)\n            .whereIn(\"uuid\", uuids as readonly string[]);\n        } else {\n          // UPSERT 모드: onConflict로 중복 처리\n          const conflictColumns = table.uniqueIndexes[0].columns;\n          const updateColumns = Object.keys(dataChunk[0]).filter(\n            (col) => col !== \"uuid\" && !conflictColumns.includes(col),\n          );\n\n          const query = wdb.insert(dataChunk).into(tableName).onConflict(conflictColumns);\n\n          // updateColumns 유무에 따라 ignore/merge 선택하고 RETURNING으로 결과 받기\n          if (updateColumns.length === 0) {\n            resultRows = await query.ignore().returning(selectFields);\n          } else {\n            resultRows = await query.merge(updateColumns).returning(selectFields);\n          }\n        }\n\n        // 양쪽 모드 공통 처리\n        for (const row of resultRows) {\n          uuidMap.set(row.uuid, row);\n          allIds.push(row.id);\n        }\n      }\n    }\n\n    // 해당 테이블 참조를 실제 밸류로 변경\n    for (const table of refTables) {\n      table.rows = table.rows.map((row) => {\n        for (const key of Object.keys(row)) {\n          const prop = row[key];\n          if (isRefField(prop) && prop.of === tableName) {\n            const parent = uuidMap.get(prop.uuid);\n            if (!parent) {\n              console.error(prop);\n              throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);\n            }\n            const resolvedValue = (parent as Record<string, unknown>)[prop.use ?? \"id\"];\n            row[key] = resolvedValue;\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: prop.of, uuid: prop.uuid, use: prop.use ?? \"id\" },\n              to: resolvedValue,\n            });\n          }\n        }\n        return row;\n      });\n    }\n\n    // 해당 테이블의 데이터 초기화\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n\n    Naite.t(\"puri:ub-upserted\", {\n      tableName,\n      mode,\n      rowCount: allIds.length,\n      returnedIds: allIds,\n    });\n\n    return allIds;\n  }\n\n  async updateBatch(\n    wdb: Knex,\n    tableName: string,\n    options?: {\n      chunkSize?: number;\n      where?: string | string[];\n    },\n  ): Promise<void> {\n    options = {\n      ...options,\n      chunkSize: options?.chunkSize ?? 500,\n      where: options?.where ?? \"id\",\n    };\n\n    if (this.hasTable(tableName) === false) {\n      return;\n    }\n    const table = this.tables.get(tableName);\n    if (!table) {\n      throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);\n    } else if (table.rows.length === 0) {\n      return;\n    }\n\n    const whereColumns = Array.isArray(options.where) ? options.where : [options.where ?? \"id\"];\n    const rows = table.rows.map((_row) => {\n      const { uuid: _, ...row } = _row; // uuid 제외\n      return row as RowWithId<string>;\n    });\n\n    await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);\n\n    Naite.t(\"puri:ub-batch-updated\", {\n      tableName,\n      rowCount: rows.length,\n      whereColumns,\n    });\n\n    // updateBatch 완료 후 처리된 데이터 제거\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n  }\n\n  // ============================================================================\n  // Private Helpers\n  // ============================================================================\n\n  /**\n   * rows를 의존성 순서에 따라 레벨별로 그룹화\n   * - 자기 참조 없는 경우 : 모든 rows가 Level 0\n   * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화\n   */\n  private buildInsertLevels(\n    rows: Record<string, unknown>[],\n    tableName: string,\n  ): { levels: Record<string, unknown>[][]; hasCircular: boolean } {\n    // 1. 자기 참조가 없으면 한 레벨로 처리\n    const hasSelfRef = rows\n      .flatMap((row) => Object.values(row))\n      .some((value) => isRefField(value) && value.of === tableName);\n    if (!hasSelfRef) return { levels: [rows], hasCircular: false };\n\n    // 2. uuid → row 매핑 (중복 uuid 방지)\n    const rowByUuid = new Map<string, Record<string, unknown>>();\n    for (const row of rows) {\n      const uuid = row.uuid as string | undefined;\n      if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);\n      rowByUuid.set(uuid, row);\n    }\n\n    let pending = Array.from(rowByUuid.values());\n    const levels: Record<string, unknown>[][] = [];\n    const inserted = new Set<string>();\n\n    // 3. 레벨별 분류\n    while (pending.length > 0) {\n      const currentLevel: Record<string, unknown>[] = [];\n      const nextPending: Record<string, unknown>[] = [];\n\n      for (const row of pending) {\n        // 이 row가 참조하는 자기 참조들\n        const selfRefs = Object.values(row).filter(\n          (value) => isRefField(value) && value.of === tableName,\n        ) as UBRef[];\n\n        // 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함\n        const canInsert = selfRefs.every((ref) => {\n          if (!rowByUuid.has(ref.uuid)) {\n            throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);\n          }\n          return inserted.has(ref.uuid);\n        });\n\n        if (canInsert) {\n          currentLevel.push(row);\n        } else {\n          nextPending.push(row);\n        }\n      }\n\n      // 순환 참조 감지\n      if (currentLevel.length === 0) return { levels: [], hasCircular: true };\n\n      // 레벨 확정 + inserted 갱신\n      levels.push(currentLevel);\n      for (const row of currentLevel) {\n        inserted.add(row.uuid as string);\n      }\n\n      pending = nextPending;\n    }\n\n    return { levels, hasCircular: false };\n  }\n}\n"],"names":["randomUUID","unique","EntityManager","Naite","assertDefined","chunk","nonNullable","batchUpdate","isRefField","field","undefined","of","uuid","UpsertBuilder","tables","Map","getTable","tableName","table","get","tableSpec","getTableSpec","tableData","references","Set","rows","uniqueIndexes","uniquesMap","set","hasTable","has","register","row","uniqueKeys","map","unqIndex","uniqueKeyArray","columns","unqCol","val","length","join","filter","isReused","uniqueKey","Object","fromEntries","entries","rowKey","rowValue","use","add","Date","JSON","stringify","push","result","t","isUuidReused","upsert","wdb","chunkSize","upsertOrInsert","insertOnly","mode","Error","some","value","refTables","Array","from","reduce","r","reference","values","find","ref","includes","extractFields","split","levels","hasCircular","buildInsertLevels","uuidMap","allIds","levelRows","resolvedRows","resolved","key","parent","to","levelChunks","selectFields","dataChunk","resultRows","insert","into","uuids","select","whereIn","conflictColumns","updateColumns","keys","col","query","onConflict","ignore","returning","merge","id","prop","console","error","resolvedValue","clear","rowCount","returnedIds","updateBatch","options","where","whereColumns","isArray","_row","_","hasSelfRef","flatMap","rowByUuid","pending","inserted","currentLevel","nextPending","selfRefs","canInsert","every"],"mappings":"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"}
|