sonamu 0.5.7 → 0.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api/base-frame.js +12 -2
- package/dist/api/caster.js +66 -2
- package/dist/api/code-converters.js +489 -2
- package/dist/api/config.d.ts +76 -0
- package/dist/api/config.d.ts.map +1 -0
- package/dist/api/config.js +32 -0
- package/dist/api/context.d.ts +1 -0
- package/dist/api/context.d.ts.map +1 -1
- package/dist/api/context.js +3 -2
- package/dist/api/decorators.d.ts.map +1 -1
- package/dist/api/decorators.js +142 -2
- package/dist/api/index.js +9 -2
- package/dist/api/sonamu.d.ts +8 -22
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +482 -2
- package/dist/bin/build-config.d.ts +2 -1
- package/dist/bin/build-config.d.ts.map +1 -1
- package/dist/bin/build-config.js +12 -2
- package/dist/bin/cli-wrapper.js +71 -2
- package/dist/bin/cli.js +418 -2
- package/dist/bin/hot-hook-register.d.ts +11 -0
- package/dist/bin/hot-hook-register.d.ts.map +1 -0
- package/dist/bin/hot-hook-register.js +21 -0
- package/dist/database/_batch_update.js +78 -2
- package/dist/database/base-model.js +247 -2
- package/dist/database/code-generator.js +53 -2
- package/dist/database/db.d.ts +2 -16
- package/dist/database/db.d.ts.map +1 -1
- package/dist/database/db.js +132 -2
- package/dist/database/knex-plugins/knex-on-duplicate-update.js +39 -2
- package/dist/database/puri-wrapper.js +109 -2
- package/dist/database/puri.d.ts +23 -16
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +539 -2
- package/dist/database/puri.types.d.ts +8 -3
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +3 -2
- package/dist/database/transaction-context.js +14 -2
- package/dist/database/upsert-builder.js +215 -2
- package/dist/entity/entity-manager.d.ts +3 -1
- package/dist/entity/entity-manager.d.ts.map +1 -1
- package/dist/entity/entity-manager.js +114 -2
- package/dist/entity/entity-utils.js +210 -2
- package/dist/entity/entity.d.ts.map +1 -1
- package/dist/entity/entity.js +651 -2
- package/dist/exceptions/error-handler.js +29 -2
- package/dist/exceptions/so-exceptions.js +85 -2
- package/dist/file-storage/driver.js +79 -2
- package/dist/file-storage/file-storage.js +75 -2
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +28 -2
- package/dist/migration/code-generation.js +558 -2
- package/dist/migration/migration-set.js +364 -2
- package/dist/migration/migrator.d.ts +0 -9
- package/dist/migration/migrator.d.ts.map +1 -1
- package/dist/migration/migrator.js +510 -2
- package/dist/migration/types.js +3 -2
- package/dist/naite/naite.d.ts +12 -0
- package/dist/naite/naite.d.ts.map +1 -0
- package/dist/naite/naite.js +72 -0
- package/dist/stream/index.js +3 -2
- package/dist/stream/sse.js +38 -2
- package/dist/syncer/api-parser.d.ts +20 -0
- package/dist/syncer/api-parser.d.ts.map +1 -0
- package/dist/syncer/api-parser.js +229 -0
- package/dist/syncer/checksum.d.ts +21 -0
- package/dist/syncer/checksum.d.ts.map +1 -0
- package/dist/syncer/checksum.js +98 -0
- package/dist/syncer/code-generator.d.ts +20 -0
- package/dist/syncer/code-generator.d.ts.map +1 -0
- package/dist/syncer/code-generator.js +141 -0
- package/dist/syncer/entity-operations.d.ts +17 -0
- package/dist/syncer/entity-operations.d.ts.map +1 -0
- package/dist/syncer/entity-operations.js +58 -0
- package/dist/syncer/file-patterns.d.ts +29 -0
- package/dist/syncer/file-patterns.d.ts.map +1 -0
- package/dist/syncer/file-patterns.js +38 -0
- package/dist/syncer/index.d.ts +6 -0
- package/dist/syncer/index.d.ts.map +1 -1
- package/dist/syncer/index.js +9 -2
- package/dist/syncer/module-loader.d.ts +35 -0
- package/dist/syncer/module-loader.d.ts.map +1 -0
- package/dist/syncer/module-loader.js +82 -0
- package/dist/syncer/syncer.d.ts +93 -108
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +375 -2
- package/dist/template/entity-converter.d.ts +14 -0
- package/dist/template/entity-converter.d.ts.map +1 -0
- package/dist/template/entity-converter.js +101 -0
- package/dist/template/helpers.d.ts +23 -0
- package/dist/template/helpers.d.ts.map +1 -0
- package/dist/template/helpers.js +64 -0
- package/dist/{templates → template/implementations}/entity.template.d.ts +3 -3
- package/dist/template/implementations/entity.template.d.ts.map +1 -0
- package/dist/template/implementations/entity.template.js +87 -0
- package/dist/{templates → template/implementations}/generated.template.d.ts +3 -3
- package/dist/template/implementations/generated.template.d.ts.map +1 -0
- package/dist/template/implementations/generated.template.js +232 -0
- package/dist/{templates → template/implementations}/generated_http.template.d.ts +3 -3
- package/dist/template/implementations/generated_http.template.d.ts.map +1 -0
- package/dist/template/implementations/generated_http.template.js +131 -0
- package/dist/{templates → template/implementations}/generated_sso.template.d.ts +3 -3
- package/dist/template/implementations/generated_sso.template.d.ts.map +1 -0
- package/dist/template/implementations/generated_sso.template.js +105 -0
- package/dist/{templates → template/implementations}/init_types.template.d.ts +3 -3
- package/dist/template/implementations/init_types.template.d.ts.map +1 -0
- package/dist/template/implementations/init_types.template.js +38 -0
- package/dist/template/implementations/model.template.d.ts +17 -0
- package/dist/template/implementations/model.template.d.ts.map +1 -0
- package/dist/template/implementations/model.template.js +171 -0
- package/dist/{templates → template/implementations}/model_test.template.d.ts +3 -3
- package/dist/template/implementations/model_test.template.d.ts.map +1 -0
- package/dist/template/implementations/model_test.template.js +35 -0
- package/dist/{templates → template/implementations}/service.template.d.ts +6 -6
- package/dist/template/implementations/service.template.d.ts.map +1 -0
- package/dist/template/implementations/service.template.js +193 -0
- package/dist/{templates → template/implementations}/view_enums_buttonset.template.d.ts +3 -3
- package/dist/template/implementations/view_enums_buttonset.template.d.ts.map +1 -0
- package/dist/template/implementations/view_enums_buttonset.template.js +31 -0
- package/dist/{templates → template/implementations}/view_enums_dropdown.template.d.ts +3 -4
- package/dist/template/implementations/view_enums_dropdown.template.d.ts.map +1 -0
- package/dist/template/implementations/view_enums_dropdown.template.js +50 -0
- package/dist/{templates → template/implementations}/view_enums_select.template.d.ts +3 -3
- package/dist/template/implementations/view_enums_select.template.d.ts.map +1 -0
- package/dist/template/implementations/view_enums_select.template.js +55 -0
- package/dist/{templates → template/implementations}/view_form.template.d.ts +5 -5
- package/dist/template/implementations/view_form.template.d.ts.map +1 -0
- package/dist/template/implementations/view_form.template.js +337 -0
- package/dist/{templates → template/implementations}/view_id_all_select.template.d.ts +3 -3
- package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -0
- package/dist/template/implementations/view_id_all_select.template.js +31 -0
- package/dist/{templates → template/implementations}/view_id_async_select.template.d.ts +3 -3
- package/dist/template/implementations/view_id_async_select.template.d.ts.map +1 -0
- package/dist/template/implementations/view_id_async_select.template.js +105 -0
- package/dist/{templates → template/implementations}/view_list.template.d.ts +5 -13
- package/dist/template/implementations/view_list.template.d.ts.map +1 -0
- package/dist/template/implementations/view_list.template.js +465 -0
- package/dist/{templates → template/implementations}/view_list_columns.template.d.ts +3 -3
- package/dist/template/implementations/view_list_columns.template.d.ts.map +1 -0
- package/dist/template/implementations/view_list_columns.template.js +49 -0
- package/dist/{templates → template/implementations}/view_search_input.template.d.ts +3 -3
- package/dist/template/implementations/view_search_input.template.d.ts.map +1 -0
- package/dist/template/implementations/view_search_input.template.js +64 -0
- package/dist/template/index.d.ts +5 -0
- package/dist/template/index.d.ts.map +1 -0
- package/dist/template/index.js +6 -0
- package/dist/template/template.d.ts +39 -0
- package/dist/template/template.d.ts.map +1 -0
- package/dist/template/template.js +47 -0
- package/dist/template/zod-converter.d.ts +18 -0
- package/dist/template/zod-converter.d.ts.map +1 -0
- package/dist/template/zod-converter.js +166 -0
- package/dist/testing/_relation-graph.js +80 -2
- package/dist/testing/fixture-manager.d.ts.map +1 -1
- package/dist/testing/fixture-manager.js +521 -2
- package/dist/types/types.d.ts +39 -40
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +289 -2
- package/dist/typings/knex.d.js +3 -2
- package/dist/utils/async-utils.d.ts +7 -0
- package/dist/utils/async-utils.d.ts.map +1 -1
- package/dist/utils/async-utils.js +57 -2
- package/dist/utils/console-util.d.ts +2 -0
- package/dist/utils/console-util.d.ts.map +1 -0
- package/dist/utils/console-util.js +6 -0
- package/dist/utils/controller.js +26 -2
- package/dist/utils/esm-utils.d.ts +45 -0
- package/dist/utils/esm-utils.d.ts.map +1 -0
- package/dist/utils/esm-utils.js +56 -0
- package/dist/utils/fs-utils.js +17 -2
- package/dist/utils/lodash-able.js +6 -2
- package/dist/utils/model.js +22 -2
- package/dist/utils/path-utils.d.ts +89 -0
- package/dist/utils/path-utils.d.ts.map +1 -0
- package/dist/utils/path-utils.js +60 -0
- package/dist/utils/process-utils.d.ts +13 -0
- package/dist/utils/process-utils.d.ts.map +1 -0
- package/dist/utils/process-utils.js +36 -0
- package/dist/utils/sql-parser.js +35 -2
- package/dist/utils/utils.d.ts +4 -7
- package/dist/utils/utils.d.ts.map +1 -1
- package/dist/utils/utils.js +33 -2
- package/dist/utils/zod-error.d.ts.map +1 -1
- package/dist/utils/zod-error.js +19 -2
- package/package.json +21 -8
- package/src/api/code-converters.ts +2 -2
- package/src/api/config.ts +142 -0
- package/src/api/context.ts +1 -0
- package/src/api/decorators.ts +1 -0
- package/src/api/sonamu.ts +81 -67
- package/src/bin/build-config.ts +2 -1
- package/src/bin/cli-wrapper.ts +10 -3
- package/src/bin/cli.ts +108 -56
- package/src/bin/hot-hook-register.ts +22 -0
- package/src/database/base-model.ts +1 -1
- package/src/database/code-generator.ts +1 -1
- package/src/database/db.ts +10 -52
- package/src/database/puri.ts +78 -53
- package/src/database/puri.types.ts +18 -5
- package/src/database/upsert-builder.ts +1 -1
- package/src/entity/entity-manager.ts +19 -15
- package/src/entity/entity.ts +4 -3
- package/src/index.ts +2 -0
- package/src/migration/code-generation.ts +1 -1
- package/src/migration/migration-set.ts +1 -1
- package/src/migration/migrator.ts +23 -152
- package/src/naite/naite.ts +70 -0
- package/src/syncer/api-parser.ts +299 -0
- package/src/syncer/checksum.ts +152 -0
- package/src/syncer/code-generator.ts +202 -0
- package/src/syncer/entity-operations.ts +68 -0
- package/src/syncer/file-patterns.ts +56 -0
- package/src/syncer/index.ts +6 -0
- package/src/syncer/module-loader.ts +125 -0
- package/src/syncer/syncer.ts +363 -1420
- package/src/template/entity-converter.ts +123 -0
- package/src/template/helpers.ts +84 -0
- package/src/{templates → template/implementations}/entity.template.ts +4 -4
- package/src/{templates → template/implementations}/generated.template.ts +9 -9
- package/src/{templates → template/implementations}/generated_http.template.ts +9 -6
- package/src/{templates → template/implementations}/generated_sso.template.ts +7 -7
- package/src/{templates → template/implementations}/init_types.template.ts +4 -4
- package/src/{templates → template/implementations}/model.template.ts +9 -9
- package/src/{templates → template/implementations}/model_test.template.ts +5 -5
- package/src/{templates → template/implementations}/service.template.ts +19 -11
- package/src/{templates → template/implementations}/view_enums_buttonset.template.ts +3 -3
- package/src/{templates → template/implementations}/view_enums_dropdown.template.ts +5 -21
- package/src/{templates → template/implementations}/view_enums_select.template.ts +4 -4
- package/src/{templates → template/implementations}/view_form.template.ts +11 -13
- package/src/{templates → template/implementations}/view_id_all_select.template.ts +3 -3
- package/src/{templates → template/implementations}/view_id_async_select.template.ts +3 -3
- package/src/{templates → template/implementations}/view_list.template.ts +13 -64
- package/src/{templates → template/implementations}/view_list_columns.template.ts +3 -3
- package/src/{templates → template/implementations}/view_search_input.template.ts +3 -3
- package/src/template/index.ts +4 -0
- package/src/template/template.ts +86 -0
- package/src/template/zod-converter.ts +219 -0
- package/src/testing/fixture-manager.ts +8 -1
- package/src/types/types.ts +38 -61
- package/src/utils/async-utils.ts +17 -0
- package/src/utils/console-util.ts +4 -0
- package/src/utils/esm-utils.ts +69 -0
- package/src/utils/path-utils.ts +102 -0
- package/src/utils/process-utils.ts +46 -0
- package/src/utils/sql-parser.ts +1 -1
- package/src/utils/utils.ts +14 -40
- package/src/utils/zod-error.ts +0 -1
- package/dist/api/base-frame.js.map +0 -1
- package/dist/api/caster.js.map +0 -1
- package/dist/api/code-converters.js.map +0 -1
- package/dist/api/context.js.map +0 -1
- package/dist/api/decorators.js.map +0 -1
- package/dist/api/index.js.map +0 -1
- package/dist/api/sonamu.js.map +0 -1
- package/dist/bin/build-config.js.map +0 -1
- package/dist/bin/cli-wrapper.js.map +0 -1
- package/dist/bin/cli.js.map +0 -1
- package/dist/database/_batch_update.js.map +0 -1
- package/dist/database/base-model.js.map +0 -1
- package/dist/database/code-generator.js.map +0 -1
- package/dist/database/db.js.map +0 -1
- package/dist/database/knex-plugins/knex-on-duplicate-update.js.map +0 -1
- package/dist/database/puri-wrapper.js.map +0 -1
- package/dist/database/puri.js.map +0 -1
- package/dist/database/puri.types.js.map +0 -1
- package/dist/database/transaction-context.js.map +0 -1
- package/dist/database/upsert-builder.js.map +0 -1
- package/dist/entity/entity-manager.js.map +0 -1
- package/dist/entity/entity-utils.js.map +0 -1
- package/dist/entity/entity.js.map +0 -1
- package/dist/exceptions/error-handler.js.map +0 -1
- package/dist/exceptions/so-exceptions.js.map +0 -1
- package/dist/file-storage/driver.js.map +0 -1
- package/dist/file-storage/file-storage.js.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/migration/code-generation.js.map +0 -1
- package/dist/migration/migration-set.js.map +0 -1
- package/dist/migration/migrator.js.map +0 -1
- package/dist/migration/types.js.map +0 -1
- package/dist/stream/index.js.map +0 -1
- package/dist/stream/sse.js.map +0 -1
- package/dist/syncer/index.js.map +0 -1
- package/dist/syncer/syncer.js.map +0 -1
- package/dist/templates/base-template.d.ts +0 -13
- package/dist/templates/base-template.d.ts.map +0 -1
- package/dist/templates/base-template.js +0 -2
- package/dist/templates/base-template.js.map +0 -1
- package/dist/templates/entity.template.d.ts.map +0 -1
- package/dist/templates/entity.template.js +0 -2
- package/dist/templates/entity.template.js.map +0 -1
- package/dist/templates/generated.template.d.ts.map +0 -1
- package/dist/templates/generated.template.js +0 -2
- package/dist/templates/generated.template.js.map +0 -1
- package/dist/templates/generated_http.template.d.ts.map +0 -1
- package/dist/templates/generated_http.template.js +0 -2
- package/dist/templates/generated_http.template.js.map +0 -1
- package/dist/templates/generated_sso.template.d.ts.map +0 -1
- package/dist/templates/generated_sso.template.js +0 -2
- package/dist/templates/generated_sso.template.js.map +0 -1
- package/dist/templates/index.d.ts +0 -2
- package/dist/templates/index.d.ts.map +0 -1
- package/dist/templates/index.js +0 -2
- package/dist/templates/index.js.map +0 -1
- package/dist/templates/init_types.template.d.ts.map +0 -1
- package/dist/templates/init_types.template.js +0 -2
- package/dist/templates/init_types.template.js.map +0 -1
- package/dist/templates/model.template.d.ts +0 -17
- package/dist/templates/model.template.d.ts.map +0 -1
- package/dist/templates/model.template.js +0 -2
- package/dist/templates/model.template.js.map +0 -1
- package/dist/templates/model_test.template.d.ts.map +0 -1
- package/dist/templates/model_test.template.js +0 -2
- package/dist/templates/model_test.template.js.map +0 -1
- package/dist/templates/service.template.d.ts.map +0 -1
- package/dist/templates/service.template.js +0 -2
- package/dist/templates/service.template.js.map +0 -1
- package/dist/templates/view_enums_buttonset.template.d.ts.map +0 -1
- package/dist/templates/view_enums_buttonset.template.js +0 -2
- package/dist/templates/view_enums_buttonset.template.js.map +0 -1
- package/dist/templates/view_enums_dropdown.template.d.ts.map +0 -1
- package/dist/templates/view_enums_dropdown.template.js +0 -2
- package/dist/templates/view_enums_dropdown.template.js.map +0 -1
- package/dist/templates/view_enums_select.template.d.ts.map +0 -1
- package/dist/templates/view_enums_select.template.js +0 -2
- package/dist/templates/view_enums_select.template.js.map +0 -1
- package/dist/templates/view_form.template.d.ts.map +0 -1
- package/dist/templates/view_form.template.js +0 -2
- package/dist/templates/view_form.template.js.map +0 -1
- package/dist/templates/view_id_all_select.template.d.ts.map +0 -1
- package/dist/templates/view_id_all_select.template.js +0 -2
- package/dist/templates/view_id_all_select.template.js.map +0 -1
- package/dist/templates/view_id_async_select.template.d.ts.map +0 -1
- package/dist/templates/view_id_async_select.template.js +0 -2
- package/dist/templates/view_id_async_select.template.js.map +0 -1
- package/dist/templates/view_list.template.d.ts.map +0 -1
- package/dist/templates/view_list.template.js +0 -2
- package/dist/templates/view_list.template.js.map +0 -1
- package/dist/templates/view_list_columns.template.d.ts.map +0 -1
- package/dist/templates/view_list_columns.template.js +0 -2
- package/dist/templates/view_list_columns.template.js.map +0 -1
- package/dist/templates/view_search_input.template.d.ts.map +0 -1
- package/dist/templates/view_search_input.template.js +0 -2
- package/dist/templates/view_search_input.template.js.map +0 -1
- package/dist/testing/_relation-graph.js.map +0 -1
- package/dist/testing/fixture-manager.js.map +0 -1
- package/dist/types/types.js.map +0 -1
- package/dist/typings/knex.d.js.map +0 -1
- package/dist/utils/async-utils.js.map +0 -1
- package/dist/utils/controller.js.map +0 -1
- package/dist/utils/fs-utils.js.map +0 -1
- package/dist/utils/lodash-able.js.map +0 -1
- package/dist/utils/model.js.map +0 -1
- package/dist/utils/sql-parser.js.map +0 -1
- package/dist/utils/utils.js.map +0 -1
- package/dist/utils/zod-error.js.map +0 -1
- package/src/templates/base-template.ts +0 -19
- package/src/templates/index.ts +0 -1
package/dist/migration/types.js
CHANGED
|
@@ -1,2 +1,3 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
1
|
+
export { };
|
|
2
|
+
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9taWdyYXRpb24vdHlwZXMudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgU29uYW11REJDb25maWcgfSBmcm9tIFwiLi4vZGF0YWJhc2UvZGJcIjtcbmltcG9ydCB7IEdlbk1pZ3JhdGlvbkNvZGUgfSBmcm9tIFwiLi4vdHlwZXMvdHlwZXNcIjtcblxuZXhwb3J0IHR5cGUgTWlncmF0aW9uQ29kZSA9IHtcbiAgbmFtZTogc3RyaW5nO1xuICBwYXRoOiBzdHJpbmc7XG59O1xuZXhwb3J0IHR5cGUgQ29ublN0cmluZyA9XG4gIGAke1wibXlzcWwyXCJ9Oi8vJHtzdHJpbmd9QCR7c3RyaW5nfToke251bWJlcn0vJHtzdHJpbmd9YDsgLy8gbXlzcWwyOi8vYWNjb3VudEBob3N0OnBvcnQvZGF0YWJhc2VcbmV4cG9ydCB0eXBlIE1pZ3JhdGlvblN0YXR1cyA9IHtcbiAgY29kZXM6IE1pZ3JhdGlvbkNvZGVbXTtcbiAgY29ubnM6IHtcbiAgICBuYW1lOiBzdHJpbmc7XG4gICAgY29ubktleToga2V5b2YgU29uYW11REJDb25maWc7XG4gICAgY29ublN0cmluZzogQ29ublN0cmluZztcbiAgICBjdXJyZW50VmVyc2lvbjogc3RyaW5nO1xuICAgIHN0YXR1czogc3RyaW5nIHwgbnVtYmVyO1xuICAgIHBlbmRpbmc6IHN0cmluZ1tdO1xuICB9W107XG4gIHByZXBhcmVkQ29kZXM6IEdlbk1pZ3JhdGlvbkNvZGVbXTtcbn07XG5leHBvcnQgdHlwZSBEQkNvbHVtbiA9IHtcbiAgRmllbGQ6IHN0cmluZztcbiAgVHlwZTogc3RyaW5nO1xuICBOdWxsOiBzdHJpbmc7XG4gIEtleTogc3RyaW5nO1xuICBEZWZhdWx0OiBzdHJpbmcgfCBudWxsO1xuICBFeHRyYTogc3RyaW5nO1xufTtcbmV4cG9ydCB0eXBlIERCSW5kZXggPSB7XG4gIFRhYmxlOiBzdHJpbmc7XG4gIE5vbl91bmlxdWU6IG51bWJlcjtcbiAgS2V5X25hbWU6IHN0cmluZztcbiAgU2VxX2luX2luZGV4OiBudW1iZXI7XG4gIENvbHVtbl9uYW1lOiBzdHJpbmc7XG4gIENvbGxhdGlvbjogc3RyaW5nIHwgbnVsbDtcbiAgQ2FyZGluYWxpdHk6IG51bWJlciB8IG51bGw7XG4gIFN1Yl9wYXJ0OiBudW1iZXIgfCBudWxsO1xuICBQYWNrZWQ6IHN0cmluZyB8IG51bGw7XG4gIE51bGw6IHN0cmluZztcbiAgSW5kZXhfdHlwZTogc3RyaW5nO1xuICBDb21tZW50OiBzdHJpbmc7XG4gIEluZGV4X2NvbW1lbnQ6IHN0cmluZztcbiAgVmlzaWJsZTogc3RyaW5nO1xuICBFeHByZXNzaW9uOiBzdHJpbmcgfCBudWxsO1xufTtcbmV4cG9ydCB0eXBlIERCRm9yZWlnbiA9IHtcbiAga2V5TmFtZTogc3RyaW5nO1xuICBmcm9tOiBzdHJpbmc7XG4gIHJlZmVyZW5jZXNUYWJsZTogc3RyaW5nO1xuICByZWZlcmVuY2VzRmllbGQ6IHN0cmluZztcbiAgb25EZWxldGU6IHN0cmluZztcbiAgb25VcGRhdGU6IHN0cmluZztcbn07XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBOENBLFdBT0UifQ==
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
export declare class Naite {
|
|
2
|
+
static t(name: string, value: any): void;
|
|
3
|
+
static get(name: string): any;
|
|
4
|
+
static getAll(): {
|
|
5
|
+
[key: string]: any;
|
|
6
|
+
};
|
|
7
|
+
static d(_message: string): void;
|
|
8
|
+
static i(_message: string): void;
|
|
9
|
+
static w(_message: string): void;
|
|
10
|
+
static e(_message: string): void;
|
|
11
|
+
}
|
|
12
|
+
//# sourceMappingURL=naite.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"naite.d.ts","sourceRoot":"","sources":["../../src/naite/naite.ts"],"names":[],"mappings":"AAEA,qBAAa,KAAK;IAEhB,MAAM,CAAC,CAAC,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,GAAG;IAuBjC,MAAM,CAAC,GAAG,CAAC,IAAI,EAAE,MAAM,GAAG,GAAG;IAS7B,MAAM,CAAC,MAAM,IAAI;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAA;KAAE;IAiBvC,MAAM,CAAC,CAAC,CAAC,QAAQ,EAAE,MAAM;IAIzB,MAAM,CAAC,CAAC,CAAC,QAAQ,EAAE,MAAM;IAIzB,MAAM,CAAC,CAAC,CAAC,QAAQ,EAAE,MAAM;IAIzB,MAAM,CAAC,CAAC,CAAC,QAAQ,EAAE,MAAM;CAI1B"}
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { Sonamu } from "../api/sonamu.js";
|
|
2
|
+
export class Naite {
|
|
3
|
+
// 테스트 로그 기록
|
|
4
|
+
static t(name, value) {
|
|
5
|
+
const context = Sonamu.getContext();
|
|
6
|
+
const store = context?.naiteStore;
|
|
7
|
+
if (!store) {
|
|
8
|
+
return;
|
|
9
|
+
}
|
|
10
|
+
if (store.has(name)) {
|
|
11
|
+
// 이미 값이 있는 경우
|
|
12
|
+
if (Array.isArray(store.get(name))) {
|
|
13
|
+
// 배열에 추가
|
|
14
|
+
store.set(name, [
|
|
15
|
+
...store.get(name),
|
|
16
|
+
value
|
|
17
|
+
]);
|
|
18
|
+
} else {
|
|
19
|
+
// 배열이 아닌 경우 배열로 변환
|
|
20
|
+
store.set(name, [
|
|
21
|
+
store.get(name),
|
|
22
|
+
value
|
|
23
|
+
]);
|
|
24
|
+
}
|
|
25
|
+
} else {
|
|
26
|
+
// 값이 없는 경우 추가
|
|
27
|
+
store.set(name, value);
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
// 테스트에서 값 가져오기
|
|
31
|
+
static get(name) {
|
|
32
|
+
const context = Sonamu.getContext();
|
|
33
|
+
if (!context?.naiteStore || !context.naiteStore.has(name)) {
|
|
34
|
+
throw new Error(`Naite.get: \`${name}\` not found`);
|
|
35
|
+
}
|
|
36
|
+
return context?.naiteStore?.get(name);
|
|
37
|
+
}
|
|
38
|
+
// 전체 리스트 가져오기
|
|
39
|
+
static getAll() {
|
|
40
|
+
const context = Sonamu.getContext();
|
|
41
|
+
if (!context?.naiteStore) {
|
|
42
|
+
return {};
|
|
43
|
+
}
|
|
44
|
+
return Object.fromEntries(context.naiteStore.entries());
|
|
45
|
+
}
|
|
46
|
+
// expect 래퍼 (CJS-ESM 이슈로 현재 사용불가)
|
|
47
|
+
// static expect(name: string) {
|
|
48
|
+
// if (!this.vitestExpect) {
|
|
49
|
+
// throw new Error("Vitest is not initialized");
|
|
50
|
+
// }
|
|
51
|
+
// return this.vitestExpect(this.get(name));
|
|
52
|
+
// }
|
|
53
|
+
// 일반 로그 레벨
|
|
54
|
+
static d(_message) {
|
|
55
|
+
// TODO: Logger 연결
|
|
56
|
+
console.log(`[DEBUG] ${_message}`);
|
|
57
|
+
}
|
|
58
|
+
static i(_message) {
|
|
59
|
+
// TODO: Logger 연결
|
|
60
|
+
console.log(`[INFO] ${_message}`);
|
|
61
|
+
}
|
|
62
|
+
static w(_message) {
|
|
63
|
+
// TODO: Logger 연결
|
|
64
|
+
console.log(`[WARN] ${_message}`);
|
|
65
|
+
}
|
|
66
|
+
static e(_message) {
|
|
67
|
+
// TODO: Logger 연결
|
|
68
|
+
console.log(`[ERROR] ${_message}`);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9uYWl0ZS9uYWl0ZS50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTb25hbXUgfSBmcm9tIFwiLi4vYXBpL3NvbmFtdVwiO1xuXG5leHBvcnQgY2xhc3MgTmFpdGUge1xuICAvLyDthYzsiqTtirgg66Gc6re4IOq4sOuhnVxuICBzdGF0aWMgdChuYW1lOiBzdHJpbmcsIHZhbHVlOiBhbnkpIHtcbiAgICBjb25zdCBjb250ZXh0ID0gU29uYW11LmdldENvbnRleHQoKTtcbiAgICBjb25zdCBzdG9yZSA9IGNvbnRleHQ/Lm5haXRlU3RvcmU7XG5cbiAgICBpZiAoIXN0b3JlKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGlmIChzdG9yZS5oYXMobmFtZSkpIHtcbiAgICAgIC8vIOydtOuvuCDqsJLsnbQg7J6I64qUIOqyveyasFxuICAgICAgaWYgKEFycmF5LmlzQXJyYXkoc3RvcmUuZ2V0KG5hbWUpKSkge1xuICAgICAgICAvLyDrsLDsl7Tsl5Ag7LaU6rCAXG4gICAgICAgIHN0b3JlLnNldChuYW1lLCBbLi4uc3RvcmUuZ2V0KG5hbWUpLCB2YWx1ZV0pO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgLy8g67Cw7Je07J20IOyVhOuLjCDqsr3smrAg67Cw7Je066GcIOuzgO2ZmFxuICAgICAgICBzdG9yZS5zZXQobmFtZSwgW3N0b3JlLmdldChuYW1lKSwgdmFsdWVdKTtcbiAgICAgIH1cbiAgICB9IGVsc2Uge1xuICAgICAgLy8g6rCS7J20IOyXhuuKlCDqsr3smrAg7LaU6rCAXG4gICAgICBzdG9yZS5zZXQobmFtZSwgdmFsdWUpO1xuICAgIH1cbiAgfVxuXG4gIC8vIO2FjOyKpO2KuOyXkOyEnCDqsJIg6rCA7KC47Jik6riwXG4gIHN0YXRpYyBnZXQobmFtZTogc3RyaW5nKTogYW55IHtcbiAgICBjb25zdCBjb250ZXh0ID0gU29uYW11LmdldENvbnRleHQoKTtcbiAgICBpZiAoIWNvbnRleHQ/Lm5haXRlU3RvcmUgfHwgIWNvbnRleHQubmFpdGVTdG9yZS5oYXMobmFtZSkpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgTmFpdGUuZ2V0OiBcXGAke25hbWV9XFxgIG5vdCBmb3VuZGApO1xuICAgIH1cbiAgICByZXR1cm4gY29udGV4dD8ubmFpdGVTdG9yZT8uZ2V0KG5hbWUpO1xuICB9XG5cbiAgLy8g7KCE7LK0IOumrOyKpO2KuCDqsIDsoLjsmKTquLBcbiAgc3RhdGljIGdldEFsbCgpOiB7IFtrZXk6IHN0cmluZ106IGFueSB9IHtcbiAgICBjb25zdCBjb250ZXh0ID0gU29uYW11LmdldENvbnRleHQoKTtcbiAgICBpZiAoIWNvbnRleHQ/Lm5haXRlU3RvcmUpIHtcbiAgICAgIHJldHVybiB7fTtcbiAgICB9XG4gICAgcmV0dXJuIE9iamVjdC5mcm9tRW50cmllcyhjb250ZXh0Lm5haXRlU3RvcmUuZW50cmllcygpKTtcbiAgfVxuXG4gIC8vIGV4cGVjdCDrnpjtjbwgKENKUy1FU00g7J207IqI66GcIO2YhOyerCDsgqzsmqnrtojqsIApXG4gIC8vIHN0YXRpYyBleHBlY3QobmFtZTogc3RyaW5nKSB7XG4gIC8vICAgaWYgKCF0aGlzLnZpdGVzdEV4cGVjdCkge1xuICAvLyAgICAgdGhyb3cgbmV3IEVycm9yKFwiVml0ZXN0IGlzIG5vdCBpbml0aWFsaXplZFwiKTtcbiAgLy8gICB9XG4gIC8vICAgcmV0dXJuIHRoaXMudml0ZXN0RXhwZWN0KHRoaXMuZ2V0KG5hbWUpKTtcbiAgLy8gfVxuXG4gIC8vIOydvOuwmCDroZzqt7gg66CI67KoXG4gIHN0YXRpYyBkKF9tZXNzYWdlOiBzdHJpbmcpIHtcbiAgICAvLyBUT0RPOiBMb2dnZXIg7Jew6rKwXG4gICAgY29uc29sZS5sb2coYFtERUJVR10gJHtfbWVzc2FnZX1gKTtcbiAgfVxuICBzdGF0aWMgaShfbWVzc2FnZTogc3RyaW5nKSB7XG4gICAgLy8gVE9ETzogTG9nZ2VyIOyXsOqysFxuICAgIGNvbnNvbGUubG9nKGBbSU5GT10gJHtfbWVzc2FnZX1gKTtcbiAgfVxuICBzdGF0aWMgdyhfbWVzc2FnZTogc3RyaW5nKSB7XG4gICAgLy8gVE9ETzogTG9nZ2VyIOyXsOqysFxuICAgIGNvbnNvbGUubG9nKGBbV0FSTl0gJHtfbWVzc2FnZX1gKTtcbiAgfVxuICBzdGF0aWMgZShfbWVzc2FnZTogc3RyaW5nKSB7XG4gICAgLy8gVE9ETzogTG9nZ2VyIOyXsOqysFxuICAgIGNvbnNvbGUubG9nKGBbRVJST1JdICR7X21lc3NhZ2V9YCk7XG4gIH1cbn1cbiJdLCJuYW1lcyI6WyJTb25hbXUiLCJOYWl0ZSIsInQiLCJuYW1lIiwidmFsdWUiLCJjb250ZXh0IiwiZ2V0Q29udGV4dCIsInN0b3JlIiwibmFpdGVTdG9yZSIsImhhcyIsIkFycmF5IiwiaXNBcnJheSIsImdldCIsInNldCIsIkVycm9yIiwiZ2V0QWxsIiwiT2JqZWN0IiwiZnJvbUVudHJpZXMiLCJlbnRyaWVzIiwiZCIsIl9tZXNzYWdlIiwiY29uc29sZSIsImxvZyIsImkiLCJ3IiwiZSJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsTUFBTSxRQUFRLG1CQUFnQjtBQUV2QyxPQUFPLE1BQU1DO0lBQ1gsWUFBWTtJQUNaLE9BQU9DLEVBQUVDLElBQVksRUFBRUMsS0FBVSxFQUFFO1FBQ2pDLE1BQU1DLFVBQVVMLE9BQU9NLFVBQVU7UUFDakMsTUFBTUMsUUFBUUYsU0FBU0c7UUFFdkIsSUFBSSxDQUFDRCxPQUFPO1lBQ1Y7UUFDRjtRQUNBLElBQUlBLE1BQU1FLEdBQUcsQ0FBQ04sT0FBTztZQUNuQixjQUFjO1lBQ2QsSUFBSU8sTUFBTUMsT0FBTyxDQUFDSixNQUFNSyxHQUFHLENBQUNULFFBQVE7Z0JBQ2xDLFNBQVM7Z0JBQ1RJLE1BQU1NLEdBQUcsQ0FBQ1YsTUFBTTt1QkFBSUksTUFBTUssR0FBRyxDQUFDVDtvQkFBT0M7aUJBQU07WUFDN0MsT0FBTztnQkFDTCxtQkFBbUI7Z0JBQ25CRyxNQUFNTSxHQUFHLENBQUNWLE1BQU07b0JBQUNJLE1BQU1LLEdBQUcsQ0FBQ1Q7b0JBQU9DO2lCQUFNO1lBQzFDO1FBQ0YsT0FBTztZQUNMLGNBQWM7WUFDZEcsTUFBTU0sR0FBRyxDQUFDVixNQUFNQztRQUNsQjtJQUNGO0lBRUEsZUFBZTtJQUNmLE9BQU9RLElBQUlULElBQVksRUFBTztRQUM1QixNQUFNRSxVQUFVTCxPQUFPTSxVQUFVO1FBQ2pDLElBQUksQ0FBQ0QsU0FBU0csY0FBYyxDQUFDSCxRQUFRRyxVQUFVLENBQUNDLEdBQUcsQ0FBQ04sT0FBTztZQUN6RCxNQUFNLElBQUlXLE1BQU0sQ0FBQyxhQUFhLEVBQUVYLEtBQUssWUFBWSxDQUFDO1FBQ3BEO1FBQ0EsT0FBT0UsU0FBU0csWUFBWUksSUFBSVQ7SUFDbEM7SUFFQSxjQUFjO0lBQ2QsT0FBT1ksU0FBaUM7UUFDdEMsTUFBTVYsVUFBVUwsT0FBT00sVUFBVTtRQUNqQyxJQUFJLENBQUNELFNBQVNHLFlBQVk7WUFDeEIsT0FBTyxDQUFDO1FBQ1Y7UUFDQSxPQUFPUSxPQUFPQyxXQUFXLENBQUNaLFFBQVFHLFVBQVUsQ0FBQ1UsT0FBTztJQUN0RDtJQUVBLGtDQUFrQztJQUNsQyxnQ0FBZ0M7SUFDaEMsOEJBQThCO0lBQzlCLG9EQUFvRDtJQUNwRCxNQUFNO0lBQ04sOENBQThDO0lBQzlDLElBQUk7SUFFSixXQUFXO0lBQ1gsT0FBT0MsRUFBRUMsUUFBZ0IsRUFBRTtRQUN6QixrQkFBa0I7UUFDbEJDLFFBQVFDLEdBQUcsQ0FBQyxDQUFDLFFBQVEsRUFBRUYsVUFBVTtJQUNuQztJQUNBLE9BQU9HLEVBQUVILFFBQWdCLEVBQUU7UUFDekIsa0JBQWtCO1FBQ2xCQyxRQUFRQyxHQUFHLENBQUMsQ0FBQyxPQUFPLEVBQUVGLFVBQVU7SUFDbEM7SUFDQSxPQUFPSSxFQUFFSixRQUFnQixFQUFFO1FBQ3pCLGtCQUFrQjtRQUNsQkMsUUFBUUMsR0FBRyxDQUFDLENBQUMsT0FBTyxFQUFFRixVQUFVO0lBQ2xDO0lBQ0EsT0FBT0ssRUFBRUwsUUFBZ0IsRUFBRTtRQUN6QixrQkFBa0I7UUFDbEJDLFFBQVFDLEdBQUcsQ0FBQyxDQUFDLFFBQVEsRUFBRUYsVUFBVTtJQUNuQztBQUNGIn0=
|
package/dist/stream/index.js
CHANGED
|
@@ -1,2 +1,3 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
1
|
+
export * from "./sse.js";
|
|
2
|
+
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zdHJlYW0vaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vc3NlXCI7XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxXQUFRIn0=
|
package/dist/stream/sse.js
CHANGED
|
@@ -1,2 +1,38 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
1
|
+
// NOTE(Haze, 251106): context provider에서 인자를 채워주면 createSSE(events)만으로 사용 가능
|
|
2
|
+
export function createSSEFactory(socket, reply, _events) {
|
|
3
|
+
return new SSEConnection(socket, reply);
|
|
4
|
+
}
|
|
5
|
+
class SSEConnection {
|
|
6
|
+
socket;
|
|
7
|
+
reply;
|
|
8
|
+
_closed = false;
|
|
9
|
+
constructor(socket, reply){
|
|
10
|
+
this.socket = socket;
|
|
11
|
+
this.reply = reply;
|
|
12
|
+
this.socket.on("close", ()=>{
|
|
13
|
+
this._closed = true;
|
|
14
|
+
});
|
|
15
|
+
}
|
|
16
|
+
publish(event, data) {
|
|
17
|
+
if (this._closed) {
|
|
18
|
+
return;
|
|
19
|
+
}
|
|
20
|
+
this.reply.sse({
|
|
21
|
+
event: event,
|
|
22
|
+
data: JSON.stringify(data)
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
async end() {
|
|
26
|
+
if (this._closed) {
|
|
27
|
+
return;
|
|
28
|
+
}
|
|
29
|
+
this.reply.sse({
|
|
30
|
+
event: "end",
|
|
31
|
+
data: "END"
|
|
32
|
+
});
|
|
33
|
+
await new Promise((resolve)=>setTimeout(resolve, 200));
|
|
34
|
+
this.reply.raw.end();
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zdHJlYW0vc3NlLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHogfSBmcm9tIFwiem9kXCI7XG5pbXBvcnQgdHlwZSB7IEZhc3RpZnlSZXF1ZXN0LCBGYXN0aWZ5UmVwbHkgfSBmcm9tIFwiZmFzdGlmeVwiO1xuXG4vLyBOT1RFKEhhemUsIDI1MTEwNik6IGNvbnRleHQgcHJvdmlkZXLsl5DshJwg7J247J6Q66W8IOyxhOybjOyjvOuptCBjcmVhdGVTU0UoZXZlbnRzKeunjOycvOuhnCDsgqzsmqkg6rCA64qlXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlU1NFRmFjdG9yeTxUIGV4dGVuZHMgei5ab2RPYmplY3Q+KFxuICBzb2NrZXQ6IEZhc3RpZnlSZXF1ZXN0W1wic29ja2V0XCJdLFxuICByZXBseTogRmFzdGlmeVJlcGx5LFxuICBfZXZlbnRzOiBUXG4pIHtcbiAgcmV0dXJuIG5ldyBTU0VDb25uZWN0aW9uPFQ+KHNvY2tldCwgcmVwbHkpO1xufVxuXG5jbGFzcyBTU0VDb25uZWN0aW9uPFQgZXh0ZW5kcyB6LlpvZE9iamVjdD4ge1xuICBwcml2YXRlIF9jbG9zZWQgPSBmYWxzZTtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIHJlYWRvbmx5IHNvY2tldDogRmFzdGlmeVJlcXVlc3RbXCJzb2NrZXRcIl0sXG4gICAgcHJpdmF0ZSByZWFkb25seSByZXBseTogRmFzdGlmeVJlcGx5XG4gICkge1xuICAgIHRoaXMuc29ja2V0Lm9uKFwiY2xvc2VcIiwgKCkgPT4ge1xuICAgICAgdGhpcy5fY2xvc2VkID0gdHJ1ZTtcbiAgICB9KTtcbiAgfVxuXG4gIHB1Ymxpc2g8SyBleHRlbmRzIGtleW9mIHouaW5mZXI8VD4+KGV2ZW50OiBLLCBkYXRhOiB6LmluZmVyPFQ+W0tdKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuX2Nsb3NlZCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIHRoaXMucmVwbHkuc3NlKHtcbiAgICAgIGV2ZW50OiBldmVudCBhcyBzdHJpbmcsXG4gICAgICBkYXRhOiBKU09OLnN0cmluZ2lmeShkYXRhKSxcbiAgICB9KTtcbiAgfVxuXG4gIGFzeW5jIGVuZCgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBpZiAodGhpcy5fY2xvc2VkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgdGhpcy5yZXBseS5zc2Uoe1xuICAgICAgZXZlbnQ6IFwiZW5kXCIsXG4gICAgICBkYXRhOiBcIkVORFwiLFxuICAgIH0pO1xuICAgIFxuICAgIGF3YWl0IG5ldyBQcm9taXNlKChyZXNvbHZlKSA9PiBzZXRUaW1lb3V0KHJlc29sdmUsIDIwMCkpO1xuICAgIHRoaXMucmVwbHkucmF3LmVuZCgpO1xuICB9XG59XG4iXSwibmFtZXMiOlsiY3JlYXRlU1NFRmFjdG9yeSIsInNvY2tldCIsInJlcGx5IiwiX2V2ZW50cyIsIlNTRUNvbm5lY3Rpb24iLCJfY2xvc2VkIiwib24iLCJwdWJsaXNoIiwiZXZlbnQiLCJkYXRhIiwic3NlIiwiSlNPTiIsInN0cmluZ2lmeSIsImVuZCIsIlByb21pc2UiLCJyZXNvbHZlIiwic2V0VGltZW91dCIsInJhdyJdLCJtYXBwaW5ncyI6IkFBR0EsNkVBQTZFO0FBQzdFLE9BQU8sU0FBU0EsaUJBQ2RDLE1BQWdDLEVBQ2hDQyxLQUFtQixFQUNuQkMsT0FBVTtJQUVWLE9BQU8sSUFBSUMsY0FBaUJILFFBQVFDO0FBQ3RDO0FBRUEsTUFBTUU7OztJQUNJQyxVQUFVLE1BQU07SUFFeEIsWUFDRSxBQUFpQkosTUFBZ0MsRUFDakQsQUFBaUJDLEtBQW1CLENBQ3BDO2FBRmlCRCxTQUFBQTthQUNBQyxRQUFBQTtRQUVqQixJQUFJLENBQUNELE1BQU0sQ0FBQ0ssRUFBRSxDQUFDLFNBQVM7WUFDdEIsSUFBSSxDQUFDRCxPQUFPLEdBQUc7UUFDakI7SUFDRjtJQUVBRSxRQUFvQ0MsS0FBUSxFQUFFQyxJQUFtQixFQUFRO1FBQ3ZFLElBQUksSUFBSSxDQUFDSixPQUFPLEVBQUU7WUFDaEI7UUFDRjtRQUVBLElBQUksQ0FBQ0gsS0FBSyxDQUFDUSxHQUFHLENBQUM7WUFDYkYsT0FBT0E7WUFDUEMsTUFBTUUsS0FBS0MsU0FBUyxDQUFDSDtRQUN2QjtJQUNGO0lBRUEsTUFBTUksTUFBcUI7UUFDekIsSUFBSSxJQUFJLENBQUNSLE9BQU8sRUFBRTtZQUNoQjtRQUNGO1FBRUEsSUFBSSxDQUFDSCxLQUFLLENBQUNRLEdBQUcsQ0FBQztZQUNiRixPQUFPO1lBQ1BDLE1BQU07UUFDUjtRQUVBLE1BQU0sSUFBSUssUUFBUSxDQUFDQyxVQUFZQyxXQUFXRCxTQUFTO1FBQ25ELElBQUksQ0FBQ2IsS0FBSyxDQUFDZSxHQUFHLENBQUNKLEdBQUc7SUFDcEI7QUFDRiJ9
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { ApiParam, ApiParamType } from "../types/types";
|
|
2
|
+
import { AbsolutePath } from "../utils/path-utils";
|
|
3
|
+
/**
|
|
4
|
+
* TypeScript 파일을 파싱하여 API 메소드 정보를 추출합니다.
|
|
5
|
+
* @api 데코레이터가 붙은 메소드들의 타입 정보를 분석합니다.
|
|
6
|
+
* @param filePath - 파싱할 TypeScript 파일의 절대 경로
|
|
7
|
+
* @returns API 메소드 정보 배열 (타입 파라미터, 파라미터, 리턴 타입 등)
|
|
8
|
+
*/
|
|
9
|
+
export declare function readApisFromFile(filePath: AbsolutePath): Promise<{
|
|
10
|
+
typeParameters: ApiParamType.TypeParam[];
|
|
11
|
+
parameters: ApiParam[];
|
|
12
|
+
returnType: ApiParamType;
|
|
13
|
+
modelName: string;
|
|
14
|
+
methodName: string;
|
|
15
|
+
path: string;
|
|
16
|
+
options: import("..").ApiDecoratorOptions;
|
|
17
|
+
streamOptions?: import("..").StreamDecoratorOptions;
|
|
18
|
+
uploadOptions?: import("..").UploadDecoratorOptions;
|
|
19
|
+
}[]>;
|
|
20
|
+
//# sourceMappingURL=api-parser.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"api-parser.d.ts","sourceRoot":"","sources":["../../src/syncer/api-parser.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAOxD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAEnD;;;;;GAKG;AACH,wBAAsB,gBAAgB,CAAC,QAAQ,EAAE,YAAY;;;;;;;;;;KA4G5D"}
|
|
@@ -0,0 +1,229 @@
|
|
|
1
|
+
import { readFile } from "node:fs/promises";
|
|
2
|
+
import ts from "typescript";
|
|
3
|
+
import { registeredApis } from "../api/decorators.js";
|
|
4
|
+
import inflection from "inflection";
|
|
5
|
+
import assert from "assert";
|
|
6
|
+
/**
|
|
7
|
+
* TypeScript 파일을 파싱하여 API 메소드 정보를 추출합니다.
|
|
8
|
+
* @api 데코레이터가 붙은 메소드들의 타입 정보를 분석합니다.
|
|
9
|
+
* @param filePath - 파싱할 TypeScript 파일의 절대 경로
|
|
10
|
+
* @returns API 메소드 정보 배열 (타입 파라미터, 파라미터, 리턴 타입 등)
|
|
11
|
+
*/ export async function readApisFromFile(filePath) {
|
|
12
|
+
if (!filePath.endsWith(".ts")) {
|
|
13
|
+
throw new Error(`${filePath} does not seem to be a TypeScript file. Please check the file path. We only support parsing TypeScript files.`);
|
|
14
|
+
}
|
|
15
|
+
const sourceFile = ts.createSourceFile(filePath, (await readFile(filePath)).toString(), ts.ScriptTarget.Latest);
|
|
16
|
+
const methods = [];
|
|
17
|
+
let modelName = "UnknownModel";
|
|
18
|
+
let methodName = "unknownMethod";
|
|
19
|
+
const visitor = (node)=>{
|
|
20
|
+
if (ts.isClassDeclaration(node)) {
|
|
21
|
+
if (node.name && ts.isIdentifier(node.name)) {
|
|
22
|
+
modelName = node.name.escapedText.toString().replace(/Class$/, "");
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
if (ts.isMethodDeclaration(node)) {
|
|
26
|
+
if (ts.isIdentifier(node.name)) {
|
|
27
|
+
methodName = node.name.escapedText.toString();
|
|
28
|
+
}
|
|
29
|
+
const typeParameters = (node.typeParameters ?? []).map((typeParam)=>{
|
|
30
|
+
const tp = typeParam;
|
|
31
|
+
return {
|
|
32
|
+
t: "type-param",
|
|
33
|
+
id: tp.name.escapedText.toString(),
|
|
34
|
+
constraint: tp.constraint ? resolveTypeNode(tp.constraint) : undefined
|
|
35
|
+
};
|
|
36
|
+
});
|
|
37
|
+
const parameters = node.parameters.map((paramDec, index)=>{
|
|
38
|
+
const defaultDef = printNode(paramDec.initializer, sourceFile);
|
|
39
|
+
// 기본값이 있는 경우 paramDec.type가 undefined로 나옴
|
|
40
|
+
return resolveParamDec({
|
|
41
|
+
name: paramDec.name,
|
|
42
|
+
type: paramDec.type,
|
|
43
|
+
optional: paramDec.questionToken !== undefined || paramDec.initializer !== undefined,
|
|
44
|
+
defaultDef
|
|
45
|
+
}, index);
|
|
46
|
+
});
|
|
47
|
+
if (node.type === undefined) {
|
|
48
|
+
throw new Error(`리턴 타입이 기재되지 않은 메소드 ${modelName}.${methodName}`);
|
|
49
|
+
}
|
|
50
|
+
const returnType = resolveTypeNode(node.type);
|
|
51
|
+
methods.push({
|
|
52
|
+
modelName,
|
|
53
|
+
methodName,
|
|
54
|
+
typeParameters,
|
|
55
|
+
parameters,
|
|
56
|
+
returnType
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
ts.forEachChild(node, visitor);
|
|
60
|
+
};
|
|
61
|
+
visitor(sourceFile);
|
|
62
|
+
if (methods.length === 0) {
|
|
63
|
+
return [];
|
|
64
|
+
}
|
|
65
|
+
// 현재 파일의 등록된 API 필터
|
|
66
|
+
const currentModelApis = registeredApis.filter((api)=>{
|
|
67
|
+
return methods.find((method)=>method.modelName === api.modelName && method.methodName === api.methodName);
|
|
68
|
+
});
|
|
69
|
+
if (currentModelApis.length === 0) {
|
|
70
|
+
// const p = path.join(tmpdir(), "sonamu-syncer-error.json");
|
|
71
|
+
// writeFileSync(p, JSON.stringify(registeredApis, null, 2));
|
|
72
|
+
// execSync(`open ${p}`);
|
|
73
|
+
throw new Error(`현재 파일에 사전 등록된 API가 없습니다. ${filePath}`);
|
|
74
|
+
}
|
|
75
|
+
// 등록된 API에 현재 메소드 타입 정보 확장
|
|
76
|
+
const extendedApis = currentModelApis.map((api)=>{
|
|
77
|
+
const foundMethod = methods.find((method)=>method.modelName === api.modelName && method.methodName === api.methodName);
|
|
78
|
+
return {
|
|
79
|
+
...api,
|
|
80
|
+
typeParameters: foundMethod.typeParameters,
|
|
81
|
+
parameters: foundMethod.parameters,
|
|
82
|
+
returnType: foundMethod.returnType
|
|
83
|
+
};
|
|
84
|
+
});
|
|
85
|
+
return extendedApis;
|
|
86
|
+
}
|
|
87
|
+
function resolveTypeNode(typeNode) {
|
|
88
|
+
switch(typeNode?.kind){
|
|
89
|
+
case ts.SyntaxKind.AnyKeyword:
|
|
90
|
+
return "any";
|
|
91
|
+
case ts.SyntaxKind.UnknownKeyword:
|
|
92
|
+
return "unknown";
|
|
93
|
+
case ts.SyntaxKind.StringKeyword:
|
|
94
|
+
return "string";
|
|
95
|
+
case ts.SyntaxKind.NumberKeyword:
|
|
96
|
+
return "number";
|
|
97
|
+
case ts.SyntaxKind.BooleanKeyword:
|
|
98
|
+
return "boolean";
|
|
99
|
+
case ts.SyntaxKind.UndefinedKeyword:
|
|
100
|
+
return "undefined";
|
|
101
|
+
case ts.SyntaxKind.NullKeyword:
|
|
102
|
+
return "null";
|
|
103
|
+
case ts.SyntaxKind.VoidKeyword:
|
|
104
|
+
return "void";
|
|
105
|
+
case ts.SyntaxKind.LiteralType:
|
|
106
|
+
const literal = typeNode.literal;
|
|
107
|
+
if (ts.isStringLiteral(literal)) {
|
|
108
|
+
return {
|
|
109
|
+
t: "string-literal",
|
|
110
|
+
value: literal.text
|
|
111
|
+
};
|
|
112
|
+
} else if (ts.isNumericLiteral(literal)) {
|
|
113
|
+
return {
|
|
114
|
+
t: "numeric-literal",
|
|
115
|
+
value: Number(literal.text)
|
|
116
|
+
};
|
|
117
|
+
} else {
|
|
118
|
+
if (literal.kind === ts.SyntaxKind.NullKeyword) {
|
|
119
|
+
return "null";
|
|
120
|
+
} else if (literal.kind === ts.SyntaxKind.UndefinedKeyword) {
|
|
121
|
+
return "undefined";
|
|
122
|
+
} else if (literal.kind === ts.SyntaxKind.TrueKeyword) {
|
|
123
|
+
return "true";
|
|
124
|
+
} else if (literal.kind === ts.SyntaxKind.FalseKeyword) {
|
|
125
|
+
return "false";
|
|
126
|
+
}
|
|
127
|
+
throw new Error("알 수 없는 리터럴");
|
|
128
|
+
}
|
|
129
|
+
case ts.SyntaxKind.ArrayType:
|
|
130
|
+
const arrNode = typeNode;
|
|
131
|
+
return {
|
|
132
|
+
t: "array",
|
|
133
|
+
elementsType: resolveTypeNode(arrNode.elementType)
|
|
134
|
+
};
|
|
135
|
+
case ts.SyntaxKind.TypeLiteral:
|
|
136
|
+
const literalNode = typeNode;
|
|
137
|
+
return {
|
|
138
|
+
t: "object",
|
|
139
|
+
props: literalNode.members.map((member)=>{
|
|
140
|
+
if (ts.isIndexSignatureDeclaration(member)) {
|
|
141
|
+
assert(member.parameters[0]);
|
|
142
|
+
const res = resolveParamDec({
|
|
143
|
+
name: member.parameters[0].name,
|
|
144
|
+
type: member.parameters[0].type
|
|
145
|
+
});
|
|
146
|
+
return resolveParamDec({
|
|
147
|
+
name: {
|
|
148
|
+
escapedText: `[${res.name}${res.optional ? "?" : ""}: ${res.type}]`
|
|
149
|
+
},
|
|
150
|
+
type: member.type
|
|
151
|
+
});
|
|
152
|
+
} else {
|
|
153
|
+
return resolveParamDec({
|
|
154
|
+
name: member.name,
|
|
155
|
+
type: member.type,
|
|
156
|
+
optional: member.questionToken !== undefined
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
})
|
|
160
|
+
};
|
|
161
|
+
case ts.SyntaxKind.TypeReference:
|
|
162
|
+
return {
|
|
163
|
+
t: "ref",
|
|
164
|
+
id: typeNode.typeName.escapedText.toString(),
|
|
165
|
+
args: typeNode.typeArguments?.map((typeArg)=>resolveTypeNode(typeArg))
|
|
166
|
+
};
|
|
167
|
+
case ts.SyntaxKind.UnionType:
|
|
168
|
+
return {
|
|
169
|
+
t: "union",
|
|
170
|
+
types: typeNode.types.map((type)=>resolveTypeNode(type))
|
|
171
|
+
};
|
|
172
|
+
case ts.SyntaxKind.IntersectionType:
|
|
173
|
+
return {
|
|
174
|
+
t: "intersection",
|
|
175
|
+
types: typeNode.types.map((type)=>resolveTypeNode(type))
|
|
176
|
+
};
|
|
177
|
+
case ts.SyntaxKind.IndexedAccessType:
|
|
178
|
+
return {
|
|
179
|
+
t: "indexed-access",
|
|
180
|
+
object: resolveTypeNode(typeNode.objectType),
|
|
181
|
+
index: resolveTypeNode(typeNode.indexType)
|
|
182
|
+
};
|
|
183
|
+
case ts.SyntaxKind.TupleType:
|
|
184
|
+
if (ts.isTupleTypeNode(typeNode)) {
|
|
185
|
+
return {
|
|
186
|
+
t: "tuple-type",
|
|
187
|
+
elements: typeNode.elements.map((elem)=>resolveTypeNode(elem))
|
|
188
|
+
};
|
|
189
|
+
}
|
|
190
|
+
break;
|
|
191
|
+
case undefined:
|
|
192
|
+
throw new Error(`typeNode undefined`);
|
|
193
|
+
}
|
|
194
|
+
console.debug(typeNode);
|
|
195
|
+
throw new Error(`알 수 없는 SyntaxKind ${typeNode.kind}`);
|
|
196
|
+
}
|
|
197
|
+
function resolveParamDec(paramDec, index = 0) {
|
|
198
|
+
const name = paramDec.name;
|
|
199
|
+
const type = resolveTypeNode(paramDec.type);
|
|
200
|
+
if (name === undefined) {
|
|
201
|
+
console.debug({
|
|
202
|
+
name,
|
|
203
|
+
type,
|
|
204
|
+
paramDec
|
|
205
|
+
});
|
|
206
|
+
}
|
|
207
|
+
const result = {
|
|
208
|
+
name: name.escapedText ? name.escapedText.toString() : `nonameAt${index}`,
|
|
209
|
+
type,
|
|
210
|
+
optional: paramDec.optional === true,
|
|
211
|
+
defaultDef: paramDec?.defaultDef
|
|
212
|
+
};
|
|
213
|
+
// 구조분해할당의 경우 타입이름 사용
|
|
214
|
+
if (ts.isObjectBindingPattern(name) && ts.isTypeReferenceNode(paramDec.type) && ts.isIdentifier(paramDec.type.typeName)) {
|
|
215
|
+
result.name = inflection.camelize(paramDec.type.typeName.text, true);
|
|
216
|
+
}
|
|
217
|
+
return result;
|
|
218
|
+
}
|
|
219
|
+
function printNode(node, sourceFile) {
|
|
220
|
+
if (node === undefined) {
|
|
221
|
+
return undefined;
|
|
222
|
+
}
|
|
223
|
+
const printer = ts.createPrinter({
|
|
224
|
+
newLine: ts.NewLineKind.LineFeed
|
|
225
|
+
});
|
|
226
|
+
return printer.printNode(ts.EmitHint.Unspecified, node, sourceFile);
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/syncer/api-parser.ts"],"sourcesContent":["import { ApiParam, ApiParamType } from \"../types/types\";\nimport { readFile } from \"fs/promises\";\nimport * as _ from \"lodash-es\";\nimport ts from \"typescript\";\nimport { ExtendedApi, registeredApis } from \"../api/decorators\";\nimport inflection from \"inflection\";\nimport assert from \"assert\";\nimport { AbsolutePath } from \"../utils/path-utils\";\n\n/**\n * TypeScript 파일을 파싱하여 API 메소드 정보를 추출합니다.\n * @api 데코레이터가 붙은 메소드들의 타입 정보를 분석합니다.\n * @param filePath - 파싱할 TypeScript 파일의 절대 경로\n * @returns API 메소드 정보 배열 (타입 파라미터, 파라미터, 리턴 타입 등)\n */\nexport async function readApisFromFile(filePath: AbsolutePath) {\n  if (!filePath.endsWith(\".ts\")) { \n    throw new Error(`${filePath} does not seem to be a TypeScript file. Please check the file path. We only support parsing TypeScript files.`);\n  }\n  \n  const sourceFile = ts.createSourceFile(\n    filePath,\n    (await readFile(filePath)).toString(),\n    ts.ScriptTarget.Latest\n  );\n\n  const methods: Omit<ExtendedApi, \"path\" | \"options\">[] = [];\n  let modelName: string = \"UnknownModel\";\n  let methodName: string = \"unknownMethod\";\n  const visitor = (node: ts.Node) => {\n    if (ts.isClassDeclaration(node)) {\n      if (node.name && ts.isIdentifier(node.name)) {\n        modelName = node.name.escapedText.toString().replace(/Class$/, \"\");\n      }\n    }\n    if (ts.isMethodDeclaration(node)) {\n      if (ts.isIdentifier(node.name)) {\n        methodName = node.name.escapedText.toString();\n      }\n\n      const typeParameters: ApiParamType.TypeParam[] = (\n        node.typeParameters ?? []\n      ).map((typeParam) => {\n        const tp = typeParam as ts.TypeParameterDeclaration;\n\n        return {\n          t: \"type-param\",\n          id: tp.name.escapedText.toString(),\n          constraint: tp.constraint\n            ? resolveTypeNode(tp.constraint)\n            : undefined,\n        };\n      });\n      const parameters: ApiParam[] = node.parameters.map((paramDec, index) => {\n        const defaultDef = printNode(paramDec.initializer, sourceFile);\n\n        // 기본값이 있는 경우 paramDec.type가 undefined로 나옴\n\n        return resolveParamDec(\n          {\n            name: paramDec.name,\n            type: paramDec.type as ts.TypeNode,\n            optional:\n              paramDec.questionToken !== undefined ||\n              paramDec.initializer !== undefined,\n            defaultDef,\n          },\n          index\n        );\n      });\n      if (node.type === undefined) {\n        throw new Error(\n          `리턴 타입이 기재되지 않은 메소드 ${modelName}.${methodName}`\n        );\n      }\n      const returnType = resolveTypeNode(node.type!);\n\n      methods.push({\n        modelName,\n        methodName,\n        typeParameters,\n        parameters,\n        returnType,\n      });\n    }\n    ts.forEachChild(node, visitor);\n  };\n  visitor(sourceFile);\n\n  if (methods.length === 0) {\n    return [];\n  }\n\n  // 현재 파일의 등록된 API 필터\n  const currentModelApis = registeredApis.filter((api) => {\n    return methods.find(\n      (method) =>\n        method.modelName === api.modelName &&\n        method.methodName === api.methodName\n    );\n  });\n  if (currentModelApis.length === 0) {\n    // const p = path.join(tmpdir(), \"sonamu-syncer-error.json\");\n    // writeFileSync(p, JSON.stringify(registeredApis, null, 2));\n    // execSync(`open ${p}`);\n    throw new Error(`현재 파일에 사전 등록된 API가 없습니다. ${filePath}`);\n  }\n\n  // 등록된 API에 현재 메소드 타입 정보 확장\n  const extendedApis = currentModelApis.map((api) => {\n    const foundMethod = methods.find(\n      (method) =>\n        method.modelName === api.modelName &&\n        method.methodName === api.methodName\n    );\n    return {\n      ...api,\n      typeParameters: foundMethod!.typeParameters,\n      parameters: foundMethod!.parameters,\n      returnType: foundMethod!.returnType,\n    };\n  });\n  return extendedApis;\n}\n\nfunction resolveTypeNode(typeNode: ts.TypeNode): ApiParamType {\n  switch (typeNode?.kind) {\n    case ts.SyntaxKind.AnyKeyword:\n      return \"any\";\n    case ts.SyntaxKind.UnknownKeyword:\n      return \"unknown\";\n    case ts.SyntaxKind.StringKeyword:\n      return \"string\";\n    case ts.SyntaxKind.NumberKeyword:\n      return \"number\";\n    case ts.SyntaxKind.BooleanKeyword:\n      return \"boolean\";\n    case ts.SyntaxKind.UndefinedKeyword:\n      return \"undefined\";\n    case ts.SyntaxKind.NullKeyword:\n      return \"null\";\n    case ts.SyntaxKind.VoidKeyword:\n      return \"void\";\n    case ts.SyntaxKind.LiteralType:\n      const literal = (typeNode as ts.LiteralTypeNode).literal;\n      if (ts.isStringLiteral(literal)) {\n        return {\n          t: \"string-literal\",\n          value: literal.text,\n        };\n      } else if (ts.isNumericLiteral(literal)) {\n        return {\n          t: \"numeric-literal\",\n          value: Number(literal.text),\n        };\n      } else {\n        if (literal.kind === ts.SyntaxKind.NullKeyword) {\n          return \"null\";\n        } else if (literal.kind === ts.SyntaxKind.UndefinedKeyword) {\n          return \"undefined\";\n        } else if (literal.kind === ts.SyntaxKind.TrueKeyword) {\n          return \"true\";\n        } else if (literal.kind === ts.SyntaxKind.FalseKeyword) {\n          return \"false\";\n        }\n        throw new Error(\"알 수 없는 리터럴\");\n      }\n    case ts.SyntaxKind.ArrayType:\n      const arrNode = typeNode as ts.ArrayTypeNode;\n      return {\n        t: \"array\",\n        elementsType: resolveTypeNode(arrNode.elementType),\n      };\n    case ts.SyntaxKind.TypeLiteral:\n      const literalNode = typeNode as ts.TypeLiteralNode;\n      return {\n        t: \"object\",\n        props: literalNode.members.map((member) => {\n          if (ts.isIndexSignatureDeclaration(member)) {\n            assert(member.parameters[0]);\n            const res = resolveParamDec({\n              name: member.parameters[0].name as ts.Identifier,\n              type: member.parameters[0].type as ts.TypeNode,\n            });\n\n            return resolveParamDec({\n              name: {\n                escapedText: `[${res.name}${res.optional ? \"?\" : \"\"}: ${\n                  res.type\n                }]`,\n              } as ts.Identifier,\n              type: member.type as ts.TypeNode,\n            });\n          } else {\n            return resolveParamDec({\n              name: (member as ts.PropertySignature).name as ts.Identifier,\n              type: (member as ts.PropertySignature).type as ts.TypeNode,\n              optional:\n                (member as ts.PropertySignature).questionToken !== undefined,\n            });\n          }\n        }),\n      };\n    case ts.SyntaxKind.TypeReference:\n      return {\n        t: \"ref\",\n        id: (\n          (typeNode as ts.TypeReferenceNode).typeName as ts.Identifier\n        ).escapedText.toString(),\n        args: (typeNode as ts.TypeReferenceNode).typeArguments?.map((typeArg) =>\n          resolveTypeNode(typeArg)\n        ),\n      };\n    case ts.SyntaxKind.UnionType:\n      return {\n        t: \"union\",\n        types: (typeNode as ts.UnionTypeNode).types.map((type) =>\n          resolveTypeNode(type)\n        ),\n      };\n    case ts.SyntaxKind.IntersectionType:\n      return {\n        t: \"intersection\",\n        types: (typeNode as ts.IntersectionTypeNode).types.map((type) =>\n          resolveTypeNode(type)\n        ),\n      };\n    case ts.SyntaxKind.IndexedAccessType:\n      return {\n        t: \"indexed-access\",\n        object: resolveTypeNode(\n          (typeNode as ts.IndexedAccessTypeNode).objectType\n        ),\n        index: resolveTypeNode(\n          (typeNode as ts.IndexedAccessTypeNode).indexType\n        ),\n      };\n    case ts.SyntaxKind.TupleType:\n      if (ts.isTupleTypeNode(typeNode)) {\n        return {\n          t: \"tuple-type\",\n          elements: typeNode.elements.map((elem) => resolveTypeNode(elem)),\n        };\n      }\n      break;\n    case undefined:\n      throw new Error(`typeNode undefined`);\n  }\n\n  console.debug(typeNode);\n  throw new Error(`알 수 없는 SyntaxKind ${typeNode.kind}`);\n}\n\nfunction resolveParamDec(\n  paramDec: {\n    name: ts.BindingName;\n    type: ts.TypeNode;\n    optional?: boolean;\n    defaultDef?: string;\n  },\n  index: number = 0\n): ApiParam {\n  const name = paramDec.name as ts.Identifier;\n  const type = resolveTypeNode(paramDec.type);\n\n  if (name === undefined) {\n    console.debug({ name, type, paramDec });\n  }\n\n  const result: ApiParam = {\n    name: name.escapedText ? name.escapedText.toString() : `nonameAt${index}`,\n    type,\n    optional: paramDec.optional === true,\n    defaultDef: paramDec?.defaultDef,\n  };\n\n  // 구조분해할당의 경우 타입이름 사용\n  if (\n    ts.isObjectBindingPattern(name) &&\n    ts.isTypeReferenceNode(paramDec.type) &&\n    ts.isIdentifier(paramDec.type.typeName)\n  ) {\n    result.name = inflection.camelize(paramDec.type.typeName.text, true);\n  }\n\n  return result;\n}\n\nfunction printNode(\n  node: ts.Node | undefined,\n  sourceFile: ts.SourceFile\n): string | undefined {\n  if (node === undefined) {\n    return undefined;\n  }\n\n  const printer = ts.createPrinter({ newLine: ts.NewLineKind.LineFeed });\n  return printer.printNode(ts.EmitHint.Unspecified, node, sourceFile);\n}\n"],"names":["readFile","ts","registeredApis","inflection","assert","readApisFromFile","filePath","endsWith","Error","sourceFile","createSourceFile","toString","ScriptTarget","Latest","methods","modelName","methodName","visitor","node","isClassDeclaration","name","isIdentifier","escapedText","replace","isMethodDeclaration","typeParameters","map","typeParam","tp","t","id","constraint","resolveTypeNode","undefined","parameters","paramDec","index","defaultDef","printNode","initializer","resolveParamDec","type","optional","questionToken","returnType","push","forEachChild","length","currentModelApis","filter","api","find","method","extendedApis","foundMethod","typeNode","kind","SyntaxKind","AnyKeyword","UnknownKeyword","StringKeyword","NumberKeyword","BooleanKeyword","UndefinedKeyword","NullKeyword","VoidKeyword","LiteralType","literal","isStringLiteral","value","text","isNumericLiteral","Number","TrueKeyword","FalseKeyword","ArrayType","arrNode","elementsType","elementType","TypeLiteral","literalNode","props","members","member","isIndexSignatureDeclaration","res","TypeReference","typeName","args","typeArguments","typeArg","UnionType","types","IntersectionType","IndexedAccessType","object","objectType","indexType","TupleType","isTupleTypeNode","elements","elem","console","debug","result","isObjectBindingPattern","isTypeReferenceNode","camelize","printer","createPrinter","newLine","NewLineKind","LineFeed","EmitHint","Unspecified"],"mappings":"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"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { PathLike } from "fs";
|
|
2
|
+
import { AbsolutePath } from "../utils/path-utils";
|
|
3
|
+
/**
|
|
4
|
+
* 체크섬 파일에 저장된 내용과 현재 실제 파일의 체크섬을 비교하여 변경된 파일을 찾습니다.
|
|
5
|
+
* @returns 변경된 파일 경로 배열. 프로젝트 루트부터 슬래시로 시작합니다. 예시: "/src/application/user/user.model.ts"
|
|
6
|
+
*/
|
|
7
|
+
export declare function findChangedFilesUsingChecksums(): Promise<AbsolutePath[]>;
|
|
8
|
+
/**
|
|
9
|
+
* 체크섬을 갱신합니다.
|
|
10
|
+
* 현재 파일들의 체크섬을 계산해서 구한 다음, 체크섬 파일에 저장된 내용과 다르면 체크섬 파일을 갱신합니다.
|
|
11
|
+
*/
|
|
12
|
+
export declare function renewChecksums(): Promise<void>;
|
|
13
|
+
/**
|
|
14
|
+
* 두 파일의 내용이 같은지 체크섬으로 비교합니다.
|
|
15
|
+
* 만약 파일이 둘 중 하나라도 없다면 비교 불가로 false 반환합니다.
|
|
16
|
+
* @param one 파일 경로
|
|
17
|
+
* @param two 파일 경로
|
|
18
|
+
* @returns boolean
|
|
19
|
+
*/
|
|
20
|
+
export declare function areFilesSame(one: PathLike, two: PathLike): Promise<boolean>;
|
|
21
|
+
//# sourceMappingURL=checksum.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"checksum.d.ts","sourceRoot":"","sources":["../../src/syncer/checksum.ts"],"names":[],"mappings":"AAEA,OAAO,EAAoB,QAAQ,EAAE,MAAM,IAAI,CAAC;AAQhD,OAAO,EACL,YAAY,EAEb,MAAM,qBAAqB,CAAC;AAQ7B;;;GAGG;AACH,wBAAsB,8BAA8B,IAAI,OAAO,CAC7D,YAAY,EAAE,CACf,CAYA;AAED;;;GAGG;AACH,wBAAsB,cAAc,IAAI,OAAO,CAAC,IAAI,CAAC,CAUpD;AAED;;;;;;GAMG;AACH,wBAAsB,YAAY,CAChC,GAAG,EAAE,QAAQ,EACb,GAAG,EAAE,QAAQ,GACZ,OAAO,CAAC,OAAO,CAAC,CASlB"}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
import path from "path";
|
|
2
|
+
import { globAsync } from "../utils/async-utils.js";
|
|
3
|
+
import { createReadStream } from "fs";
|
|
4
|
+
import { readFile, writeFile } from "node:fs/promises";
|
|
5
|
+
import { exists } from "../utils/fs-utils.js";
|
|
6
|
+
import crypto from "crypto";
|
|
7
|
+
import equal from "fast-deep-equal";
|
|
8
|
+
import * as _ from "lodash-es";
|
|
9
|
+
import { Sonamu } from "../api/sonamu.js";
|
|
10
|
+
import { getChecksumPatternGroupInAbsolutePath } from "./file-patterns.js";
|
|
11
|
+
/**
|
|
12
|
+
* 체크섬 파일에 저장된 내용과 현재 실제 파일의 체크섬을 비교하여 변경된 파일을 찾습니다.
|
|
13
|
+
* @returns 변경된 파일 경로 배열. 프로젝트 루트부터 슬래시로 시작합니다. 예시: "/src/application/user/user.model.ts"
|
|
14
|
+
*/ export async function findChangedFilesUsingChecksums() {
|
|
15
|
+
const calculatedChecksums = await getCurrentChecksums();
|
|
16
|
+
const savedChecksums = await getPreviousChecksums();
|
|
17
|
+
const isSame = equal(calculatedChecksums, savedChecksums);
|
|
18
|
+
if (isSame) {
|
|
19
|
+
return [];
|
|
20
|
+
}
|
|
21
|
+
const diff = _.differenceWith(calculatedChecksums, savedChecksums, _.isEqual);
|
|
22
|
+
return diff.map((r)=>r.path);
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* 체크섬을 갱신합니다.
|
|
26
|
+
* 현재 파일들의 체크섬을 계산해서 구한 다음, 체크섬 파일에 저장된 내용과 다르면 체크섬 파일을 갱신합니다.
|
|
27
|
+
*/ export async function renewChecksums() {
|
|
28
|
+
const calculatedChecksums = await getCurrentChecksums();
|
|
29
|
+
const savedChecksums = await getPreviousChecksums();
|
|
30
|
+
const isSame = equal(calculatedChecksums, savedChecksums);
|
|
31
|
+
if (isSame) {
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
await saveChecksums(calculatedChecksums);
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* 두 파일의 내용이 같은지 체크섬으로 비교합니다.
|
|
38
|
+
* 만약 파일이 둘 중 하나라도 없다면 비교 불가로 false 반환합니다.
|
|
39
|
+
* @param one 파일 경로
|
|
40
|
+
* @param two 파일 경로
|
|
41
|
+
* @returns boolean
|
|
42
|
+
*/ export async function areFilesSame(one, two) {
|
|
43
|
+
if (!await exists(one) || !await exists(two)) {
|
|
44
|
+
return false;
|
|
45
|
+
}
|
|
46
|
+
const oneChecksum = await getChecksumOfFile(one);
|
|
47
|
+
const twoChecksum = await getChecksumOfFile(two);
|
|
48
|
+
return oneChecksum === twoChecksum;
|
|
49
|
+
}
|
|
50
|
+
async function getCurrentChecksums() {
|
|
51
|
+
const filePaths = (await Promise.all(Object.entries(getChecksumPatternGroupInAbsolutePath()).map(async ([_fileType, pattern])=>{
|
|
52
|
+
return globAsync(pattern);
|
|
53
|
+
}))).flat().sort();
|
|
54
|
+
const fileChecksums = await Promise.all(filePaths.map(async (filePath)=>{
|
|
55
|
+
return {
|
|
56
|
+
path: filePath,
|
|
57
|
+
checksum: await getChecksumOfFile(filePath)
|
|
58
|
+
};
|
|
59
|
+
}));
|
|
60
|
+
return fileChecksums;
|
|
61
|
+
}
|
|
62
|
+
async function getPreviousChecksums() {
|
|
63
|
+
const checksumFilePath = getChecksumFilePath();
|
|
64
|
+
if (!await exists(checksumFilePath)) {
|
|
65
|
+
return [];
|
|
66
|
+
}
|
|
67
|
+
const previousChecksums = JSON.parse(await readFile(checksumFilePath, "utf-8")).map((r)=>({
|
|
68
|
+
path: path.join(Sonamu.apiRootPath, r.path),
|
|
69
|
+
checksum: r.checksum
|
|
70
|
+
}));
|
|
71
|
+
return previousChecksums;
|
|
72
|
+
}
|
|
73
|
+
function getChecksumFilePath() {
|
|
74
|
+
return path.join(Sonamu.apiRootPath, "sonamu.lock");
|
|
75
|
+
}
|
|
76
|
+
async function saveChecksums(checksums) {
|
|
77
|
+
const checksumFilePath = getChecksumFilePath();
|
|
78
|
+
await writeFile(checksumFilePath, JSON.stringify(checksums.map((r)=>({
|
|
79
|
+
path: path.relative(Sonamu.apiRootPath, r.path),
|
|
80
|
+
checksum: r.checksum
|
|
81
|
+
})), null, 2), "utf-8");
|
|
82
|
+
console.log("checksum saved", checksumFilePath);
|
|
83
|
+
}
|
|
84
|
+
async function getChecksumOfFile(filePath) {
|
|
85
|
+
return new Promise((resolve, reject)=>{
|
|
86
|
+
const hash = crypto.createHash("sha1");
|
|
87
|
+
const input = createReadStream(filePath);
|
|
88
|
+
input.on("error", reject);
|
|
89
|
+
input.on("data", function(chunk) {
|
|
90
|
+
hash.update(chunk);
|
|
91
|
+
});
|
|
92
|
+
input.on("close", function() {
|
|
93
|
+
resolve(hash.digest("hex"));
|
|
94
|
+
});
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/syncer/checksum.ts"],"sourcesContent":["import path from \"path\";\nimport { globAsync } from \"../utils/async-utils\";\nimport { createReadStream, PathLike } from \"fs\";\nimport { readFile, writeFile } from \"fs/promises\";\nimport { exists } from \"../utils/fs-utils\";\n\nimport crypto from \"crypto\";\nimport equal from \"fast-deep-equal\";\nimport * as _ from \"lodash-es\";\nimport { Sonamu } from \"../api/sonamu\";\nimport {\n  AbsolutePath,\n  ApiRelativePath,\n} from \"../utils/path-utils\";\nimport { getChecksumPatternGroupInAbsolutePath } from \"./file-patterns\";\n\ntype PathAndChecksum = {\n  path: AbsolutePath;\n  checksum: string;\n};\n\n/**\n * 체크섬 파일에 저장된 내용과 현재 실제 파일의 체크섬을 비교하여 변경된 파일을 찾습니다.\n * @returns 변경된 파일 경로 배열. 프로젝트 루트부터 슬래시로 시작합니다. 예시: \"/src/application/user/user.model.ts\"\n */\nexport async function findChangedFilesUsingChecksums(): Promise<\n  AbsolutePath[]\n> {\n  const calculatedChecksums = await getCurrentChecksums();\n  const savedChecksums = await getPreviousChecksums();\n\n  const isSame = equal(calculatedChecksums, savedChecksums);\n  if (isSame) {\n    return [];\n  }\n\n  const diff = _.differenceWith(calculatedChecksums, savedChecksums, _.isEqual);\n\n  return diff.map((r) => r.path);\n}\n\n/**\n * 체크섬을 갱신합니다.\n * 현재 파일들의 체크섬을 계산해서 구한 다음, 체크섬 파일에 저장된 내용과 다르면 체크섬 파일을 갱신합니다.\n */\nexport async function renewChecksums(): Promise<void> {\n  const calculatedChecksums = await getCurrentChecksums();\n  const savedChecksums = await getPreviousChecksums();\n\n  const isSame = equal(calculatedChecksums, savedChecksums);\n  if (isSame) {\n    return;\n  }\n\n  await saveChecksums(calculatedChecksums);\n}\n\n/**\n * 두 파일의 내용이 같은지 체크섬으로 비교합니다.\n * 만약 파일이 둘 중 하나라도 없다면 비교 불가로 false 반환합니다.\n * @param one 파일 경로\n * @param two 파일 경로\n * @returns boolean\n */\nexport async function areFilesSame(\n  one: PathLike,\n  two: PathLike\n): Promise<boolean> {\n  if (!(await exists(one)) || !(await exists(two))) {\n    return false;\n  }\n\n  const oneChecksum = await getChecksumOfFile(one);\n  const twoChecksum = await getChecksumOfFile(two);\n\n  return oneChecksum === twoChecksum;\n}\n\nasync function getCurrentChecksums(): Promise<PathAndChecksum[]> {\n  const filePaths = (\n    await Promise.all(\n      Object.entries(getChecksumPatternGroupInAbsolutePath()).map(\n        async ([_fileType, pattern]) => {\n          return globAsync(pattern) as Promise<AbsolutePath[]>;\n        }\n      )\n    )\n  )\n    .flat()\n    .sort();\n\n  const fileChecksums = await Promise.all(\n    filePaths.map(async (filePath) => {\n      return {\n        path: filePath,\n        checksum: await getChecksumOfFile(filePath),\n      };\n    })\n  );\n\n  return fileChecksums;\n}\n\nasync function getPreviousChecksums(): Promise<PathAndChecksum[]> {\n  const checksumFilePath = getChecksumFilePath();\n  if (!(await exists(checksumFilePath))) {\n    return [];\n  }\n\n  const previousChecksums = JSON.parse(\n    await readFile(checksumFilePath, \"utf-8\")\n  ).map((r: { path: ApiRelativePath; checksum: string }) => ({\n    path: path.join(Sonamu.apiRootPath, r.path), // 체크섬 파일에서 읽을 때: API 상대 경로 → 절대 경로\n    checksum: r.checksum,\n  })) as PathAndChecksum[];\n  return previousChecksums;\n}\n\nfunction getChecksumFilePath(): AbsolutePath {\n  return path.join(Sonamu.apiRootPath, \"sonamu.lock\") as AbsolutePath;\n}\n\nasync function saveChecksums(checksums: PathAndChecksum[]): Promise<void> {\n  const checksumFilePath = getChecksumFilePath();\n  await writeFile(\n    checksumFilePath,\n    JSON.stringify(\n      checksums.map((r) => ({\n        path: path.relative(Sonamu.apiRootPath, r.path), // 체크섬 파일에 저장할 때: 절대 경로 → API 상대 경로\n        checksum: r.checksum,\n      })),\n      null,\n      2\n    ),\n    \"utf-8\"\n  );\n  console.log(\"checksum saved\", checksumFilePath);\n}\n\nasync function getChecksumOfFile(filePath: PathLike): Promise<string> {\n  return new Promise<string>((resolve, reject) => {\n    const hash = crypto.createHash(\"sha1\");\n    const input = createReadStream(filePath);\n    input.on(\"error\", reject);\n    input.on(\"data\", function (chunk: any) {\n      hash.update(chunk);\n    });\n    input.on(\"close\", function () {\n      resolve(hash.digest(\"hex\"));\n    });\n  });\n}\n"],"names":["path","globAsync","createReadStream","readFile","writeFile","exists","crypto","equal","_","Sonamu","getChecksumPatternGroupInAbsolutePath","findChangedFilesUsingChecksums","calculatedChecksums","getCurrentChecksums","savedChecksums","getPreviousChecksums","isSame","diff","differenceWith","isEqual","map","r","renewChecksums","saveChecksums","areFilesSame","one","two","oneChecksum","getChecksumOfFile","twoChecksum","filePaths","Promise","all","Object","entries","_fileType","pattern","flat","sort","fileChecksums","filePath","checksum","checksumFilePath","getChecksumFilePath","previousChecksums","JSON","parse","join","apiRootPath","checksums","stringify","relative","console","log","resolve","reject","hash","createHash","input","on","chunk","update","digest"],"mappings":"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"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { GenerateOptions, PathAndCode, TemplateKey, TemplateOptions } from "../types/types";
|
|
2
|
+
import { AbsolutePath } from "../utils/path-utils";
|
|
3
|
+
/**
|
|
4
|
+
* 템플릿을 렌더링하고 파일로 생성합니다.
|
|
5
|
+
* overwrite 옵션이 false인 경우, 이미 존재하는 파일은 건너뜁니다.
|
|
6
|
+
* @param key - 템플릿 키 (예: "entity", "model", "service" 등)
|
|
7
|
+
* @param templateOptions - 템플릿 렌더링에 필요한 옵션
|
|
8
|
+
* @param _generateOptions - 생성 옵션 (overwrite 여부)
|
|
9
|
+
* @returns 생성된 파일 경로 배열
|
|
10
|
+
*/
|
|
11
|
+
export declare function generateTemplate(key: TemplateKey, templateOptions: any, _generateOptions?: GenerateOptions): Promise<AbsolutePath[]>;
|
|
12
|
+
/**
|
|
13
|
+
* 템플릿을 렌더링하여 PathAndCode 객체를 반환합니다.
|
|
14
|
+
* 파일로 쓰지 않고 메모리상에서만 렌더링합니다.
|
|
15
|
+
* @param key - 템플릿 키
|
|
16
|
+
* @param options - 템플릿 렌더링 옵션
|
|
17
|
+
* @returns 경로와 코드 쌍의 배열
|
|
18
|
+
*/
|
|
19
|
+
export declare function renderTemplate<T extends keyof TemplateOptions>(key: T, options: TemplateOptions[T]): Promise<PathAndCode[]>;
|
|
20
|
+
//# sourceMappingURL=code-generator.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"code-generator.d.ts","sourceRoot":"","sources":["../../src/syncer/code-generator.ts"],"names":[],"mappings":"AAGA,OAAO,EACL,eAAe,EACf,WAAW,EACX,WAAW,EACX,eAAe,EAChB,MAAM,gBAAgB,CAAC;AAWxB,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAEnD;;;;;;;GAOG;AACH,wBAAsB,gBAAgB,CACpC,GAAG,EAAE,WAAW,EAChB,eAAe,EAAE,GAAG,EACpB,gBAAgB,CAAC,EAAE,eAAe,GACjC,OAAO,CAAC,YAAY,EAAE,CAAC,CA2CzB;AAED;;;;;;GAMG;AACH,wBAAsB,cAAc,CAAC,CAAC,SAAS,MAAM,eAAe,EAClE,GAAG,EAAE,CAAC,EACN,OAAO,EAAE,eAAe,CAAC,CAAC,CAAC,GAC1B,OAAO,CAAC,WAAW,EAAE,CAAC,CAkBxB"}
|