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/api/sonamu.js
CHANGED
|
@@ -1,2 +1,482 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,"__esModule",{value:true});Object.defineProperty(exports,"Sonamu",{enumerable:true,get:function(){return Sonamu}});var _async_hooks=require("async_hooks");var _chalk=/*#__PURE__*/_interop_require_default(require("chalk"));var _fastify=/*#__PURE__*/_interop_require_default(require("fastify"));var _promises=require("fs/promises");var _path=/*#__PURE__*/_interop_require_default(require("path"));var _fsutils=require("../utils/fs-utils");var _datefnstz=require("date-fns-tz");var _zod=require("zod");var _db=require("../database/db");var _knexonduplicateupdate=require("../database/knex-plugins/knex-on-duplicate-update");var _soexceptions=require("../exceptions/so-exceptions");var _sse=require("../stream/sse");var _types=require("../types/types");var _controller=require("../utils/controller");var _utils=require("../utils/utils");var _zoderror=require("../utils/zod-error");var _caster=require("./caster");var _codeconverters=require("./code-converters");var _passport=/*#__PURE__*/_interop_require_default(require("@fastify/passport"));function _array_like_to_array(arr,len){if(len==null||len>arr.length)len=arr.length;for(var i=0,arr2=new Array(len);i<len;i++)arr2[i]=arr[i];return arr2}function _array_with_holes(arr){if(Array.isArray(arr))return arr}function asyncGeneratorStep(gen,resolve,reject,_next,_throw,key,arg){try{var info=gen[key](arg);var value=info.value}catch(error){reject(error);return}if(info.done){resolve(value)}else{Promise.resolve(value).then(_next,_throw)}}function _async_to_generator(fn){return function(){var self=this,args=arguments;return new Promise(function(resolve,reject){var gen=fn.apply(self,args);function _next(value){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"next",value)}function _throw(err){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"throw",err)}_next(undefined)})}}function _class_call_check(instance,Constructor){if(!(instance instanceof Constructor)){throw new TypeError("Cannot call a class as a function")}}function _defineProperties(target,props){for(var i=0;i<props.length;i++){var descriptor=props[i];descriptor.enumerable=descriptor.enumerable||false;descriptor.configurable=true;if("value"in descriptor)descriptor.writable=true;Object.defineProperty(target,descriptor.key,descriptor)}}function _create_class(Constructor,protoProps,staticProps){if(protoProps)_defineProperties(Constructor.prototype,protoProps);if(staticProps)_defineProperties(Constructor,staticProps);return Constructor}function _define_property(obj,key,value){if(key in obj){Object.defineProperty(obj,key,{value:value,enumerable:true,configurable:true,writable:true})}else{obj[key]=value}return obj}function _instanceof(left,right){if(right!=null&&typeof Symbol!=="undefined"&&right[Symbol.hasInstance]){return!!right[Symbol.hasInstance](left)}else{return left instanceof right}}function _interop_require_default(obj){return obj&&obj.__esModule?obj:{default:obj}}function _getRequireWildcardCache(nodeInterop){if(typeof WeakMap!=="function")return null;var cacheBabelInterop=new WeakMap;var cacheNodeInterop=new WeakMap;return(_getRequireWildcardCache=function(nodeInterop){return nodeInterop?cacheNodeInterop:cacheBabelInterop})(nodeInterop)}function _interop_require_wildcard(obj,nodeInterop){if(!nodeInterop&&obj&&obj.__esModule){return obj}if(obj===null||typeof obj!=="object"&&typeof obj!=="function"){return{default:obj}}var cache=_getRequireWildcardCache(nodeInterop);if(cache&&cache.has(obj)){return cache.get(obj)}var newObj={__proto__:null};var hasPropertyDescriptor=Object.defineProperty&&Object.getOwnPropertyDescriptor;for(var key in obj){if(key!=="default"&&Object.prototype.hasOwnProperty.call(obj,key)){var desc=hasPropertyDescriptor?Object.getOwnPropertyDescriptor(obj,key):null;if(desc&&(desc.get||desc.set)){Object.defineProperty(newObj,key,desc)}else{newObj[key]=obj[key]}}}newObj.default=obj;if(cache){cache.set(obj,newObj)}return newObj}function _iterable_to_array_limit(arr,i){var _i=arr==null?null:typeof Symbol!=="undefined"&&arr[Symbol.iterator]||arr["@@iterator"];if(_i==null)return;var _arr=[];var _n=true;var _d=false;var _s,_e;try{for(_i=_i.call(arr);!(_n=(_s=_i.next()).done);_n=true){_arr.push(_s.value);if(i&&_arr.length===i)break}}catch(err){_d=true;_e=err}finally{try{if(!_n&&_i["return"]!=null)_i["return"]()}finally{if(_d)throw _e}}return _arr}function _non_iterable_rest(){throw new TypeError("Invalid attempt to destructure non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}function _object_spread(target){for(var i=1;i<arguments.length;i++){var source=arguments[i]!=null?arguments[i]:{};var ownKeys=Object.keys(source);if(typeof Object.getOwnPropertySymbols==="function"){ownKeys=ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function(sym){return Object.getOwnPropertyDescriptor(source,sym).enumerable}))}ownKeys.forEach(function(key){_define_property(target,key,source[key])})}return target}function _sliced_to_array(arr,i){return _array_with_holes(arr)||_iterable_to_array_limit(arr,i)||_unsupported_iterable_to_array(arr,i)||_non_iterable_rest()}function _unsupported_iterable_to_array(o,minLen){if(!o)return;if(typeof o==="string")return _array_like_to_array(o,minLen);var n=Object.prototype.toString.call(o).slice(8,-1);if(n==="Object"&&o.constructor)n=o.constructor.name;if(n==="Map"||n==="Set")return Array.from(n);if(n==="Arguments"||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))return _array_like_to_array(o,minLen)}function _ts_generator(thisArg,body){var f,y,t,_={label:0,sent:function(){if(t[0]&1)throw t[1];return t[1]},trys:[],ops:[]},g=Object.create((typeof Iterator==="function"?Iterator:Object).prototype);return g.next=verb(0),g["throw"]=verb(1),g["return"]=verb(2),typeof Symbol==="function"&&(g[Symbol.iterator]=function(){return this}),g;function verb(n){return function(v){return step([n,v])}}function step(op){if(f)throw new TypeError("Generator is already executing.");while(g&&(g=0,op[0]&&(_=0)),_)try{if(f=1,y&&(t=op[0]&2?y["return"]:op[0]?y["throw"]||((t=y["return"])&&t.call(y),0):y.next)&&!(t=t.call(y,op[1])).done)return t;if(y=0,t)op=[op[0]&2,t.value];switch(op[0]){case 0:case 1:t=op;break;case 4:_.label++;return{value:op[1],done:false};case 5:_.label++;y=op[1];op=[0];continue;case 7:op=_.ops.pop();_.trys.pop();continue;default:if(!(t=_.trys,t=t.length>0&&t[t.length-1])&&(op[0]===6||op[0]===2)){_=0;continue}if(op[0]===3&&(!t||op[1]>t[0]&&op[1]<t[3])){_.label=op[1];break}if(op[0]===6&&_.label<t[1]){_.label=t[1];t=op;break}if(t&&_.label<t[2]){_.label=t[2];_.ops.push(op);break}if(t[2])_.ops.pop();_.trys.pop();continue}op=body.call(thisArg,_)}catch(e){op=[6,e];y=0}finally{f=t=0}if(op[0]&5)throw op[1];return{value:op[0]?op[1]:void 0,done:true}}}var SonamuClass=/*#__PURE__*/function(){"use strict";function SonamuClass(){_class_call_check(this,SonamuClass);_define_property(this,"isInitialized",false);_define_property(this,"asyncLocalStorage",new _async_hooks.AsyncLocalStorage);_define_property(this,"uploadStorage",new _async_hooks.AsyncLocalStorage);_define_property(this,"_apiRootPath",null);_define_property(this,"_dbConfig",null);_define_property(this,"_syncer",null);_define_property(this,"_config",null);_define_property(this,"_secrets",null);_define_property(this,"_storage",null);_define_property(this,"watcher",null);_define_property(this,"pendingFiles",[]);_define_property(this,"hmrStartTime",0);_define_property(this,"server",null)}_create_class(SonamuClass,[{key:"getContext",value:function getContext(){var store=this.asyncLocalStorage.getStore();if(store===null||store===void 0?void 0:store.context){return store.context}throw new Error("Sonamu cannot find context")}},{key:"getUploadContext",value:function getUploadContext(){var store=this.uploadStorage.getStore();if(store===null||store===void 0?void 0:store.uploadContext){return store.uploadContext}throw new Error("Sonamu cannot find upload context. Did you use @upload decorator?")}},{key:"apiRootPath",get:function get(){if(this._apiRootPath===null){throw new Error("Sonamu has not been initialized")}return this._apiRootPath},set:function set(apiRootPath){this._apiRootPath=apiRootPath}},{key:"appRootPath",get:function get(){return this.apiRootPath.split(_path.default.sep).slice(0,-1).join(_path.default.sep)}},{key:"dbConfig",get:function get(){if(this._dbConfig===null){throw new Error("Sonamu has not been initialized")}return this._dbConfig},set:function set(dbConfig){this._dbConfig=dbConfig}},{key:"syncer",get:function get(){if(this._syncer===null){throw new Error("Sonamu has not been initialized")}return this._syncer},set:function set(syncer){this._syncer=syncer}},{key:"config",get:function get(){if(this._config===null){throw new Error("Sonamu has not been initialized")}return this._config},set:function set(config){this._config=config}},{key:"secrets",get:function get(){return this._secrets},set:function set(secrets){this._secrets=secrets}},{key:"storage",get:function get(){return this._storage},set:function set(storage){this._storage=storage}},{key:"initForTesting",value:function initForTesting(){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,this.init(true,false,undefined,true)];case 1:_state.sent();return[2]}})}).call(this)}},{key:"init",value:function init(){var doSilent=arguments.length>0&&arguments[0]!==void 0?arguments[0]:false,enableSync=arguments.length>1&&arguments[1]!==void 0?arguments[1]:true,apiRootPath=arguments.length>2?arguments[2]:void 0,forTesting=arguments.length>3&&arguments[3]!==void 0?arguments[3]:false;return _async_to_generator(function(){var configPath,secretsPath,_,_1,_2,_3,_4,EntityManager,Syncer;return _ts_generator(this,function(_state){switch(_state.label){case 0:if(this.isInitialized){return[2]}!doSilent&&console.time(_chalk.default.cyan("Sonamu.init".concat(forTesting?" for testing":"")));this.apiRootPath=apiRootPath!==null&&apiRootPath!==void 0?apiRootPath:(0,_utils.findApiRootPath)();configPath=_path.default.join(this.apiRootPath,"sonamu.config.json");secretsPath=_path.default.join(this.apiRootPath,"sonamu.secrets.json");return[4,(0,_fsutils.exists)(configPath)];case 1:if(!_state.sent()){throw new Error("Cannot find sonamu.config.json in ".concat(configPath))}_=this;_1=JSON.parse;return[4,(0,_promises.readFile)(configPath)];case 2:_.config=_1.apply(JSON,[_state.sent().toString()]);return[4,(0,_fsutils.exists)(secretsPath)];case 3:if(!_state.sent())return[3,5];_2=this;_3=JSON.parse;return[4,(0,_promises.readFile)(secretsPath)];case 4:_2.secrets=_3.apply(JSON,[_state.sent().toString()]);_state.label=5;case 5:_4=this;return[4,_db.DB.readKnexfile()];case 6:_4.dbConfig=_state.sent();!doSilent&&console.log(_chalk.default.green("DB Config Loaded!"));(0,_knexonduplicateupdate.attachOnDuplicateUpdate)();if(forTesting){this.isInitialized=true;return[2]}return[4,Promise.resolve().then(function(){return /*#__PURE__*/_interop_require_wildcard(require("../entity/entity-manager"))})];case 7:EntityManager=_state.sent().EntityManager;return[4,EntityManager.autoload(doSilent)];case 8:_state.sent();return[4,Promise.resolve().then(function(){return /*#__PURE__*/_interop_require_wildcard(require("../syncer/syncer"))})];case 9:Syncer=_state.sent().Syncer;this.syncer=new Syncer;return[4,this.syncer.autoloadModels()];case 10:_state.sent();return[4,this.syncer.autoloadTypes()];case 11:_state.sent();return[4,this.syncer.autoloadApis()];case 12:_state.sent();if(!((0,_controller.isLocal)()&&!(0,_controller.isTest)()&&enableSync))return[3,14];return[4,this.syncer.sync()];case 13:_state.sent();this.startWatcher();this.syncer.syncUI();_state.label=14;case 14:this.isInitialized=true;!doSilent&&console.timeEnd(_chalk.default.cyan("Sonamu.init"));return[2]}})}).call(this)}},{key:"createServer",value:function createServer(options,initOptions){return _async_to_generator(function(){var server,_options_plugins;return _ts_generator(this,function(_state){switch(_state.label){case 0:server=(0,_fastify.default)(options.fastify);this.server=server;if(options.storage){this.storage=options.storage}if(options.plugins){this.registerPlugins(server,options.plugins)}if(options.auth){;if(!((_options_plugins=options.plugins)===null||_options_plugins===void 0?void 0:_options_plugins.session)){throw new Error("Auth requires session plugin. Please add plugins.session configuration.")}this.registerAuth(server,options.auth)}return[4,this.withFastify(server,options.apiConfig,{enableSync:initOptions===null||initOptions===void 0?void 0:initOptions.enableSync,doSilent:initOptions===null||initOptions===void 0?void 0:initOptions.doSilent})];case 1:_state.sent();return[4,this.boot(server,options)];case 2:_state.sent();return[2,server]}})}).call(this)}},{key:"withFastify",value:function withFastify(server,config,options){return _async_to_generator(function(){var _this,timezone,DATE_FORMAT,ISO_DATE_REGEX;return _ts_generator(this,function(_state){switch(_state.label){case 0:_this=this;if(!(this.isInitialized===false))return[3,2];return[4,this.init(options===null||options===void 0?void 0:options.doSilent,options===null||options===void 0?void 0:options.enableSync)];case 1:_state.sent();_state.label=2;case 2:this.server=server;timezone=this.config.timezone;if(timezone){DATE_FORMAT="yyyy-MM-dd'T'HH:mm:ssXXX";ISO_DATE_REGEX=/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(\.\d{3})?Z$/;server.setReplySerializer(function(payload){return JSON.stringify(payload,function(_key,value){if(typeof value==="string"&&ISO_DATE_REGEX.test(value)){return(0,_datefnstz.formatInTimeZone)(new Date(value),timezone,DATE_FORMAT)}return value})});!(options===null||options===void 0?void 0:options.doSilent)&&console.log(_chalk.default.green("Timezone set to ".concat(timezone)))}server.get("".concat(this.config.route.prefix,"/routes"),function(_request,_reply){return _async_to_generator(function(){return _ts_generator(this,function(_state){return[2,this.syncer.apis]})}).call(_this)});server.get("".concat(this.config.route.prefix,"/healthcheck"),function(_request,_reply){return _async_to_generator(function(){return _ts_generator(this,function(_state){return[2,"ok"]})})()});if((0,_controller.isLocal)()){server.all("*",function(request,reply){var found=_this.syncer.apis.find(function(api){var _api_options_httpMethod;return _this.config.route.prefix+api.path===request.url.split("?")[0]&&((_api_options_httpMethod=api.options.httpMethod)!==null&&_api_options_httpMethod!==void 0?_api_options_httpMethod:"GET")===request.method.toUpperCase()});if(found){return _this.getApiHandler(found,config)(request,reply)}throw new _soexceptions.NotFoundException("존재하지 않는 API 접근입니다.")})}else{this.syncer.apis.map(function(api){if(_this.syncer.models[api.modelName]===undefined){throw new Error("정의되지 않은 모델에 접근 ".concat(api.modelName))}server.route({method:api.options.httpMethod,url:_this.config.route.prefix+api.path,handler:_this.getApiHandler(api,config)})})}return[2]}})}).call(this)}},{key:"getApiHandler",value:function getApiHandler(api,config){var _this=this;return function(request,reply){return _async_to_generator(function(){var _api_options_guards,ReqType,which,reqBody,_request_which,messages,_api_options_contentType,_ref,cacheKey,cacheTtl,cachedData,createSSE,_request_user,context,_tmp,model;return _ts_generator(this,function(_state){switch(_state.label){case 0:((_api_options_guards=api.options.guards)!==null&&_api_options_guards!==void 0?_api_options_guards:[]).every(function(guard){return config.guardHandler(guard,request,api)});ReqType=(0,_codeconverters.getZodObjectFromApi)(api,this.syncer.types);which=api.options.httpMethod==="GET"?"query":"body";try{;reqBody=(0,_caster.fastifyCaster)(ReqType).parse((_request_which=request[which])!==null&&_request_which!==void 0?_request_which:{})}catch(e){if(_instanceof(e,_zod.ZodError)){messages=(0,_zoderror.humanizeZodError)(e).map(function(issue){return issue.message}).join(" ");throw new _soexceptions.BadRequestException(messages,{zodError:e})}else{throw e}}reply.type((_api_options_contentType=api.options.contentType)!==null&&_api_options_contentType!==void 0?_api_options_contentType:"application/json");return[4,function(){return _async_to_generator(function(){var cacheKeyRes,cacheKey,cacheTtl,cachedData,e;return _ts_generator(this,function(_state){switch(_state.label){case 0:if(!config.cache)return[3,5];_state.label=1;case 1:_state.trys.push([1,3,,4]);cacheKeyRes=config.cache.resolveKey(api.path,reqBody);if(cacheKeyRes.cache===false){return[2,{cacheKey:null,cachedData:null}]}cacheKey=cacheKeyRes.key;cacheTtl=cacheKeyRes.ttl;return[4,config.cache.get(cacheKey)];case 2:cachedData=_state.sent();return[2,{cacheKey:cacheKey,cacheTtl:cacheTtl,cachedData:cachedData}];case 3:e=_state.sent();console.error(e);return[3,4];case 4:return[2,{cacheKey:null,cachedData:null}];case 5:return[2,{cacheKey:null,cachedData:null}]}})})()}()];case 1:_ref=_state.sent(),cacheKey=_ref.cacheKey,cacheTtl=_ref.cacheTtl,cachedData=_ref.cachedData;if(cachedData!==null){return[2,cachedData]}createSSE=(function(_request,_reply,_events){return(0,_sse.createSSEFactory)(_request.socket,_reply,_events)}).bind(null,request,reply);_tmp=[{}];return[4,Promise.resolve(config.contextProvider({request:request,reply:reply,headers:request.headers,createSSE:createSSE,user:(_request_user=request.user)!==null&&_request_user!==void 0?_request_user:null,passport:{login:request.login.bind(request),logout:request.logout.bind(request)}},request,reply))];case 2:context=_object_spread.apply(void 0,_tmp.concat([_state.sent()]));model=this.syncer.models[api.modelName];return[2,this.asyncLocalStorage.run({context:context},function(){return _async_to_generator(function(){var result,_api_options_contentType;return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,model[api.methodName].apply(model,api.parameters.map(function(param){if(_types.ApiParamType.isContext(param.type)){return context}else{return reqBody[param.name]}}))];case 1:result=_state.sent();reply.type((_api_options_contentType=api.options.contentType)!==null&&_api_options_contentType!==void 0?_api_options_contentType:"application/json");if(!(config.cache&&cacheKey))return[3,3];return[4,config.cache.put(cacheKey,result,cacheTtl)];case 2:_state.sent();_state.label=3;case 3:return[2,result]}})})()})]}})}).call(_this)}}},{key:"startWatcher",value:function startWatcher(){var _this=this;var watchPath=_path.default.join(this.apiRootPath,"src");var chokidar=require("chokidar");this.watcher=chokidar.watch(watchPath,{ignored:function(path,stats){return!!(stats===null||stats===void 0?void 0:stats.isFile())&&!path.endsWith(".ts")&&!path.endsWith(".json")||path.endsWith("src/index.ts")},persistent:true,ignoreInitial:true});this.watcher.on("all",function(event,filePath){return _async_to_generator(function(){var e;return _ts_generator(this,function(_state){switch(_state.label){case 0:if(event!=="change"&&event!=="add"){return[2]}_state.label=1;case 1:_state.trys.push([1,3,,4]);return[4,this.handleFileChange(event,filePath)];case 2:_state.sent();return[3,4];case 3:e=_state.sent();console.error(e);return[3,4];case 4:return[2]}})}).call(_this)})}},{key:"runScript",value:function runScript(fn){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,this.init(true,false,undefined,false)];case 1:_state.sent();_state.label=2;case 2:_state.trys.push([2,,4,6]);return[4,fn()];case 3:_state.sent();return[3,6];case 4:return[4,this.destroy()];case 5:_state.sent();return[7];case 6:return[2]}})}).call(this)}},{key:"registerPlugins",value:function registerPlugins(server,plugins){if(!plugins){return}var pluginsModules={cors:"@fastify/cors",formbody:"@fastify/formbody",multipart:"@fastify/multipart",qs:"fastify-qs",sse:"fastify-sse-v2",static:"@fastify/static",session:"@fastify/secure-session"};var registerPlugin=function(key,pluginName){var option=plugins[key];if(!option)return;if(option===true){server.register(Promise.resolve(pluginName).then(function(p){return /*#__PURE__*/_interop_require_wildcard(require(p))}))}else{server.register(Promise.resolve(pluginName).then(function(p){return /*#__PURE__*/_interop_require_wildcard(require(p))}),option)}};Object.entries(pluginsModules).forEach(function(param){var _param=_sliced_to_array(param,2),key=_param[0],pluginName=_param[1];registerPlugin(key,pluginName)});if(plugins.custom){plugins.custom(server)}}},{key:"registerAuth",value:function registerAuth(server,options){return _async_to_generator(function(){return _ts_generator(this,function(_state){server.register(_passport.default.initialize());server.register(_passport.default.secureSession());if(typeof options==="boolean"){_passport.default.registerUserSerializer(function(user,_request){return _async_to_generator(function(){return _ts_generator(this,function(_state){return[2,user]})})()});_passport.default.registerUserDeserializer(function(serialized,_request){return _async_to_generator(function(){return _ts_generator(this,function(_state){return[2,serialized]})})()})}else{_passport.default.registerUserSerializer(options.userSerializer);_passport.default.registerUserDeserializer(options.userDeserializer)}return[2]})})()}},{key:"boot",value:function boot(server,options){return _async_to_generator(function(){var _this,_options_listen,_options_listen1,_options_lifecycle,_options_listen_port,port,_options_listen_host,host,shutdown,_options_lifecycle1;return _ts_generator(this,function(_state){_this=this;port=(_options_listen_port=(_options_listen=options.listen)===null||_options_listen===void 0?void 0:_options_listen.port)!==null&&_options_listen_port!==void 0?_options_listen_port:3e3;host=(_options_listen_host=(_options_listen1=options.listen)===null||_options_listen1===void 0?void 0:_options_listen1.host)!==null&&_options_listen_host!==void 0?_options_listen_host:"localhost";server.addHook("onClose",function(){return _async_to_generator(function(){var _options_lifecycle_onShutdown,_options_lifecycle;return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,(_options_lifecycle=options.lifecycle)===null||_options_lifecycle===void 0?void 0:(_options_lifecycle_onShutdown=_options_lifecycle.onShutdown)===null||_options_lifecycle_onShutdown===void 0?void 0:_options_lifecycle_onShutdown.call(_options_lifecycle,server)];case 1:_state.sent();return[4,this.destroy()];case 2:_state.sent();return[2]}})}).call(_this)});shutdown=function(){return _async_to_generator(function(){var err;return _ts_generator(this,function(_state){switch(_state.label){case 0:_state.trys.push([0,2,,3]);return[4,server.close()];case 1:_state.sent();process.exit(0);return[3,3];case 2:err=_state.sent();console.error("Error during shutdown:",err);process.exit(1);return[3,3];case 3:return[2]}})})()};process.on("SIGINT",shutdown);process.on("SIGTERM",shutdown);if((_options_lifecycle=options.lifecycle)===null||_options_lifecycle===void 0?void 0:_options_lifecycle.onError){;server.setErrorHandler((_options_lifecycle1=options.lifecycle)===null||_options_lifecycle1===void 0?void 0:_options_lifecycle1.onError)}server.listen({port:port,host:host}).then(function(){return _async_to_generator(function(){var _options_lifecycle_onStart,_options_lifecycle;return _ts_generator(this,function(_state){switch(_state.label){case 0:return[4,(_options_lifecycle=options.lifecycle)===null||_options_lifecycle===void 0?void 0:(_options_lifecycle_onStart=_options_lifecycle.onStart)===null||_options_lifecycle_onStart===void 0?void 0:_options_lifecycle_onStart.call(_options_lifecycle,server)];case 1:_state.sent();return[2]}})})()}).catch(function(err){return _async_to_generator(function(){return _ts_generator(this,function(_state){switch(_state.label){case 0:console.error(_chalk.default.red("Failed to start server:",err));return[4,shutdown()];case 1:_state.sent();return[2]}})})()});return[2]})}).call(this)}},{key:"handleFileChange",value:function handleFileChange(event,filePath){return _async_to_generator(function(){var relativePath;return _ts_generator(this,function(_state){switch(_state.label){case 0:if(this.pendingFiles.length===0){this.hmrStartTime=Date.now()}this.pendingFiles.push(filePath);relativePath=filePath.replace(this.apiRootPath,"api");console.log(_chalk.default.bold("Detected(".concat(event,"): ").concat(_chalk.default.blue(relativePath))));return[4,this.syncer.syncFromWatcher([filePath])];case 1:_state.sent();this.pendingFiles=this.pendingFiles.slice(1);if(!(this.pendingFiles.length===0))return[3,3];return[4,this.finishHMR()];case 2:_state.sent();_state.label=3;case 3:return[2]}})}).call(this)}},{key:"finishHMR",value:function finishHMR(){return _async_to_generator(function(){var _,_1,endTime,totalTime,msg,margin;return _ts_generator(this,function(_state){switch(_state.label){case 0:_1=(_=this.syncer).saveChecksums;return[4,this.syncer.getCurrentChecksums()];case 1:return[4,_1.apply(_,[_state.sent()])];case 2:_state.sent();endTime=Date.now();totalTime=endTime-this.hmrStartTime;msg="HMR Done! ".concat(_chalk.default.bold.white("".concat(totalTime,"ms")));margin=Math.max(0,(process.stdout.columns-msg.length)/2);console.log(_chalk.default.black.bgGreen(" ".repeat(margin)+msg+" ".repeat(margin)));return[2]}})}).call(this)}},{key:"destroy",value:function destroy(){return _async_to_generator(function(){var _this_watcher,_this_storage,BaseModel;return _ts_generator(this,function(_state){switch(_state.label){case 0:BaseModel=require("../database/base-model").BaseModel;return[4,BaseModel.destroy()];case 1:_state.sent();return[4,(_this_watcher=this.watcher)===null||_this_watcher===void 0?void 0:_this_watcher.close()];case 2:_state.sent();(_this_storage=this.storage)===null||_this_storage===void 0?void 0:_this_storage.destroy();return[2]}})}).call(this)}}]);return SonamuClass}();var Sonamu=new SonamuClass;
|
|
2
|
-
|
|
1
|
+
import { AsyncLocalStorage } from "async_hooks";
|
|
2
|
+
import chalk from "chalk";
|
|
3
|
+
import fastify from "fastify";
|
|
4
|
+
import { readFile } from "node:fs/promises";
|
|
5
|
+
import path from "path";
|
|
6
|
+
import { exists } from "../utils/fs-utils.js";
|
|
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";
|
|
27
|
+
class SonamuClass {
|
|
28
|
+
isInitialized = false;
|
|
29
|
+
asyncLocalStorage = new AsyncLocalStorage();
|
|
30
|
+
uploadStorage = new AsyncLocalStorage();
|
|
31
|
+
getContext() {
|
|
32
|
+
const store = this.asyncLocalStorage.getStore();
|
|
33
|
+
if (store?.context) {
|
|
34
|
+
return store.context;
|
|
35
|
+
}
|
|
36
|
+
if (process.env.NODE_ENV === "test") {
|
|
37
|
+
// 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴
|
|
38
|
+
return {
|
|
39
|
+
request: null,
|
|
40
|
+
reply: null,
|
|
41
|
+
headers: {},
|
|
42
|
+
createSSE: ()=>{},
|
|
43
|
+
naiteStore: new Map()
|
|
44
|
+
};
|
|
45
|
+
} else {
|
|
46
|
+
throw new Error("Sonamu cannot find context");
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
getUploadContext() {
|
|
50
|
+
const store = this.uploadStorage.getStore();
|
|
51
|
+
if (store?.uploadContext) {
|
|
52
|
+
return store.uploadContext;
|
|
53
|
+
}
|
|
54
|
+
throw new Error("Sonamu cannot find upload context. Did you use @upload decorator?");
|
|
55
|
+
}
|
|
56
|
+
_apiRootPath = null;
|
|
57
|
+
set apiRootPath(apiRootPath) {
|
|
58
|
+
this._apiRootPath = apiRootPath;
|
|
59
|
+
}
|
|
60
|
+
get apiRootPath() {
|
|
61
|
+
if (this._apiRootPath === null) {
|
|
62
|
+
throw new Error("Sonamu has not been initialized");
|
|
63
|
+
}
|
|
64
|
+
return this._apiRootPath;
|
|
65
|
+
}
|
|
66
|
+
get appRootPath() {
|
|
67
|
+
return this.apiRootPath.split(path.sep).slice(0, -1).join(path.sep);
|
|
68
|
+
}
|
|
69
|
+
_dbConfig = null;
|
|
70
|
+
set dbConfig(dbConfig) {
|
|
71
|
+
this._dbConfig = dbConfig;
|
|
72
|
+
}
|
|
73
|
+
get dbConfig() {
|
|
74
|
+
if (this._dbConfig === null) {
|
|
75
|
+
throw new Error("Sonamu has not been initialized");
|
|
76
|
+
}
|
|
77
|
+
return this._dbConfig;
|
|
78
|
+
}
|
|
79
|
+
_syncer = null;
|
|
80
|
+
set syncer(syncer) {
|
|
81
|
+
this._syncer = syncer;
|
|
82
|
+
}
|
|
83
|
+
get syncer() {
|
|
84
|
+
if (this._syncer === null) {
|
|
85
|
+
throw new Error("Sonamu has not been initialized");
|
|
86
|
+
}
|
|
87
|
+
return this._syncer;
|
|
88
|
+
}
|
|
89
|
+
_config = null;
|
|
90
|
+
set config(config) {
|
|
91
|
+
this._config = config;
|
|
92
|
+
}
|
|
93
|
+
get config() {
|
|
94
|
+
if (this._config === null) {
|
|
95
|
+
throw new Error("Sonamu has not been initialized");
|
|
96
|
+
}
|
|
97
|
+
return this._config;
|
|
98
|
+
}
|
|
99
|
+
_secrets = null;
|
|
100
|
+
set secrets(secrets) {
|
|
101
|
+
this._secrets = secrets;
|
|
102
|
+
}
|
|
103
|
+
get secrets() {
|
|
104
|
+
return this._secrets;
|
|
105
|
+
}
|
|
106
|
+
_storage = null;
|
|
107
|
+
set storage(storage) {
|
|
108
|
+
this._storage = storage;
|
|
109
|
+
}
|
|
110
|
+
get storage() {
|
|
111
|
+
return this._storage;
|
|
112
|
+
}
|
|
113
|
+
// HMR 처리
|
|
114
|
+
watcher = null;
|
|
115
|
+
pendingFiles = [];
|
|
116
|
+
hmrStartTime = 0;
|
|
117
|
+
server = null;
|
|
118
|
+
async initForTesting() {
|
|
119
|
+
await this.init(true, false, undefined, true);
|
|
120
|
+
}
|
|
121
|
+
async init(doSilent = false, enableSync = true, apiRootPath, forTesting = false) {
|
|
122
|
+
if (this.isInitialized) {
|
|
123
|
+
return;
|
|
124
|
+
}
|
|
125
|
+
!doSilent && console.time(chalk.cyan(`Sonamu.init${forTesting ? " for testing" : ""}`));
|
|
126
|
+
// API 루트 패스
|
|
127
|
+
this.apiRootPath = apiRootPath ?? findApiRootPath();
|
|
128
|
+
this.config = await loadConfig(this.apiRootPath);
|
|
129
|
+
const secretsPath = path.join(this.apiRootPath, "sonamu.secrets.json");
|
|
130
|
+
if (await exists(secretsPath)) {
|
|
131
|
+
this.secrets = JSON.parse((await readFile(secretsPath)).toString());
|
|
132
|
+
}
|
|
133
|
+
// DB 로드
|
|
134
|
+
this.dbConfig = DB.generateDBConfig(this.config.database);
|
|
135
|
+
!doSilent && console.log(chalk.green("DB Config Loaded!"));
|
|
136
|
+
attachOnDuplicateUpdate();
|
|
137
|
+
// 테스팅인 경우 엔티티 로드 & 싱크 없이 중단
|
|
138
|
+
if (forTesting) {
|
|
139
|
+
this.isInitialized = true;
|
|
140
|
+
return;
|
|
141
|
+
}
|
|
142
|
+
// Entity 로드
|
|
143
|
+
const { EntityManager } = await import("../entity/entity-manager.js");
|
|
144
|
+
await EntityManager.autoload(doSilent);
|
|
145
|
+
// Syncer
|
|
146
|
+
const { Syncer } = await import("../syncer/syncer.js");
|
|
147
|
+
this.syncer = new Syncer();
|
|
148
|
+
// Autoload: Models / Types / APIs
|
|
149
|
+
await this.syncer.autoloadTypes();
|
|
150
|
+
await this.syncer.autoloadModels();
|
|
151
|
+
await this.syncer.autoloadApis();
|
|
152
|
+
await Template.autoload();
|
|
153
|
+
if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {
|
|
154
|
+
await this.syncer.sync();
|
|
155
|
+
this.startWatcher();
|
|
156
|
+
this.syncer.syncUI();
|
|
157
|
+
}
|
|
158
|
+
this.isInitialized = true;
|
|
159
|
+
!doSilent && console.timeEnd(chalk.cyan("Sonamu.init"));
|
|
160
|
+
}
|
|
161
|
+
async createServer(initOptions) {
|
|
162
|
+
if (this.isInitialized === false) {
|
|
163
|
+
await this.init(initOptions?.doSilent, initOptions?.enableSync);
|
|
164
|
+
}
|
|
165
|
+
const options = this.config.server;
|
|
166
|
+
const server = fastify(options.fastify);
|
|
167
|
+
this.server = server;
|
|
168
|
+
// Storage 설정 저장
|
|
169
|
+
if (options.storage) {
|
|
170
|
+
this.storage = options.storage;
|
|
171
|
+
}
|
|
172
|
+
// 플러그인 등록
|
|
173
|
+
if (options.plugins) {
|
|
174
|
+
this.registerPlugins(server, options.plugins);
|
|
175
|
+
}
|
|
176
|
+
if (options.auth) {
|
|
177
|
+
if (!options.plugins?.session) {
|
|
178
|
+
throw new Error("Auth requires session plugin. Please add plugins.session configuration.");
|
|
179
|
+
}
|
|
180
|
+
this.registerAuth(server, options.auth);
|
|
181
|
+
}
|
|
182
|
+
// API 라우팅 설정
|
|
183
|
+
await this.withFastify(server, options.apiConfig, {
|
|
184
|
+
enableSync: initOptions?.enableSync,
|
|
185
|
+
doSilent: initOptions?.doSilent
|
|
186
|
+
});
|
|
187
|
+
// 서버 시작
|
|
188
|
+
await this.boot(server, options);
|
|
189
|
+
return server;
|
|
190
|
+
}
|
|
191
|
+
async withFastify(server, config, options) {
|
|
192
|
+
if (this.isInitialized === false) {
|
|
193
|
+
await this.init(options?.doSilent, options?.enableSync);
|
|
194
|
+
}
|
|
195
|
+
this.server = server;
|
|
196
|
+
// timezone 설정
|
|
197
|
+
const timezone = this.config.api.timezone;
|
|
198
|
+
if (timezone) {
|
|
199
|
+
const DATE_FORMAT = "yyyy-MM-dd'T'HH:mm:ssXXX";
|
|
200
|
+
// ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)
|
|
201
|
+
const ISO_DATE_REGEX = /^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(\.\d{3})?Z$/;
|
|
202
|
+
server.setReplySerializer((payload)=>{
|
|
203
|
+
return JSON.stringify(payload, (_key, value)=>{
|
|
204
|
+
if (typeof value === "string" && ISO_DATE_REGEX.test(value)) {
|
|
205
|
+
return formatInTimeZone(new Date(value), timezone, DATE_FORMAT);
|
|
206
|
+
}
|
|
207
|
+
return value;
|
|
208
|
+
});
|
|
209
|
+
});
|
|
210
|
+
!options?.doSilent && console.log(chalk.green(`Timezone set to ${timezone}`));
|
|
211
|
+
}
|
|
212
|
+
// 전체 라우팅 리스트
|
|
213
|
+
server.get(`${this.config.api.route.prefix}/routes`, async (_request, _reply)=>{
|
|
214
|
+
return this.syncer.apis;
|
|
215
|
+
});
|
|
216
|
+
// Healthcheck API
|
|
217
|
+
server.get(`${this.config.api.route.prefix}/healthcheck`, async (_request, _reply)=>{
|
|
218
|
+
return "ok";
|
|
219
|
+
});
|
|
220
|
+
// API 라우팅 (로컬HMR 상태와 구분)
|
|
221
|
+
if (isLocal()) {
|
|
222
|
+
server.all("*", (request, reply)=>{
|
|
223
|
+
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
|
+
if (found) {
|
|
225
|
+
return this.getApiHandler(found, config)(request, reply);
|
|
226
|
+
}
|
|
227
|
+
throw new NotFoundException("존재하지 않는 API 접근입니다.");
|
|
228
|
+
});
|
|
229
|
+
} else {
|
|
230
|
+
this.syncer.apis.map((api)=>{
|
|
231
|
+
// model
|
|
232
|
+
if (this.syncer.models[api.modelName] === undefined) {
|
|
233
|
+
throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);
|
|
234
|
+
}
|
|
235
|
+
// route
|
|
236
|
+
server.route({
|
|
237
|
+
method: api.options.httpMethod,
|
|
238
|
+
url: this.config.api.route.prefix + api.path,
|
|
239
|
+
handler: this.getApiHandler(api, config)
|
|
240
|
+
}); // END server.route
|
|
241
|
+
});
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
getApiHandler(api, config) {
|
|
245
|
+
return async (request, reply)=>{
|
|
246
|
+
(api.options.guards ?? []).every((guard)=>config.guardHandler(guard, request, api));
|
|
247
|
+
// 파라미터 정보로 zod 스키마 빌드
|
|
248
|
+
const ReqType = getZodObjectFromApi(api, this.syncer.types);
|
|
249
|
+
// request 파싱
|
|
250
|
+
const which = api.options.httpMethod === "GET" ? "query" : "body";
|
|
251
|
+
let reqBody;
|
|
252
|
+
try {
|
|
253
|
+
reqBody = fastifyCaster(ReqType).parse(request[which] ?? {});
|
|
254
|
+
} catch (e) {
|
|
255
|
+
if (e instanceof ZodError) {
|
|
256
|
+
const messages = humanizeZodError(e).map((issue)=>issue.message).join(" ");
|
|
257
|
+
throw new BadRequestException(messages, {
|
|
258
|
+
zodError: e
|
|
259
|
+
});
|
|
260
|
+
} else {
|
|
261
|
+
throw e;
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
// Content-Type
|
|
265
|
+
reply.type(api.options.contentType ?? "application/json");
|
|
266
|
+
// 캐시
|
|
267
|
+
const { cacheKey, cacheTtl, cachedData } = await (async ()=>{
|
|
268
|
+
if (config.cache) {
|
|
269
|
+
try {
|
|
270
|
+
const cacheKeyRes = config.cache.resolveKey(api.path, reqBody);
|
|
271
|
+
if (cacheKeyRes.cache === false) {
|
|
272
|
+
return {
|
|
273
|
+
cacheKey: null,
|
|
274
|
+
cachedData: null
|
|
275
|
+
};
|
|
276
|
+
}
|
|
277
|
+
const cacheKey = cacheKeyRes.key;
|
|
278
|
+
const cacheTtl = cacheKeyRes.ttl;
|
|
279
|
+
const cachedData = await config.cache.get(cacheKey);
|
|
280
|
+
return {
|
|
281
|
+
cacheKey,
|
|
282
|
+
cacheTtl,
|
|
283
|
+
cachedData
|
|
284
|
+
};
|
|
285
|
+
} catch (e) {
|
|
286
|
+
console.error(e);
|
|
287
|
+
}
|
|
288
|
+
return {
|
|
289
|
+
cacheKey: null,
|
|
290
|
+
cachedData: null
|
|
291
|
+
};
|
|
292
|
+
}
|
|
293
|
+
return {
|
|
294
|
+
cacheKey: null,
|
|
295
|
+
cachedData: null
|
|
296
|
+
};
|
|
297
|
+
})();
|
|
298
|
+
if (cachedData !== null) {
|
|
299
|
+
return cachedData;
|
|
300
|
+
}
|
|
301
|
+
// createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.
|
|
302
|
+
const createSSE = ((_request, _reply, _events)=>createSSEFactory(_request.socket, _reply, _events)).bind(null, request, reply);
|
|
303
|
+
const context = {
|
|
304
|
+
...await Promise.resolve(config.contextProvider({
|
|
305
|
+
request,
|
|
306
|
+
reply,
|
|
307
|
+
headers: request.headers,
|
|
308
|
+
createSSE,
|
|
309
|
+
naiteStore: new Map(),
|
|
310
|
+
// auth
|
|
311
|
+
user: request.user ?? null,
|
|
312
|
+
passport: {
|
|
313
|
+
login: request.login.bind(request),
|
|
314
|
+
logout: request.logout.bind(request)
|
|
315
|
+
}
|
|
316
|
+
}, request, reply))
|
|
317
|
+
};
|
|
318
|
+
const model = this.syncer.models[api.modelName];
|
|
319
|
+
return this.asyncLocalStorage.run({
|
|
320
|
+
context
|
|
321
|
+
}, async ()=>{
|
|
322
|
+
const result = await model[api.methodName].apply(model, api.parameters.map((param)=>{
|
|
323
|
+
// Context 인젝션
|
|
324
|
+
if (ApiParamType.isContext(param.type)) {
|
|
325
|
+
return context;
|
|
326
|
+
} else {
|
|
327
|
+
return reqBody[param.name];
|
|
328
|
+
}
|
|
329
|
+
}));
|
|
330
|
+
reply.type(api.options.contentType ?? "application/json");
|
|
331
|
+
// 캐시 키 있는 경우 갱신 후 저장
|
|
332
|
+
if (config.cache && cacheKey) {
|
|
333
|
+
await config.cache.put(cacheKey, result, cacheTtl);
|
|
334
|
+
}
|
|
335
|
+
return result;
|
|
336
|
+
});
|
|
337
|
+
};
|
|
338
|
+
}
|
|
339
|
+
startWatcher() {
|
|
340
|
+
const watchPath = [
|
|
341
|
+
path.join(this.apiRootPath, "src"),
|
|
342
|
+
path.join(this.apiRootPath, "sonamu.config.ts")
|
|
343
|
+
];
|
|
344
|
+
this.watcher = chokidar.watch(watchPath, {
|
|
345
|
+
ignored: (path, stats)=>!!stats?.isFile() && !path.endsWith(".ts") && !path.endsWith(".json"),
|
|
346
|
+
persistent: true,
|
|
347
|
+
ignoreInitial: true
|
|
348
|
+
});
|
|
349
|
+
this.watcher.on("all", async (event, filePath)=>{
|
|
350
|
+
const absolutePath = filePath;
|
|
351
|
+
assert(absolutePath.startsWith(this.apiRootPath), "File path is not within the API root path");
|
|
352
|
+
if (event !== "change" && event !== "add") {
|
|
353
|
+
return;
|
|
354
|
+
}
|
|
355
|
+
try {
|
|
356
|
+
// sonamu.config.ts 변경 시 재시작
|
|
357
|
+
const isConfigTs = filePath === path.join(this.apiRootPath, "sonamu.config.ts");
|
|
358
|
+
if (isConfigTs) {
|
|
359
|
+
const relativePath = filePath.replace(this.apiRootPath, "api");
|
|
360
|
+
console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`));
|
|
361
|
+
process.kill(process.pid, "SIGUSR2");
|
|
362
|
+
return;
|
|
363
|
+
}
|
|
364
|
+
await this.handleFileChange(event, absolutePath);
|
|
365
|
+
} catch (e) {
|
|
366
|
+
console.error(e);
|
|
367
|
+
}
|
|
368
|
+
});
|
|
369
|
+
}
|
|
370
|
+
/*
|
|
371
|
+
A function that automatically handles init and destroy when using Sonamu via scripts.
|
|
372
|
+
*/ async runScript(fn) {
|
|
373
|
+
await this.init(true, false, undefined, false);
|
|
374
|
+
try {
|
|
375
|
+
await fn();
|
|
376
|
+
} finally{
|
|
377
|
+
await this.destroy();
|
|
378
|
+
}
|
|
379
|
+
}
|
|
380
|
+
registerPlugins(server, plugins) {
|
|
381
|
+
if (!plugins) {
|
|
382
|
+
return;
|
|
383
|
+
}
|
|
384
|
+
const pluginsModules = {
|
|
385
|
+
cors: "@fastify/cors",
|
|
386
|
+
formbody: "@fastify/formbody",
|
|
387
|
+
multipart: "@fastify/multipart",
|
|
388
|
+
qs: "fastify-qs",
|
|
389
|
+
sse: "fastify-sse-v2",
|
|
390
|
+
static: "@fastify/static",
|
|
391
|
+
session: "@fastify/secure-session"
|
|
392
|
+
};
|
|
393
|
+
const registerPlugin = (key, pluginName)=>{
|
|
394
|
+
const option = plugins[key];
|
|
395
|
+
if (!option) return;
|
|
396
|
+
if (option === true) {
|
|
397
|
+
server.register(import(pluginName));
|
|
398
|
+
} else {
|
|
399
|
+
server.register(import(pluginName), option);
|
|
400
|
+
}
|
|
401
|
+
};
|
|
402
|
+
Object.entries(pluginsModules).forEach(([key, pluginName])=>{
|
|
403
|
+
registerPlugin(key, pluginName);
|
|
404
|
+
});
|
|
405
|
+
if (plugins.custom) {
|
|
406
|
+
plugins.custom(server);
|
|
407
|
+
}
|
|
408
|
+
}
|
|
409
|
+
async registerAuth(server, options) {
|
|
410
|
+
server.register(fastifyPassport.initialize());
|
|
411
|
+
server.register(fastifyPassport.secureSession());
|
|
412
|
+
if (typeof options === "boolean") {
|
|
413
|
+
fastifyPassport.registerUserSerializer(async (user, _request)=>user);
|
|
414
|
+
fastifyPassport.registerUserDeserializer(async (serialized, _request)=>serialized);
|
|
415
|
+
} else {
|
|
416
|
+
fastifyPassport.registerUserSerializer(options.userSerializer);
|
|
417
|
+
fastifyPassport.registerUserDeserializer(options.userDeserializer);
|
|
418
|
+
}
|
|
419
|
+
}
|
|
420
|
+
async boot(server, options) {
|
|
421
|
+
const port = options.listen?.port ?? 3000;
|
|
422
|
+
const host = options.listen?.host ?? "localhost";
|
|
423
|
+
server.addHook("onClose", async ()=>{
|
|
424
|
+
await options.lifecycle?.onShutdown?.(server);
|
|
425
|
+
await this.destroy();
|
|
426
|
+
});
|
|
427
|
+
const shutdown = async ()=>{
|
|
428
|
+
try {
|
|
429
|
+
await server.close();
|
|
430
|
+
process.exit(0);
|
|
431
|
+
} catch (err) {
|
|
432
|
+
console.error("Error during shutdown:", err);
|
|
433
|
+
process.exit(1);
|
|
434
|
+
}
|
|
435
|
+
};
|
|
436
|
+
process.on("SIGINT", shutdown);
|
|
437
|
+
process.on("SIGTERM", shutdown);
|
|
438
|
+
if (options.lifecycle?.onError) {
|
|
439
|
+
server.setErrorHandler(options.lifecycle?.onError);
|
|
440
|
+
}
|
|
441
|
+
server.listen({
|
|
442
|
+
port,
|
|
443
|
+
host
|
|
444
|
+
}).then(async ()=>{
|
|
445
|
+
await options.lifecycle?.onStart?.(server);
|
|
446
|
+
}).catch(async (err)=>{
|
|
447
|
+
console.error(chalk.red("Failed to start server:", err));
|
|
448
|
+
await shutdown();
|
|
449
|
+
});
|
|
450
|
+
}
|
|
451
|
+
async handleFileChange(event, filePath) {
|
|
452
|
+
// 첫 번째 파일이면 HMR 시작 시간 기록
|
|
453
|
+
if (this.pendingFiles.length === 0) {
|
|
454
|
+
this.hmrStartTime = Date.now();
|
|
455
|
+
}
|
|
456
|
+
this.pendingFiles.push(filePath);
|
|
457
|
+
const relativePath = path.relative(this.apiRootPath, filePath);
|
|
458
|
+
console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));
|
|
459
|
+
await this.syncer.syncFromWatcher(event, filePath);
|
|
460
|
+
// 처리 완료된 파일을 대기 목록에서 제거
|
|
461
|
+
this.pendingFiles = this.pendingFiles.slice(1);
|
|
462
|
+
// 모든 파일 처리가 완료되면 최종 메시지 출력
|
|
463
|
+
if (this.pendingFiles.length === 0) {
|
|
464
|
+
await this.finishHMR();
|
|
465
|
+
}
|
|
466
|
+
}
|
|
467
|
+
async finishHMR() {
|
|
468
|
+
await this.syncer.renewChecksums();
|
|
469
|
+
const endTime = Date.now();
|
|
470
|
+
const totalTime = endTime - this.hmrStartTime;
|
|
471
|
+
const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;
|
|
472
|
+
console.log(chalk.black.bgGreen(centerText(msg)));
|
|
473
|
+
}
|
|
474
|
+
async destroy() {
|
|
475
|
+
await BaseModel.destroy();
|
|
476
|
+
await this.watcher?.close();
|
|
477
|
+
this.storage?.destroy();
|
|
478
|
+
}
|
|
479
|
+
}
|
|
480
|
+
export const Sonamu = new SonamuClass();
|
|
481
|
+
|
|
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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"}
|
|
@@ -4,8 +4,9 @@
|
|
|
4
4
|
export declare const BUILD_DIR = "dist";
|
|
5
5
|
/**
|
|
6
6
|
* SWC 빌드 명령어
|
|
7
|
+
* .swcrc 설정 사용
|
|
7
8
|
*/
|
|
8
|
-
export declare const SWC_BUILD_COMMAND = "swc src -d dist --strip-leading-paths
|
|
9
|
+
export declare const SWC_BUILD_COMMAND = "swc src -d dist --strip-leading-paths";
|
|
9
10
|
/**
|
|
10
11
|
* TSC 타입 체크 명령어
|
|
11
12
|
*/
|