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
package/dist/api/sonamu.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"sonamu.d.ts","sourceRoot":"","sources":["../../src/api/sonamu.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"sonamu.d.ts","sourceRoot":"","sources":["../../src/api/sonamu.ts"],"names":[],"mappings":"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"}
|
package/dist/api/sonamu.js
CHANGED
|
@@ -1,29 +1,7 @@
|
|
|
1
|
+
import assert from "assert";
|
|
1
2
|
import { AsyncLocalStorage } from "async_hooks";
|
|
2
|
-
import chalk from "chalk";
|
|
3
|
-
import fastify from "fastify";
|
|
4
|
-
import { readFile } from "node:fs/promises";
|
|
5
3
|
import path from "path";
|
|
6
|
-
import {
|
|
7
|
-
import chokidar from "chokidar";
|
|
8
|
-
import { formatInTimeZone } from "date-fns-tz";
|
|
9
|
-
import { ZodError } from "zod";
|
|
10
|
-
import { DB } from "../database/db.js";
|
|
11
|
-
import { attachOnDuplicateUpdate } from "../database/knex-plugins/knex-on-duplicate-update.js";
|
|
12
|
-
import { BadRequestException, NotFoundException } from "../exceptions/so-exceptions.js";
|
|
13
|
-
import { createSSEFactory } from "../stream/sse.js";
|
|
14
|
-
import { ApiParamType } from "../types/types.js";
|
|
15
|
-
import { isLocal, isTest } from "../utils/controller.js";
|
|
16
|
-
import { findApiRootPath } from "../utils/utils.js";
|
|
17
|
-
import { humanizeZodError } from "../utils/zod-error.js";
|
|
18
|
-
import { fastifyCaster } from "./caster.js";
|
|
19
|
-
import { getZodObjectFromApi } from "./code-converters.js";
|
|
20
|
-
import fastifyPassport from "@fastify/passport";
|
|
21
|
-
import { loadConfig } from "./config.js";
|
|
22
|
-
import { isHotReloadServer } from "../utils/esm-utils.js";
|
|
23
|
-
import { Template } from "../template/index.js";
|
|
24
|
-
import assert from "assert";
|
|
25
|
-
import { centerText } from "../utils/console-util.js";
|
|
26
|
-
import { BaseModel } from "../database/base-model.js";
|
|
4
|
+
import { Naite } from "../naite/naite.js";
|
|
27
5
|
class SonamuClass {
|
|
28
6
|
isInitialized = false;
|
|
29
7
|
asyncLocalStorage = new AsyncLocalStorage();
|
|
@@ -40,6 +18,7 @@ class SonamuClass {
|
|
|
40
18
|
reply: null,
|
|
41
19
|
headers: {},
|
|
42
20
|
createSSE: ()=>{},
|
|
21
|
+
// biome-ignore lint/suspicious/noExplicitAny: 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴
|
|
43
22
|
naiteStore: new Map()
|
|
44
23
|
};
|
|
45
24
|
} else {
|
|
@@ -122,18 +101,29 @@ class SonamuClass {
|
|
|
122
101
|
if (this.isInitialized) {
|
|
123
102
|
return;
|
|
124
103
|
}
|
|
125
|
-
!doSilent
|
|
104
|
+
if (!doSilent) {
|
|
105
|
+
const chalk = (await import("chalk")).default;
|
|
106
|
+
console.time(chalk.cyan(`Sonamu.init${forTesting ? " for testing" : ""}`));
|
|
107
|
+
}
|
|
126
108
|
// API 루트 패스
|
|
109
|
+
const { findApiRootPath } = await import("../utils/utils.js");
|
|
127
110
|
this.apiRootPath = apiRootPath ?? findApiRootPath();
|
|
111
|
+
const { loadConfig } = await import("./config.js");
|
|
128
112
|
this.config = await loadConfig(this.apiRootPath);
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
113
|
+
// sonamu.config.ts 기본값 설정
|
|
114
|
+
this.config.database.database = this.config.database.database ?? "postgresql";
|
|
115
|
+
if (process.env.ANTHROPIC_API_KEY) {
|
|
116
|
+
this.secrets = {
|
|
117
|
+
anthropic_api_key: process.env.ANTHROPIC_API_KEY
|
|
118
|
+
};
|
|
132
119
|
}
|
|
133
120
|
// DB 로드
|
|
121
|
+
const { DB } = await import("../database/db.js");
|
|
134
122
|
this.dbConfig = DB.generateDBConfig(this.config.database);
|
|
135
|
-
|
|
136
|
-
|
|
123
|
+
if (!doSilent) {
|
|
124
|
+
const chalk = (await import("chalk")).default;
|
|
125
|
+
console.log(chalk.green("DB Config Loaded!"));
|
|
126
|
+
}
|
|
137
127
|
// 테스팅인 경우 엔티티 로드 & 싱크 없이 중단
|
|
138
128
|
if (forTesting) {
|
|
139
129
|
this.isInitialized = true;
|
|
@@ -149,20 +139,31 @@ class SonamuClass {
|
|
|
149
139
|
await this.syncer.autoloadTypes();
|
|
150
140
|
await this.syncer.autoloadModels();
|
|
151
141
|
await this.syncer.autoloadApis();
|
|
152
|
-
await
|
|
142
|
+
const { TemplateManager } = await import("../template/index.js");
|
|
143
|
+
await TemplateManager.autoload();
|
|
144
|
+
const { isLocal, isTest } = await import("../utils/controller.js");
|
|
145
|
+
if (isLocal()) {
|
|
146
|
+
// 로컬에서는 코드 생성을 위해 Biome 셋업이 필요함 (현재 apiRootPath 전달하여 실행)
|
|
147
|
+
(await import("../utils/formatter.js")).setupBiome(this.apiRootPath);
|
|
148
|
+
}
|
|
149
|
+
const { isHotReloadServer } = await import("../utils/controller.js");
|
|
153
150
|
if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {
|
|
154
151
|
await this.syncer.sync();
|
|
155
|
-
this.startWatcher();
|
|
152
|
+
await this.startWatcher();
|
|
156
153
|
this.syncer.syncUI();
|
|
157
154
|
}
|
|
158
155
|
this.isInitialized = true;
|
|
159
|
-
!doSilent
|
|
156
|
+
if (!doSilent) {
|
|
157
|
+
const chalk = (await import("chalk")).default;
|
|
158
|
+
console.timeEnd(chalk.cyan("Sonamu.init"));
|
|
159
|
+
}
|
|
160
160
|
}
|
|
161
161
|
async createServer(initOptions) {
|
|
162
162
|
if (this.isInitialized === false) {
|
|
163
163
|
await this.init(initOptions?.doSilent, initOptions?.enableSync);
|
|
164
164
|
}
|
|
165
165
|
const options = this.config.server;
|
|
166
|
+
const fastify = (await import("fastify")).default;
|
|
166
167
|
const server = fastify(options.fastify);
|
|
167
168
|
this.server = server;
|
|
168
169
|
// Storage 설정 저장
|
|
@@ -171,13 +172,13 @@ class SonamuClass {
|
|
|
171
172
|
}
|
|
172
173
|
// 플러그인 등록
|
|
173
174
|
if (options.plugins) {
|
|
174
|
-
this.registerPlugins(server, options.plugins);
|
|
175
|
+
await this.registerPlugins(server, options.plugins);
|
|
175
176
|
}
|
|
176
177
|
if (options.auth) {
|
|
177
178
|
if (!options.plugins?.session) {
|
|
178
179
|
throw new Error("Auth requires session plugin. Please add plugins.session configuration.");
|
|
179
180
|
}
|
|
180
|
-
this.registerAuth(server, options.auth);
|
|
181
|
+
await this.registerAuth(server, options.auth);
|
|
181
182
|
}
|
|
182
183
|
// API 라우팅 설정
|
|
183
184
|
await this.withFastify(server, options.apiConfig, {
|
|
@@ -196,9 +197,16 @@ class SonamuClass {
|
|
|
196
197
|
// timezone 설정
|
|
197
198
|
const timezone = this.config.api.timezone;
|
|
198
199
|
if (timezone) {
|
|
199
|
-
|
|
200
|
+
// 타임존에 맞게 응답 날짜 스트링을 변환해주어야 합니다.
|
|
201
|
+
// 가령 timezone이 "Asia/Seoul" 이면
|
|
202
|
+
// "2025-11-21T00:00:00.000Z" 를 "2025-11-21T09:00:00+09:00" 으로 변환해주어야 합니다.
|
|
203
|
+
const { formatInTimeZone } = await import("date-fns-tz");
|
|
200
204
|
// ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)
|
|
201
205
|
const ISO_DATE_REGEX = /^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(\.\d{3})?Z$/;
|
|
206
|
+
// T를 둘러싼 작은따옴표가 없다면 "2025-11-19176354618900018:56:29+09:00"와 같은 결과가 나옵니다.
|
|
207
|
+
// 이는 date-fns 특입니다.
|
|
208
|
+
// 이렇게 해도 괜찮습니다. "2025-11-19T18:56:29+09:00" 모양으로 잘 나옵니다.
|
|
209
|
+
const DATE_FORMAT = "yyyy-MM-dd'T'HH:mm:ssXXX";
|
|
202
210
|
server.setReplySerializer((payload)=>{
|
|
203
211
|
return JSON.stringify(payload, (_key, value)=>{
|
|
204
212
|
if (typeof value === "string" && ISO_DATE_REGEX.test(value)) {
|
|
@@ -207,7 +215,10 @@ class SonamuClass {
|
|
|
207
215
|
return value;
|
|
208
216
|
});
|
|
209
217
|
});
|
|
210
|
-
!options?.doSilent
|
|
218
|
+
if (!options?.doSilent) {
|
|
219
|
+
const chalk = (await import("chalk")).default;
|
|
220
|
+
console.log(chalk.green(`Timezone set to ${timezone}`));
|
|
221
|
+
}
|
|
211
222
|
}
|
|
212
223
|
// 전체 라우팅 리스트
|
|
213
224
|
server.get(`${this.config.api.route.prefix}/routes`, async (_request, _reply)=>{
|
|
@@ -218,42 +229,49 @@ class SonamuClass {
|
|
|
218
229
|
return "ok";
|
|
219
230
|
});
|
|
220
231
|
// API 라우팅 (로컬HMR 상태와 구분)
|
|
232
|
+
const { isLocal } = await import("../utils/controller.js");
|
|
221
233
|
if (isLocal()) {
|
|
222
|
-
server.all("*", (request, reply)=>{
|
|
234
|
+
server.all("*", async (request, reply)=>{
|
|
223
235
|
const found = this.syncer.apis.find((api)=>this.config.api.route.prefix + api.path === request.url.split("?")[0] && (api.options.httpMethod ?? "GET") === request.method.toUpperCase());
|
|
224
236
|
if (found) {
|
|
225
237
|
return this.getApiHandler(found, config)(request, reply);
|
|
226
238
|
}
|
|
239
|
+
const { NotFoundException } = await import("../exceptions/so-exceptions.js");
|
|
227
240
|
throw new NotFoundException("존재하지 않는 API 접근입니다.");
|
|
228
241
|
});
|
|
229
242
|
} else {
|
|
230
|
-
this.syncer.apis
|
|
243
|
+
for (const api of this.syncer.apis){
|
|
231
244
|
// model
|
|
232
245
|
if (this.syncer.models[api.modelName] === undefined) {
|
|
233
246
|
throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);
|
|
234
247
|
}
|
|
235
248
|
// route
|
|
236
249
|
server.route({
|
|
237
|
-
method: api.options.httpMethod,
|
|
250
|
+
method: api.options.httpMethod ?? "GET",
|
|
238
251
|
url: this.config.api.route.prefix + api.path,
|
|
239
252
|
handler: this.getApiHandler(api, config)
|
|
240
253
|
}); // END server.route
|
|
241
|
-
}
|
|
254
|
+
}
|
|
242
255
|
}
|
|
243
256
|
}
|
|
244
257
|
getApiHandler(api, config) {
|
|
245
258
|
return async (request, reply)=>{
|
|
246
259
|
(api.options.guards ?? []).every((guard)=>config.guardHandler(guard, request, api));
|
|
247
260
|
// 파라미터 정보로 zod 스키마 빌드
|
|
261
|
+
const { getZodObjectFromApi } = await import("./code-converters.js");
|
|
248
262
|
const ReqType = getZodObjectFromApi(api, this.syncer.types);
|
|
249
263
|
// request 파싱
|
|
250
264
|
const which = api.options.httpMethod === "GET" ? "query" : "body";
|
|
251
265
|
let reqBody;
|
|
252
266
|
try {
|
|
267
|
+
const { fastifyCaster } = await import("./caster.js");
|
|
253
268
|
reqBody = fastifyCaster(ReqType).parse(request[which] ?? {});
|
|
254
269
|
} catch (e) {
|
|
270
|
+
const { ZodError } = await import("zod");
|
|
255
271
|
if (e instanceof ZodError) {
|
|
272
|
+
const { humanizeZodError } = await import("../utils/zod-error.js");
|
|
256
273
|
const messages = humanizeZodError(e).map((issue)=>issue.message).join(" ");
|
|
274
|
+
const { BadRequestException } = await import("../exceptions/so-exceptions.js");
|
|
257
275
|
throw new BadRequestException(messages, {
|
|
258
276
|
zodError: e
|
|
259
277
|
});
|
|
@@ -299,6 +317,7 @@ class SonamuClass {
|
|
|
299
317
|
return cachedData;
|
|
300
318
|
}
|
|
301
319
|
// createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.
|
|
320
|
+
const { createSSEFactory } = await import("../stream/sse.js");
|
|
302
321
|
const createSSE = ((_request, _reply, _events)=>createSSEFactory(_request.socket, _reply, _events)).bind(null, request, reply);
|
|
303
322
|
const context = {
|
|
304
323
|
...await Promise.resolve(config.contextProvider({
|
|
@@ -306,7 +325,7 @@ class SonamuClass {
|
|
|
306
325
|
reply,
|
|
307
326
|
headers: request.headers,
|
|
308
327
|
createSSE,
|
|
309
|
-
naiteStore:
|
|
328
|
+
naiteStore: Naite.createStore(),
|
|
310
329
|
// auth
|
|
311
330
|
user: request.user ?? null,
|
|
312
331
|
passport: {
|
|
@@ -319,6 +338,8 @@ class SonamuClass {
|
|
|
319
338
|
return this.asyncLocalStorage.run({
|
|
320
339
|
context
|
|
321
340
|
}, async ()=>{
|
|
341
|
+
const { ApiParamType } = await import("../types/types.js");
|
|
342
|
+
// biome-ignore lint/suspicious/noExplicitAny: model은 모델 인스턴스이므로 메서드 호출 가능
|
|
322
343
|
const result = await model[api.methodName].apply(model, api.parameters.map((param)=>{
|
|
323
344
|
// Context 인젝션
|
|
324
345
|
if (ApiParamType.isContext(param.type)) {
|
|
@@ -336,11 +357,12 @@ class SonamuClass {
|
|
|
336
357
|
});
|
|
337
358
|
};
|
|
338
359
|
}
|
|
339
|
-
startWatcher() {
|
|
360
|
+
async startWatcher() {
|
|
340
361
|
const watchPath = [
|
|
341
362
|
path.join(this.apiRootPath, "src"),
|
|
342
363
|
path.join(this.apiRootPath, "sonamu.config.ts")
|
|
343
364
|
];
|
|
365
|
+
const chokidar = (await import("chokidar")).default;
|
|
344
366
|
this.watcher = chokidar.watch(watchPath, {
|
|
345
367
|
ignored: (path, stats)=>!!stats?.isFile() && !path.endsWith(".ts") && !path.endsWith(".json"),
|
|
346
368
|
persistent: true,
|
|
@@ -357,6 +379,7 @@ class SonamuClass {
|
|
|
357
379
|
const isConfigTs = filePath === path.join(this.apiRootPath, "sonamu.config.ts");
|
|
358
380
|
if (isConfigTs) {
|
|
359
381
|
const relativePath = filePath.replace(this.apiRootPath, "api");
|
|
382
|
+
const chalk = (await import("chalk")).default;
|
|
360
383
|
console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`));
|
|
361
384
|
process.kill(process.pid, "SIGUSR2");
|
|
362
385
|
return;
|
|
@@ -377,7 +400,7 @@ class SonamuClass {
|
|
|
377
400
|
await this.destroy();
|
|
378
401
|
}
|
|
379
402
|
}
|
|
380
|
-
registerPlugins(server, plugins) {
|
|
403
|
+
async registerPlugins(server, plugins) {
|
|
381
404
|
if (!plugins) {
|
|
382
405
|
return;
|
|
383
406
|
}
|
|
@@ -390,23 +413,25 @@ class SonamuClass {
|
|
|
390
413
|
static: "@fastify/static",
|
|
391
414
|
session: "@fastify/secure-session"
|
|
392
415
|
};
|
|
393
|
-
const registerPlugin = (key, pluginName)=>{
|
|
416
|
+
const registerPlugin = async (key, pluginName)=>{
|
|
394
417
|
const option = plugins[key];
|
|
395
418
|
if (!option) return;
|
|
396
419
|
if (option === true) {
|
|
397
|
-
server.register(import(pluginName));
|
|
420
|
+
server.register((await import(pluginName)).default);
|
|
398
421
|
} else {
|
|
399
|
-
server.register(import(pluginName), option);
|
|
422
|
+
server.register((await import(pluginName)).default, option);
|
|
400
423
|
}
|
|
401
424
|
};
|
|
402
|
-
|
|
403
|
-
registerPlugin(key, pluginName);
|
|
404
|
-
}
|
|
425
|
+
for (const [key, pluginName] of Object.entries(pluginsModules)){
|
|
426
|
+
await registerPlugin(key, pluginName);
|
|
427
|
+
}
|
|
405
428
|
if (plugins.custom) {
|
|
406
429
|
plugins.custom(server);
|
|
407
430
|
}
|
|
408
431
|
}
|
|
409
432
|
async registerAuth(server, options) {
|
|
433
|
+
// await import("fastify");
|
|
434
|
+
const fastifyPassport = (await import("@fastify/passport")).default;
|
|
410
435
|
server.register(fastifyPassport.initialize());
|
|
411
436
|
server.register(fastifyPassport.secureSession());
|
|
412
437
|
if (typeof options === "boolean") {
|
|
@@ -444,6 +469,7 @@ class SonamuClass {
|
|
|
444
469
|
}).then(async ()=>{
|
|
445
470
|
await options.lifecycle?.onStart?.(server);
|
|
446
471
|
}).catch(async (err)=>{
|
|
472
|
+
const chalk = (await import("chalk")).default;
|
|
447
473
|
console.error(chalk.red("Failed to start server:", err));
|
|
448
474
|
await shutdown();
|
|
449
475
|
});
|
|
@@ -455,6 +481,7 @@ class SonamuClass {
|
|
|
455
481
|
}
|
|
456
482
|
this.pendingFiles.push(filePath);
|
|
457
483
|
const relativePath = path.relative(this.apiRootPath, filePath);
|
|
484
|
+
const chalk = (await import("chalk")).default;
|
|
458
485
|
console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));
|
|
459
486
|
await this.syncer.syncFromWatcher(event, filePath);
|
|
460
487
|
// 처리 완료된 파일을 대기 목록에서 제거
|
|
@@ -468,10 +495,15 @@ class SonamuClass {
|
|
|
468
495
|
await this.syncer.renewChecksums();
|
|
469
496
|
const endTime = Date.now();
|
|
470
497
|
const totalTime = endTime - this.hmrStartTime;
|
|
498
|
+
const [chalk, { centerText }] = await Promise.all([
|
|
499
|
+
(await import("chalk")).default,
|
|
500
|
+
import("../utils/console-util.js")
|
|
501
|
+
]);
|
|
471
502
|
const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;
|
|
472
503
|
console.log(chalk.black.bgGreen(centerText(msg)));
|
|
473
504
|
}
|
|
474
505
|
async destroy() {
|
|
506
|
+
const { BaseModel } = await import("../database/base-model.js");
|
|
475
507
|
await BaseModel.destroy();
|
|
476
508
|
await this.watcher?.close();
|
|
477
509
|
this.storage?.destroy();
|
|
@@ -479,4 +511,4 @@ class SonamuClass {
|
|
|
479
511
|
}
|
|
480
512
|
export const Sonamu = new SonamuClass();
|
|
481
513
|
|
|
482
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/api/sonamu.ts"],"sourcesContent":["import { AsyncLocalStorage } from \"async_hooks\";\nimport chalk from \"chalk\";\nimport fastify from \"fastify\";\nimport { readFile } from \"fs/promises\";\nimport path from \"path\";\nimport { exists } from \"../utils/fs-utils\";\nimport chokidar, { type FSWatcher } from \"chokidar\";\nimport { formatInTimeZone } from \"date-fns-tz\";\nimport type { FastifyInstance, FastifyReply, FastifyRequest } from \"fastify\";\nimport type { IncomingMessage, Server, ServerResponse } from \"http\";\nimport { ZodError, ZodObject } from \"zod\";\nimport { DB, SonamuDBConfig } from \"../database/db\";\nimport { attachOnDuplicateUpdate } from \"../database/knex-plugins/knex-on-duplicate-update\";\nimport {\n  BadRequestException,\n  NotFoundException,\n} from \"../exceptions/so-exceptions\";\nimport type { Driver } from \"../file-storage/driver\";\nimport { createSSEFactory } from \"../stream/sse\";\nimport type { Syncer } from \"../syncer/syncer\";\nimport { ApiParamType, SonamuFastifyConfig } from \"../types/types\";\nimport { isLocal, isTest } from \"../utils/controller\";\nimport { findApiRootPath } from \"../utils/utils\";\nimport { humanizeZodError } from \"../utils/zod-error\";\nimport { fastifyCaster } from \"./caster\";\nimport { getZodObjectFromApi } from \"./code-converters\";\nimport type { AuthContext, Context, UploadContext } from \"./context\";\nimport type { ExtendedApi } from \"./decorators\";\nimport fastifyPassport from \"@fastify/passport\";\nimport { loadConfig, SonamuConfig, SonamuServerOptions } from \"./config\";\nimport { AbsolutePath } from \"../utils/path-utils\";\nimport { isHotReloadServer } from \"../utils/esm-utils\";\nimport { Template } from \"../template\";\nimport assert from \"assert\";\nimport { centerText } from \"../utils/console-util\";\nimport { BaseModel } from \"../database/base-model\";\n\nexport type SonamuSecrets = {\n  [key: string]: string;\n};\nclass SonamuClass {\n  public isInitialized: boolean = false;\n  public asyncLocalStorage: AsyncLocalStorage<{\n    context: Context;\n  }> = new AsyncLocalStorage();\n\n  public uploadStorage: AsyncLocalStorage<{\n    uploadContext: UploadContext;\n  }> = new AsyncLocalStorage();\n\n  public getContext(): Context {\n    const store = this.asyncLocalStorage.getStore();\n    if (store?.context) {\n      return store.context;\n    }\n\n    if (process.env.NODE_ENV === \"test\") {\n      // 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n      return {\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: () => {},\n        naiteStore: new Map<string, any>(),\n      } as unknown as Context;\n    } else {\n      throw new Error(\"Sonamu cannot find context\");\n    }\n  }\n\n  public getUploadContext(): UploadContext {\n    const store = this.uploadStorage.getStore();\n    if (store?.uploadContext) {\n      return store.uploadContext;\n    }\n    throw new Error(\n      \"Sonamu cannot find upload context. Did you use @upload decorator?\"\n    );\n  }\n\n  private _apiRootPath: AbsolutePath | null = null;\n  set apiRootPath(apiRootPath: AbsolutePath) {\n    this._apiRootPath = apiRootPath;\n  }\n  get apiRootPath(): AbsolutePath {\n    if (this._apiRootPath === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._apiRootPath!;\n  }\n  get appRootPath(): string {\n    return this.apiRootPath.split(path.sep).slice(0, -1).join(path.sep);\n  }\n\n  private _dbConfig: SonamuDBConfig | null = null;\n  set dbConfig(dbConfig: SonamuDBConfig) {\n    this._dbConfig = dbConfig;\n  }\n  get dbConfig(): SonamuDBConfig {\n    if (this._dbConfig === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._dbConfig!;\n  }\n\n  private _syncer: Syncer | null = null;\n  set syncer(syncer: Syncer) {\n    this._syncer = syncer;\n  }\n  get syncer(): Syncer {\n    if (this._syncer === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._syncer!;\n  }\n\n  private _config: SonamuConfig | null = null;\n  set config(config: SonamuConfig) {\n    this._config = config;\n  }\n  get config(): SonamuConfig {\n    if (this._config === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._config;\n  }\n\n  private _secrets: SonamuSecrets | null = null;\n  set secrets(secrets: SonamuSecrets) {\n    this._secrets = secrets;\n  }\n  get secrets(): SonamuSecrets | null {\n    return this._secrets;\n  }\n\n  private _storage: Driver | null = null;\n  set storage(storage: Driver) {\n    this._storage = storage;\n  }\n  get storage(): Driver | null {\n    return this._storage;\n  }\n\n  // HMR 처리\n  public watcher: FSWatcher | null = null;\n  private pendingFiles: string[] = [];\n  private hmrStartTime: number = 0;\n\n  public server: FastifyInstance | null = null;\n\n  async initForTesting() {\n    await this.init(true, false, undefined, true);\n  }\n\n  async init(\n    doSilent: boolean = false,\n    enableSync: boolean = true,\n    apiRootPath?: AbsolutePath,\n    forTesting: boolean = false\n  ) {\n    if (this.isInitialized) {\n      return;\n    }\n    !doSilent &&\n      console.time(\n        chalk.cyan(`Sonamu.init${forTesting ? \" for testing\" : \"\"}`)\n      );\n\n    // API 루트 패스\n    this.apiRootPath = apiRootPath ?? findApiRootPath();\n    this.config = await loadConfig(this.apiRootPath);\n    const secretsPath = path.join(this.apiRootPath, \"sonamu.secrets.json\");\n    if (await exists(secretsPath)) {\n      this.secrets = JSON.parse(\n        (await readFile(secretsPath)).toString()\n      ) as SonamuSecrets;\n    }\n\n    // DB 로드\n    this.dbConfig = DB.generateDBConfig(this.config.database);\n    !doSilent && console.log(chalk.green(\"DB Config Loaded!\"));\n    attachOnDuplicateUpdate();\n\n    // 테스팅인 경우 엔티티 로드 & 싱크 없이 중단\n    if (forTesting) {\n      this.isInitialized = true;\n      return;\n    }\n\n    // Entity 로드\n    const { EntityManager } = await import(\"../entity/entity-manager\");\n    await EntityManager.autoload(doSilent);\n\n    // Syncer\n    const { Syncer } = await import(\"../syncer/syncer\");\n    this.syncer = new Syncer();\n\n    // Autoload: Models / Types / APIs\n    await this.syncer.autoloadTypes();\n    await this.syncer.autoloadModels();\n    await this.syncer.autoloadApis();\n\n    await Template.autoload();\n\n    if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {\n      await this.syncer.sync();\n\n      this.startWatcher();\n\n      this.syncer.syncUI();\n    }\n\n    this.isInitialized = true;\n    !doSilent && console.timeEnd(chalk.cyan(\"Sonamu.init\"));\n  }\n\n  async createServer(initOptions?: {\n    enableSync?: boolean;\n    doSilent?: boolean;\n  }) {\n    if (this.isInitialized === false) {\n      await this.init(initOptions?.doSilent, initOptions?.enableSync);\n    }\n\n    const options = this.config.server;\n    const server = fastify(options.fastify);\n    this.server = server;\n\n    // Storage 설정 저장\n    if (options.storage) {\n      this.storage = options.storage;\n    }\n\n    // 플러그인 등록\n    if (options.plugins) {\n      this.registerPlugins(server, options.plugins);\n    }\n\n    if (options.auth) {\n      if (!options.plugins?.session) {\n        throw new Error(\n          \"Auth requires session plugin. Please add plugins.session configuration.\"\n        );\n      }\n\n      this.registerAuth(server, options.auth);\n    }\n\n    // API 라우팅 설정\n    await this.withFastify(server, options.apiConfig, {\n      enableSync: initOptions?.enableSync,\n      doSilent: initOptions?.doSilent,\n    });\n\n    // 서버 시작\n    await this.boot(server, options);\n\n    return server;\n  }\n\n  async withFastify(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    options?: {\n      enableSync?: boolean;\n      doSilent?: boolean;\n    }\n  ) {\n    if (this.isInitialized === false) {\n      await this.init(options?.doSilent, options?.enableSync);\n    }\n\n    this.server = server;\n\n    // timezone 설정\n    const timezone = this.config.api.timezone;\n    if (timezone) {\n      const DATE_FORMAT = \"yyyy-MM-dd'T'HH:mm:ssXXX\";\n      // ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)\n      const ISO_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(\\.\\d{3})?Z$/;\n\n      server.setReplySerializer((payload) => {\n        return JSON.stringify(payload, (_key, value) => {\n          if (typeof value === \"string\" && ISO_DATE_REGEX.test(value)) {\n            return formatInTimeZone(new Date(value), timezone, DATE_FORMAT);\n          }\n          return value;\n        });\n      });\n      !options?.doSilent &&\n        console.log(chalk.green(`Timezone set to ${timezone}`));\n    }\n\n    // 전체 라우팅 리스트\n    server.get(\n      `${this.config.api.route.prefix}/routes`,\n      async (_request, _reply): Promise<any> => {\n        return this.syncer.apis;\n      }\n    );\n\n    // Healthcheck API\n    server.get(\n      `${this.config.api.route.prefix}/healthcheck`,\n      async (_request, _reply): Promise<string> => {\n        return \"ok\";\n      }\n    );\n\n    // API 라우팅 (로컬HMR 상태와 구분)\n    if (isLocal()) {\n      server.all(\"*\", (request, reply) => {\n        const found = this.syncer.apis.find(\n          (api) =>\n            this.config.api.route.prefix + api.path ===\n              request.url.split(\"?\")[0] &&\n            (api.options.httpMethod ?? \"GET\") === request.method.toUpperCase()\n        );\n        if (found) {\n          return this.getApiHandler(found, config)(request, reply);\n        }\n        throw new NotFoundException(\"존재하지 않는 API 접근입니다.\");\n      });\n    } else {\n      this.syncer.apis.map((api) => {\n        // model\n        if (this.syncer.models[api.modelName] === undefined) {\n          throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);\n        }\n\n        // route\n        server.route({\n          method: api.options.httpMethod!,\n          url: this.config.api.route.prefix + api.path,\n          handler: this.getApiHandler(api, config),\n        }); // END server.route\n      });\n    }\n  }\n\n  getApiHandler(api: ExtendedApi, config: SonamuFastifyConfig) {\n    return async (\n      request: FastifyRequest,\n      reply: FastifyReply\n    ): Promise<unknown> => {\n      (api.options.guards ?? []).every((guard) =>\n        config.guardHandler(guard, request, api)\n      );\n\n      // 파라미터 정보로 zod 스키마 빌드\n      const ReqType = getZodObjectFromApi(api, this.syncer.types);\n\n      // request 파싱\n      const which = api.options.httpMethod === \"GET\" ? \"query\" : \"body\";\n      let reqBody: {\n        [key: string]: unknown;\n      };\n      try {\n        reqBody = fastifyCaster(ReqType).parse(request[which] ?? {});\n      } catch (e) {\n        if (e instanceof ZodError) {\n          const messages = humanizeZodError(e)\n            .map((issue) => issue.message)\n            .join(\" \");\n          throw new BadRequestException(messages, {\n            zodError: e,\n          });\n        } else {\n          throw e;\n        }\n      }\n\n      // Content-Type\n      reply.type(api.options.contentType ?? \"application/json\");\n\n      // 캐시\n      const { cacheKey, cacheTtl, cachedData } = await (async () => {\n        if (config.cache) {\n          try {\n            const cacheKeyRes = config.cache.resolveKey(api.path, reqBody);\n            if (cacheKeyRes.cache === false) {\n              return { cacheKey: null, cachedData: null };\n            }\n\n            const cacheKey = cacheKeyRes.key;\n            const cacheTtl = cacheKeyRes.ttl;\n            const cachedData = await config.cache.get(cacheKey);\n            return { cacheKey, cacheTtl, cachedData };\n          } catch (e) {\n            console.error(e);\n          }\n          return { cacheKey: null, cachedData: null };\n        }\n        return { cacheKey: null, cachedData: null };\n      })();\n      if (cachedData !== null) {\n        return cachedData;\n      }\n\n      // createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.\n      const createSSE = (<T extends ZodObject>(\n        _request: FastifyRequest,\n        _reply: FastifyReply,\n        _events: T\n      ) => createSSEFactory(_request.socket, _reply, _events)).bind(\n        null,\n        request,\n        reply\n      );\n\n      const context: Context = {\n        ...(await Promise.resolve(\n          config.contextProvider(\n            {\n              request,\n              reply,\n              headers: request.headers,\n              createSSE,\n              naiteStore: new Map<string, any>(),\n              // auth\n              user: request.user ?? null,\n              passport: {\n                login: request.login.bind(\n                  request\n                ) as AuthContext[\"passport\"][\"login\"],\n                logout: request.logout.bind(\n                  request\n                ) as AuthContext[\"passport\"][\"logout\"],\n              },\n            },\n            request,\n            reply\n          )\n        )),\n      };\n\n      const model = this.syncer.models[api.modelName];\n      return this.asyncLocalStorage.run({ context }, async () => {\n        const result = await (model as any)[api.methodName].apply(\n          model,\n          api.parameters.map((param) => {\n            // Context 인젝션\n            if (ApiParamType.isContext(param.type)) {\n              return context;\n            } else {\n              return reqBody[param.name];\n            }\n          })\n        );\n        reply.type(api.options.contentType ?? \"application/json\");\n\n        // 캐시 키 있는 경우 갱신 후 저장\n        if (config.cache && cacheKey) {\n          await config.cache.put(cacheKey, result, cacheTtl);\n        }\n        return result;\n      });\n    };\n  }\n\n  startWatcher(): void {\n    const watchPath = [\n      path.join(this.apiRootPath, \"src\"),\n      path.join(this.apiRootPath, \"sonamu.config.ts\"),\n    ];\n\n    this.watcher = chokidar.watch(watchPath, {\n      ignored: (path, stats) =>\n        !!stats?.isFile() && !path.endsWith(\".ts\") && !path.endsWith(\".json\"),\n      persistent: true,\n      ignoreInitial: true,\n    });\n\n    this.watcher.on(\"all\", async (event: string, filePath: string) => {\n      const absolutePath = filePath as AbsolutePath;\n      assert(\n        absolutePath.startsWith(this.apiRootPath),\n        \"File path is not within the API root path\"\n      );\n\n      if (event !== \"change\" && event !== \"add\") {\n        return;\n      }\n\n      try {\n        // sonamu.config.ts 변경 시 재시작\n        const isConfigTs =\n          filePath === path.join(this.apiRootPath, \"sonamu.config.ts\");\n\n        if (isConfigTs) {\n          const relativePath = filePath.replace(this.apiRootPath, \"api\");\n          console.log(\n            chalk.bold(\n              `Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`\n            )\n          );\n          process.kill(process.pid, \"SIGUSR2\");\n          return;\n        }\n\n        await this.handleFileChange(event, absolutePath);\n      } catch (e) {\n        console.error(e);\n      }\n    });\n  }\n\n  /*\n     A function that automatically handles init and destroy when using Sonamu via scripts.    \n  */\n  async runScript(fn: () => Promise<void>) {\n    await this.init(true, false, undefined, false);\n    try {\n      await fn();\n    } finally {\n      await this.destroy();\n    }\n  }\n\n  private registerPlugins(\n    server: FastifyInstance,\n    plugins: SonamuServerOptions[\"plugins\"]\n  ) {\n    if (!plugins) {\n      return;\n    }\n\n    const pluginsModules = {\n      cors: \"@fastify/cors\",\n      formbody: \"@fastify/formbody\",\n      multipart: \"@fastify/multipart\",\n      qs: \"fastify-qs\",\n      sse: \"fastify-sse-v2\",\n      static: \"@fastify/static\",\n      session: \"@fastify/secure-session\",\n    } as const;\n\n    const registerPlugin = <K extends keyof NonNullable<typeof plugins>>(\n      key: K,\n      pluginName: string\n    ) => {\n      const option = plugins[key];\n      if (!option) return;\n\n      if (option === true) {\n        server.register(import(pluginName));\n      } else {\n        server.register(import(pluginName), option);\n      }\n    };\n\n    Object.entries(pluginsModules).forEach(([key, pluginName]) => {\n      registerPlugin(key as keyof typeof plugins, pluginName);\n    });\n\n    if (plugins.custom) {\n      plugins.custom(server);\n    }\n  }\n\n  private async registerAuth(\n    server: FastifyInstance,\n    options: NonNullable<SonamuServerOptions[\"auth\"]>\n  ) {\n    server.register(fastifyPassport.initialize());\n    server.register(fastifyPassport.secureSession());\n\n    if (typeof options === \"boolean\") {\n      fastifyPassport.registerUserSerializer(async (user, _request) => user);\n      fastifyPassport.registerUserDeserializer(\n        async (serialized, _request) => serialized\n      );\n    } else {\n      fastifyPassport.registerUserSerializer(options.userSerializer);\n      fastifyPassport.registerUserDeserializer(options.userDeserializer);\n    }\n  }\n\n  private async boot(server: FastifyInstance, options: SonamuServerOptions) {\n    const port = options.listen?.port ?? 3000;\n    const host = options.listen?.host ?? \"localhost\";\n\n    server.addHook(\"onClose\", async () => {\n      await options.lifecycle?.onShutdown?.(server);\n      await this.destroy();\n    });\n\n    const shutdown = async () => {\n      try {\n        await server.close();\n        process.exit(0);\n      } catch (err) {\n        console.error(\"Error during shutdown:\", err);\n        process.exit(1);\n      }\n    };\n\n    process.on(\"SIGINT\", shutdown);\n    process.on(\"SIGTERM\", shutdown);\n\n    if (options.lifecycle?.onError) {\n      server.setErrorHandler(options.lifecycle?.onError);\n    }\n\n    server\n      .listen({ port, host })\n      .then(async () => {\n        await options.lifecycle?.onStart?.(server);\n      })\n      .catch(async (err) => {\n        console.error(chalk.red(\"Failed to start server:\", err));\n        await shutdown();\n      });\n  }\n\n  private async handleFileChange(\n    event: string,\n    filePath: AbsolutePath\n  ): Promise<void> {\n    // 첫 번째 파일이면 HMR 시작 시간 기록\n    if (this.pendingFiles.length === 0) {\n      this.hmrStartTime = Date.now();\n    }\n    this.pendingFiles.push(filePath);\n\n    const relativePath = path.relative(this.apiRootPath, filePath);\n    console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));\n\n    await this.syncer.syncFromWatcher(event, filePath);\n\n    // 처리 완료된 파일을 대기 목록에서 제거\n    this.pendingFiles = this.pendingFiles.slice(1);\n\n    // 모든 파일 처리가 완료되면 최종 메시지 출력\n    if (this.pendingFiles.length === 0) {\n      await this.finishHMR();\n    }\n  }\n\n  private async finishHMR(): Promise<void> {\n    await this.syncer.renewChecksums();\n\n    const endTime = Date.now();\n    const totalTime = endTime - this.hmrStartTime;\n    const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;\n\n    console.log(chalk.black.bgGreen(centerText(msg)));\n  }\n\n  async destroy(): Promise<void> {\n    await BaseModel.destroy();\n    await this.watcher?.close();\n    this.storage?.destroy();\n  }\n}\nexport const Sonamu = new SonamuClass();\n"],"names":["AsyncLocalStorage","chalk","fastify","readFile","path","exists","chokidar","formatInTimeZone","ZodError","DB","attachOnDuplicateUpdate","BadRequestException","NotFoundException","createSSEFactory","ApiParamType","isLocal","isTest","findApiRootPath","humanizeZodError","fastifyCaster","getZodObjectFromApi","fastifyPassport","loadConfig","isHotReloadServer","Template","assert","centerText","BaseModel","SonamuClass","isInitialized","asyncLocalStorage","uploadStorage","getContext","store","getStore","context","process","env","NODE_ENV","request","reply","headers","createSSE","naiteStore","Map","Error","getUploadContext","uploadContext","_apiRootPath","apiRootPath","appRootPath","split","sep","slice","join","_dbConfig","dbConfig","_syncer","syncer","_config","config","_secrets","secrets","_storage","storage","watcher","pendingFiles","hmrStartTime","server","initForTesting","init","undefined","doSilent","enableSync","forTesting","console","time","cyan","secretsPath","JSON","parse","toString","generateDBConfig","database","log","green","EntityManager","autoload","Syncer","autoloadTypes","autoloadModels","autoloadApis","sync","startWatcher","syncUI","timeEnd","createServer","initOptions","options","plugins","registerPlugins","auth","session","registerAuth","withFastify","apiConfig","boot","timezone","api","DATE_FORMAT","ISO_DATE_REGEX","setReplySerializer","payload","stringify","_key","value","test","Date","get","route","prefix","_request","_reply","apis","all","found","find","url","httpMethod","method","toUpperCase","getApiHandler","map","models","modelName","handler","guards","every","guard","guardHandler","ReqType","types","which","reqBody","e","messages","issue","message","zodError","type","contentType","cacheKey","cacheTtl","cachedData","cache","cacheKeyRes","resolveKey","key","ttl","error","_events","socket","bind","Promise","resolve","contextProvider","user","passport","login","logout","model","run","result","methodName","apply","parameters","param","isContext","name","put","watchPath","watch","ignored","stats","isFile","endsWith","persistent","ignoreInitial","on","event","filePath","absolutePath","startsWith","isConfigTs","relativePath","replace","bold","blue","kill","pid","handleFileChange","runScript","fn","destroy","pluginsModules","cors","formbody","multipart","qs","sse","static","registerPlugin","pluginName","option","register","Object","entries","forEach","custom","initialize","secureSession","registerUserSerializer","registerUserDeserializer","serialized","userSerializer","userDeserializer","port","listen","host","addHook","lifecycle","onShutdown","shutdown","close","exit","err","onError","setErrorHandler","then","onStart","catch","red","length","now","push","relative","syncFromWatcher","finishHMR","renewChecksums","endTime","totalTime","msg","white","black","bgGreen","Sonamu"],"mappings":"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"}
|
|
514
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/api/sonamu.ts"],"sourcesContent":["import assert from \"assert\";\nimport { AsyncLocalStorage } from \"async_hooks\";\nimport type { FSWatcher } from \"chokidar\";\nimport type { FastifyInstance, FastifyReply, FastifyRequest } from \"fastify\";\nimport type { IncomingMessage, Server, ServerResponse } from \"http\";\nimport path from \"path\";\nimport type { ZodObject } from \"zod\";\nimport type { SonamuDBConfig } from \"../database/db\";\nimport type { Driver } from \"../file-storage/driver\";\nimport { Naite } from \"../naite/naite\";\nimport type { Syncer } from \"../syncer/syncer\";\nimport type { SonamuFastifyConfig } from \"../types/types\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\nimport type { SonamuConfig, SonamuServerOptions } from \"./config\";\nimport type { AuthContext, Context, UploadContext } from \"./context\";\nimport type { ExtendedApi } from \"./decorators\";\n\nexport type SonamuSecrets = {\n  anthropic_api_key?: string;\n};\nclass SonamuClass {\n  public isInitialized: boolean = false;\n  public asyncLocalStorage: AsyncLocalStorage<{\n    context: Context;\n  }> = new AsyncLocalStorage();\n\n  public uploadStorage: AsyncLocalStorage<{\n    uploadContext: UploadContext;\n  }> = new AsyncLocalStorage();\n\n  public getContext(): Context {\n    const store = this.asyncLocalStorage.getStore();\n    if (store?.context) {\n      return store.context;\n    }\n\n    if (process.env.NODE_ENV === \"test\") {\n      // 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n      return {\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: () => {},\n        // biome-ignore lint/suspicious/noExplicitAny: 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n        naiteStore: new Map<string, any>(),\n      } as unknown as Context;\n    } else {\n      throw new Error(\"Sonamu cannot find context\");\n    }\n  }\n\n  public getUploadContext(): UploadContext {\n    const store = this.uploadStorage.getStore();\n    if (store?.uploadContext) {\n      return store.uploadContext;\n    }\n    throw new Error(\"Sonamu cannot find upload context. Did you use @upload decorator?\");\n  }\n\n  private _apiRootPath: AbsolutePath | null = null;\n  set apiRootPath(apiRootPath: AbsolutePath) {\n    this._apiRootPath = apiRootPath;\n  }\n  get apiRootPath(): AbsolutePath {\n    if (this._apiRootPath === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._apiRootPath;\n  }\n  get appRootPath(): string {\n    return this.apiRootPath.split(path.sep).slice(0, -1).join(path.sep);\n  }\n\n  private _dbConfig: SonamuDBConfig | null = null;\n  set dbConfig(dbConfig: SonamuDBConfig) {\n    this._dbConfig = dbConfig;\n  }\n  get dbConfig(): SonamuDBConfig {\n    if (this._dbConfig === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._dbConfig;\n  }\n\n  private _syncer: Syncer | null = null;\n  set syncer(syncer: Syncer) {\n    this._syncer = syncer;\n  }\n  get syncer(): Syncer {\n    if (this._syncer === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._syncer;\n  }\n\n  private _config: SonamuConfig | null = null;\n  set config(config: SonamuConfig) {\n    this._config = config;\n  }\n  get config(): SonamuConfig {\n    if (this._config === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._config;\n  }\n\n  private _secrets: SonamuSecrets | null = null;\n  set secrets(secrets: SonamuSecrets) {\n    this._secrets = secrets;\n  }\n  get secrets(): SonamuSecrets | null {\n    return this._secrets;\n  }\n\n  private _storage: Driver | null = null;\n  set storage(storage: Driver) {\n    this._storage = storage;\n  }\n  get storage(): Driver | null {\n    return this._storage;\n  }\n\n  // HMR 처리\n  public watcher: FSWatcher | null = null;\n  private pendingFiles: string[] = [];\n  private hmrStartTime: number = 0;\n\n  public server: FastifyInstance | null = null;\n\n  async initForTesting() {\n    await this.init(true, false, undefined, true);\n  }\n\n  async init(\n    doSilent: boolean = false,\n    enableSync: boolean = true,\n    apiRootPath?: AbsolutePath,\n    forTesting: boolean = false,\n  ) {\n    if (this.isInitialized) {\n      return;\n    }\n\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.time(chalk.cyan(`Sonamu.init${forTesting ? \" for testing\" : \"\"}`));\n    }\n\n    // API 루트 패스\n    const { findApiRootPath } = await import(\"../utils/utils\");\n    this.apiRootPath = apiRootPath ?? findApiRootPath();\n\n    const { loadConfig } = await import(\"./config\");\n    this.config = await loadConfig(this.apiRootPath);\n    // sonamu.config.ts 기본값 설정\n    this.config.database.database = this.config.database.database ?? \"postgresql\";\n\n    if (process.env.ANTHROPIC_API_KEY) {\n      this.secrets = {\n        anthropic_api_key: process.env.ANTHROPIC_API_KEY,\n      };\n    }\n\n    // DB 로드\n    const { DB } = await import(\"../database/db\");\n    this.dbConfig = DB.generateDBConfig(this.config.database);\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.log(chalk.green(\"DB Config Loaded!\"));\n    }\n\n    // 테스팅인 경우 엔티티 로드 & 싱크 없이 중단\n    if (forTesting) {\n      this.isInitialized = true;\n      return;\n    }\n\n    // Entity 로드\n    const { EntityManager } = await import(\"../entity/entity-manager\");\n    await EntityManager.autoload(doSilent);\n\n    // Syncer\n    const { Syncer } = await import(\"../syncer/syncer\");\n    this.syncer = new Syncer();\n\n    // Autoload: Models / Types / APIs\n    await this.syncer.autoloadTypes();\n    await this.syncer.autoloadModels();\n    await this.syncer.autoloadApis();\n\n    const { TemplateManager } = await import(\"../template\");\n    await TemplateManager.autoload();\n\n    const { isLocal, isTest } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      // 로컬에서는 코드 생성을 위해 Biome 셋업이 필요함 (현재 apiRootPath 전달하여 실행)\n      (await import(\"../utils/formatter\")).setupBiome(this.apiRootPath);\n    }\n\n    const { isHotReloadServer } = await import(\"../utils/controller\");\n    if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {\n      await this.syncer.sync();\n\n      await this.startWatcher();\n\n      this.syncer.syncUI();\n    }\n\n    this.isInitialized = true;\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.timeEnd(chalk.cyan(\"Sonamu.init\"));\n    }\n  }\n\n  async createServer(initOptions?: { enableSync?: boolean; doSilent?: boolean }) {\n    if (this.isInitialized === false) {\n      await this.init(initOptions?.doSilent, initOptions?.enableSync);\n    }\n\n    const options = this.config.server;\n    const fastify = (await import(\"fastify\")).default;\n    const server = fastify(options.fastify);\n    this.server = server;\n\n    // Storage 설정 저장\n    if (options.storage) {\n      this.storage = options.storage;\n    }\n\n    // 플러그인 등록\n    if (options.plugins) {\n      await this.registerPlugins(server, options.plugins);\n    }\n\n    if (options.auth) {\n      if (!options.plugins?.session) {\n        throw new Error(\"Auth requires session plugin. Please add plugins.session configuration.\");\n      }\n\n      await this.registerAuth(server, options.auth);\n    }\n\n    // API 라우팅 설정\n    await this.withFastify(server, options.apiConfig, {\n      enableSync: initOptions?.enableSync,\n      doSilent: initOptions?.doSilent,\n    });\n\n    // 서버 시작\n    await this.boot(server, options);\n\n    return server;\n  }\n\n  async withFastify(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    options?: {\n      enableSync?: boolean;\n      doSilent?: boolean;\n    },\n  ) {\n    if (this.isInitialized === false) {\n      await this.init(options?.doSilent, options?.enableSync);\n    }\n\n    this.server = server;\n\n    // timezone 설정\n    const timezone = this.config.api.timezone;\n    if (timezone) {\n      // 타임존에 맞게 응답 날짜 스트링을 변환해주어야 합니다.\n      // 가령 timezone이 \"Asia/Seoul\" 이면\n      // \"2025-11-21T00:00:00.000Z\" 를 \"2025-11-21T09:00:00+09:00\" 으로 변환해주어야 합니다.\n      const { formatInTimeZone } = await import(\"date-fns-tz\");\n\n      // ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)\n      const ISO_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(\\.\\d{3})?Z$/;\n\n      // T를 둘러싼 작은따옴표가 없다면 \"2025-11-19176354618900018:56:29+09:00\"와 같은 결과가 나옵니다.\n      // 이는 date-fns 특입니다.\n      // 이렇게 해도 괜찮습니다. \"2025-11-19T18:56:29+09:00\" 모양으로 잘 나옵니다.\n      const DATE_FORMAT = \"yyyy-MM-dd'T'HH:mm:ssXXX\";\n\n      server.setReplySerializer((payload) => {\n        return JSON.stringify(payload, (_key, value) => {\n          if (typeof value === \"string\" && ISO_DATE_REGEX.test(value)) {\n            return formatInTimeZone(\n              new Date(value),\n              timezone as `${string}/${string}`,\n              DATE_FORMAT,\n            );\n          }\n          return value;\n        });\n      });\n      if (!options?.doSilent) {\n        const chalk = (await import(\"chalk\")).default;\n        console.log(chalk.green(`Timezone set to ${timezone}`));\n      }\n    }\n\n    // 전체 라우팅 리스트\n    server.get(\n      `${this.config.api.route.prefix}/routes`,\n      async (_request, _reply): Promise<typeof this.syncer.apis> => {\n        return this.syncer.apis;\n      },\n    );\n\n    // Healthcheck API\n    server.get(\n      `${this.config.api.route.prefix}/healthcheck`,\n      async (_request, _reply): Promise<string> => {\n        return \"ok\";\n      },\n    );\n\n    // API 라우팅 (로컬HMR 상태와 구분)\n    const { isLocal } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      server.all(\"*\", async (request, reply) => {\n        const found = this.syncer.apis.find(\n          (api) =>\n            this.config.api.route.prefix + api.path === request.url.split(\"?\")[0] &&\n            (api.options.httpMethod ?? \"GET\") === request.method.toUpperCase(),\n        );\n        if (found) {\n          return this.getApiHandler(found, config)(request, reply);\n        }\n        const { NotFoundException } = await import(\"../exceptions/so-exceptions\");\n        throw new NotFoundException(\"존재하지 않는 API 접근입니다.\");\n      });\n    } else {\n      for (const api of this.syncer.apis) {\n        // model\n        if (this.syncer.models[api.modelName] === undefined) {\n          throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);\n        }\n\n        // route\n        server.route({\n          method: api.options.httpMethod ?? \"GET\",\n          url: this.config.api.route.prefix + api.path,\n          handler: this.getApiHandler(api, config),\n        }); // END server.route\n      }\n    }\n  }\n\n  getApiHandler(\n    api: ExtendedApi,\n    config: SonamuFastifyConfig,\n  ): (request: FastifyRequest, reply: FastifyReply) => Promise<unknown> {\n    return async (request: FastifyRequest, reply: FastifyReply): Promise<unknown> => {\n      (api.options.guards ?? []).every((guard) => config.guardHandler(guard, request, api));\n\n      // 파라미터 정보로 zod 스키마 빌드\n      const { getZodObjectFromApi } = await import(\"./code-converters\");\n      const ReqType = getZodObjectFromApi(api, this.syncer.types);\n\n      // request 파싱\n      const which = api.options.httpMethod === \"GET\" ? \"query\" : \"body\";\n      let reqBody: {\n        [key: string]: unknown;\n      };\n      try {\n        const { fastifyCaster } = await import(\"./caster\");\n        reqBody = fastifyCaster(ReqType).parse(request[which] ?? {});\n      } catch (e) {\n        const { ZodError } = await import(\"zod\");\n        if (e instanceof ZodError) {\n          const { humanizeZodError } = await import(\"../utils/zod-error\");\n          const messages = humanizeZodError(e)\n            .map((issue) => issue.message)\n            .join(\" \");\n          const { BadRequestException } = await import(\"../exceptions/so-exceptions\");\n          throw new BadRequestException(messages, {\n            zodError: e,\n          });\n        } else {\n          throw e;\n        }\n      }\n\n      // Content-Type\n      reply.type(api.options.contentType ?? \"application/json\");\n\n      // 캐시\n      const { cacheKey, cacheTtl, cachedData } = await (async () => {\n        if (config.cache) {\n          try {\n            const cacheKeyRes = config.cache.resolveKey(api.path, reqBody);\n            if (cacheKeyRes.cache === false) {\n              return { cacheKey: null, cachedData: null };\n            }\n\n            const cacheKey = cacheKeyRes.key;\n            const cacheTtl = cacheKeyRes.ttl;\n            const cachedData = await config.cache.get(cacheKey);\n            return { cacheKey, cacheTtl, cachedData };\n          } catch (e) {\n            console.error(e);\n          }\n          return { cacheKey: null, cachedData: null };\n        }\n        return { cacheKey: null, cachedData: null };\n      })();\n      if (cachedData !== null) {\n        return cachedData;\n      }\n\n      // createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.\n      const { createSSEFactory } = await import(\"../stream/sse\");\n      const createSSE = (<T extends ZodObject>(\n        _request: FastifyRequest,\n        _reply: FastifyReply,\n        _events: T,\n      ) => createSSEFactory(_request.socket, _reply, _events)).bind(null, request, reply);\n\n      const context: Context = {\n        ...(await Promise.resolve(\n          config.contextProvider(\n            {\n              request,\n              reply,\n              headers: request.headers,\n              createSSE,\n              naiteStore: Naite.createStore(),\n              // auth\n              user: request.user ?? null,\n              passport: {\n                login: request.login.bind(request) as AuthContext[\"passport\"][\"login\"],\n                logout: request.logout.bind(request) as AuthContext[\"passport\"][\"logout\"],\n              },\n            },\n            request,\n            reply,\n          ),\n        )),\n      };\n\n      const model = this.syncer.models[api.modelName];\n      return this.asyncLocalStorage.run({ context }, async () => {\n        const { ApiParamType } = await import(\"../types/types\");\n        // biome-ignore lint/suspicious/noExplicitAny: model은 모델 인스턴스이므로 메서드 호출 가능\n        const result = await (model as any)[api.methodName].apply(\n          model,\n          api.parameters.map((param) => {\n            // Context 인젝션\n            if (ApiParamType.isContext(param.type)) {\n              return context;\n            } else {\n              return reqBody[param.name];\n            }\n          }),\n        );\n        reply.type(api.options.contentType ?? \"application/json\");\n\n        // 캐시 키 있는 경우 갱신 후 저장\n        if (config.cache && cacheKey) {\n          await config.cache.put(cacheKey, result, cacheTtl);\n        }\n        return result;\n      });\n    };\n  }\n\n  async startWatcher(): Promise<void> {\n    const watchPath = [\n      path.join(this.apiRootPath, \"src\"),\n      path.join(this.apiRootPath, \"sonamu.config.ts\"),\n    ];\n\n    const chokidar = (await import(\"chokidar\")).default;\n    this.watcher = chokidar.watch(watchPath, {\n      ignored: (path, stats) =>\n        !!stats?.isFile() && !path.endsWith(\".ts\") && !path.endsWith(\".json\"),\n      persistent: true,\n      ignoreInitial: true,\n    });\n\n    this.watcher.on(\"all\", async (event: string, filePath: string) => {\n      const absolutePath = filePath as AbsolutePath;\n      assert(\n        absolutePath.startsWith(this.apiRootPath),\n        \"File path is not within the API root path\",\n      );\n\n      if (event !== \"change\" && event !== \"add\") {\n        return;\n      }\n\n      try {\n        // sonamu.config.ts 변경 시 재시작\n        const isConfigTs = filePath === path.join(this.apiRootPath, \"sonamu.config.ts\");\n\n        if (isConfigTs) {\n          const relativePath = filePath.replace(this.apiRootPath, \"api\");\n          const chalk = (await import(\"chalk\")).default;\n          console.log(\n            chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`),\n          );\n          process.kill(process.pid, \"SIGUSR2\");\n          return;\n        }\n\n        await this.handleFileChange(event, absolutePath);\n      } catch (e) {\n        console.error(e);\n      }\n    });\n  }\n\n  /*\n     A function that automatically handles init and destroy when using Sonamu via scripts.    \n  */\n  async runScript(fn: () => Promise<void>) {\n    await this.init(true, false, undefined, false);\n    try {\n      await fn();\n    } finally {\n      await this.destroy();\n    }\n  }\n\n  private async registerPlugins(server: FastifyInstance, plugins: SonamuServerOptions[\"plugins\"]) {\n    if (!plugins) {\n      return;\n    }\n\n    const pluginsModules = {\n      cors: \"@fastify/cors\",\n      formbody: \"@fastify/formbody\",\n      multipart: \"@fastify/multipart\",\n      qs: \"fastify-qs\",\n      sse: \"fastify-sse-v2\",\n      static: \"@fastify/static\",\n      session: \"@fastify/secure-session\",\n    } as const;\n\n    const registerPlugin = async <K extends keyof NonNullable<typeof plugins>>(\n      key: K,\n      pluginName: string,\n    ) => {\n      const option = plugins[key];\n      if (!option) return;\n\n      if (option === true) {\n        server.register((await import(pluginName)).default);\n      } else {\n        server.register((await import(pluginName)).default, option);\n      }\n    };\n\n    for (const [key, pluginName] of Object.entries(pluginsModules)) {\n      await registerPlugin(key as keyof typeof plugins, pluginName);\n    }\n\n    if (plugins.custom) {\n      plugins.custom(server);\n    }\n  }\n\n  private async registerAuth(\n    server: FastifyInstance,\n    options: NonNullable<SonamuServerOptions[\"auth\"]>,\n  ) {\n    // await import(\"fastify\");\n    const fastifyPassport = (await import(\"@fastify/passport\")).default;\n    server.register(fastifyPassport.initialize());\n    server.register(fastifyPassport.secureSession());\n\n    if (typeof options === \"boolean\") {\n      fastifyPassport.registerUserSerializer(async (user, _request) => user);\n      fastifyPassport.registerUserDeserializer(async (serialized, _request) => serialized);\n    } else {\n      fastifyPassport.registerUserSerializer(options.userSerializer);\n      fastifyPassport.registerUserDeserializer(options.userDeserializer);\n    }\n  }\n\n  private async boot(server: FastifyInstance, options: SonamuServerOptions) {\n    const port = options.listen?.port ?? 3000;\n    const host = options.listen?.host ?? \"localhost\";\n\n    server.addHook(\"onClose\", async () => {\n      await options.lifecycle?.onShutdown?.(server);\n      await this.destroy();\n    });\n\n    const shutdown = async () => {\n      try {\n        await server.close();\n        process.exit(0);\n      } catch (err) {\n        console.error(\"Error during shutdown:\", err);\n        process.exit(1);\n      }\n    };\n\n    process.on(\"SIGINT\", shutdown);\n    process.on(\"SIGTERM\", shutdown);\n\n    if (options.lifecycle?.onError) {\n      server.setErrorHandler(options.lifecycle?.onError);\n    }\n\n    server\n      .listen({ port, host })\n      .then(async () => {\n        await options.lifecycle?.onStart?.(server);\n      })\n      .catch(async (err) => {\n        const chalk = (await import(\"chalk\")).default;\n        console.error(chalk.red(\"Failed to start server:\", err));\n        await shutdown();\n      });\n  }\n\n  private async handleFileChange(event: string, filePath: AbsolutePath): Promise<void> {\n    // 첫 번째 파일이면 HMR 시작 시간 기록\n    if (this.pendingFiles.length === 0) {\n      this.hmrStartTime = Date.now();\n    }\n    this.pendingFiles.push(filePath);\n\n    const relativePath = path.relative(this.apiRootPath, filePath);\n    const chalk = (await import(\"chalk\")).default;\n    console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));\n\n    await this.syncer.syncFromWatcher(event, filePath);\n\n    // 처리 완료된 파일을 대기 목록에서 제거\n    this.pendingFiles = this.pendingFiles.slice(1);\n\n    // 모든 파일 처리가 완료되면 최종 메시지 출력\n    if (this.pendingFiles.length === 0) {\n      await this.finishHMR();\n    }\n  }\n\n  private async finishHMR(): Promise<void> {\n    await this.syncer.renewChecksums();\n\n    const endTime = Date.now();\n    const totalTime = endTime - this.hmrStartTime;\n    const [chalk, { centerText }] = await Promise.all([\n      (await import(\"chalk\")).default,\n      import(\"../utils/console-util\"),\n    ]);\n    const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;\n\n    console.log(chalk.black.bgGreen(centerText(msg)));\n  }\n\n  async destroy(): Promise<void> {\n    const { BaseModel } = await import(\"../database/base-model\");\n    await BaseModel.destroy();\n    await this.watcher?.close();\n    this.storage?.destroy();\n  }\n}\nexport const Sonamu = new SonamuClass();\n"],"names":["assert","AsyncLocalStorage","path","Naite","SonamuClass","isInitialized","asyncLocalStorage","uploadStorage","getContext","store","getStore","context","process","env","NODE_ENV","request","reply","headers","createSSE","naiteStore","Map","Error","getUploadContext","uploadContext","_apiRootPath","apiRootPath","appRootPath","split","sep","slice","join","_dbConfig","dbConfig","_syncer","syncer","_config","config","_secrets","secrets","_storage","storage","watcher","pendingFiles","hmrStartTime","server","initForTesting","init","undefined","doSilent","enableSync","forTesting","chalk","default","console","time","cyan","findApiRootPath","loadConfig","database","ANTHROPIC_API_KEY","anthropic_api_key","DB","generateDBConfig","log","green","EntityManager","autoload","Syncer","autoloadTypes","autoloadModels","autoloadApis","TemplateManager","isLocal","isTest","setupBiome","isHotReloadServer","sync","startWatcher","syncUI","timeEnd","createServer","initOptions","options","fastify","plugins","registerPlugins","auth","session","registerAuth","withFastify","apiConfig","boot","timezone","api","formatInTimeZone","ISO_DATE_REGEX","DATE_FORMAT","setReplySerializer","payload","JSON","stringify","_key","value","test","Date","get","route","prefix","_request","_reply","apis","all","found","find","url","httpMethod","method","toUpperCase","getApiHandler","NotFoundException","models","modelName","handler","guards","every","guard","guardHandler","getZodObjectFromApi","ReqType","types","which","reqBody","fastifyCaster","parse","e","ZodError","humanizeZodError","messages","map","issue","message","BadRequestException","zodError","type","contentType","cacheKey","cacheTtl","cachedData","cache","cacheKeyRes","resolveKey","key","ttl","error","createSSEFactory","_events","socket","bind","Promise","resolve","contextProvider","createStore","user","passport","login","logout","model","run","ApiParamType","result","methodName","apply","parameters","param","isContext","name","put","watchPath","chokidar","watch","ignored","stats","isFile","endsWith","persistent","ignoreInitial","on","event","filePath","absolutePath","startsWith","isConfigTs","relativePath","replace","bold","blue","kill","pid","handleFileChange","runScript","fn","destroy","pluginsModules","cors","formbody","multipart","qs","sse","static","registerPlugin","pluginName","option","register","Object","entries","custom","fastifyPassport","initialize","secureSession","registerUserSerializer","registerUserDeserializer","serialized","userSerializer","userDeserializer","port","listen","host","addHook","lifecycle","onShutdown","shutdown","close","exit","err","onError","setErrorHandler","then","onStart","catch","red","length","now","push","relative","syncFromWatcher","finishHMR","renewChecksums","endTime","totalTime","centerText","msg","white","black","bgGreen","BaseModel","Sonamu"],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"validator.d.ts","sourceRoot":"","sources":["../../src/api/validator.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,iBAAiB,aAmE5B,CAAC;AAEH,qBAAa,kBAAmB,SAAQ,KAAK;gBAC/B,OAAO,EAAE,MAAM;CAI5B;AAED,wBAAgB,kBAAkB,CAAC,UAAU,EAAE,MAAM,QAMpD"}
|