sonamu 0.6.0 → 0.7.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.
Files changed (406) hide show
  1. package/.swcrc.project-default +18 -0
  2. package/bin/cli.js +24 -0
  3. package/dist/ai/agents/agent.d.ts +11 -0
  4. package/dist/ai/agents/agent.d.ts.map +1 -0
  5. package/dist/ai/agents/agent.js +65 -0
  6. package/dist/ai/agents/index.d.ts +3 -0
  7. package/dist/ai/agents/index.d.ts.map +1 -0
  8. package/dist/ai/agents/index.js +4 -0
  9. package/dist/ai/agents/types.d.ts +43 -0
  10. package/dist/ai/agents/types.d.ts.map +1 -0
  11. package/dist/ai/agents/types.js +3 -0
  12. package/dist/ai/index.d.ts +2 -0
  13. package/dist/ai/index.d.ts.map +1 -0
  14. package/dist/ai/index.js +3 -0
  15. package/dist/ai/providers/rtzr/api.d.ts +22 -0
  16. package/dist/ai/providers/rtzr/api.d.ts.map +1 -0
  17. package/dist/ai/providers/rtzr/api.js +28 -0
  18. package/dist/ai/providers/rtzr/error.d.ts +18 -0
  19. package/dist/ai/providers/rtzr/error.d.ts.map +1 -0
  20. package/dist/ai/providers/rtzr/error.js +29 -0
  21. package/dist/ai/providers/rtzr/index.d.ts +5 -0
  22. package/dist/ai/providers/rtzr/index.d.ts.map +1 -0
  23. package/dist/ai/providers/rtzr/index.js +6 -0
  24. package/dist/ai/providers/rtzr/model.d.ts +52 -0
  25. package/dist/ai/providers/rtzr/model.d.ts.map +1 -0
  26. package/dist/ai/providers/rtzr/model.js +137 -0
  27. package/dist/ai/providers/rtzr/options.d.ts +7 -0
  28. package/dist/ai/providers/rtzr/options.d.ts.map +1 -0
  29. package/dist/ai/providers/rtzr/options.js +47 -0
  30. package/dist/ai/providers/rtzr/provider.d.ts +18 -0
  31. package/dist/ai/providers/rtzr/provider.d.ts.map +1 -0
  32. package/dist/ai/providers/rtzr/provider.js +54 -0
  33. package/dist/ai/providers/rtzr/utils.d.ts +19 -0
  34. package/dist/ai/providers/rtzr/utils.d.ts.map +1 -0
  35. package/dist/ai/providers/rtzr/utils.js +88 -0
  36. package/dist/api/base-frame.d.ts +2 -2
  37. package/dist/api/base-frame.d.ts.map +1 -1
  38. package/dist/api/base-frame.js +2 -1
  39. package/dist/api/caster.d.ts.map +1 -1
  40. package/dist/api/caster.js +6 -1
  41. package/dist/api/code-converters.d.ts +58 -14
  42. package/dist/api/code-converters.d.ts.map +1 -1
  43. package/dist/api/code-converters.js +178 -409
  44. package/dist/api/config.d.ts +27 -13
  45. package/dist/api/config.d.ts.map +1 -1
  46. package/dist/api/config.js +19 -26
  47. package/dist/api/context.d.ts +4 -3
  48. package/dist/api/context.d.ts.map +1 -1
  49. package/dist/api/context.js +1 -1
  50. package/dist/api/decorators.d.ts +20 -6
  51. package/dist/api/decorators.d.ts.map +1 -1
  52. package/dist/api/decorators.js +111 -18
  53. package/dist/api/index.d.ts +2 -2
  54. package/dist/api/index.d.ts.map +1 -1
  55. package/dist/api/index.js +3 -3
  56. package/dist/api/sonamu.d.ts +7 -7
  57. package/dist/api/sonamu.d.ts.map +1 -1
  58. package/dist/api/sonamu.js +83 -51
  59. package/dist/api/validator.d.ts +6 -0
  60. package/dist/api/validator.d.ts.map +1 -0
  61. package/dist/api/validator.js +81 -0
  62. package/dist/bin/build-config.d.ts +5 -1
  63. package/dist/bin/build-config.d.ts.map +1 -1
  64. package/dist/bin/build-config.js +5 -2
  65. package/dist/bin/cli.js +165 -64
  66. package/dist/bin/loader-register.d.ts +2 -0
  67. package/dist/bin/loader-register.d.ts.map +1 -0
  68. package/dist/bin/loader-register.js +34 -0
  69. package/dist/database/_batch_update.d.ts +5 -3
  70. package/dist/database/_batch_update.d.ts.map +1 -1
  71. package/dist/database/_batch_update.js +30 -13
  72. package/dist/database/base-model.d.ts +96 -10
  73. package/dist/database/base-model.d.ts.map +1 -1
  74. package/dist/database/base-model.js +232 -89
  75. package/dist/database/base-model.types.d.ts +93 -0
  76. package/dist/database/base-model.types.d.ts.map +1 -0
  77. package/dist/database/base-model.types.js +10 -0
  78. package/dist/database/code-generator.d.ts +1 -1
  79. package/dist/database/code-generator.d.ts.map +1 -1
  80. package/dist/database/code-generator.js +11 -10
  81. package/dist/database/db.d.ts +5 -6
  82. package/dist/database/db.d.ts.map +1 -1
  83. package/dist/database/db.js +22 -25
  84. package/dist/database/puri-subset.test-d.js +81 -0
  85. package/dist/database/puri-subset.types.d.ts +123 -0
  86. package/dist/database/puri-subset.types.d.ts.map +1 -0
  87. package/dist/database/puri-subset.types.js +16 -0
  88. package/dist/database/puri-wrapper.d.ts +13 -11
  89. package/dist/database/puri-wrapper.d.ts.map +1 -1
  90. package/dist/database/puri-wrapper.js +2 -2
  91. package/dist/database/puri.d.ts +25 -14
  92. package/dist/database/puri.d.ts.map +1 -1
  93. package/dist/database/puri.js +83 -21
  94. package/dist/database/puri.types.d.ts +21 -7
  95. package/dist/database/puri.types.d.ts.map +1 -1
  96. package/dist/database/puri.types.js +4 -1
  97. package/dist/database/transaction-context.d.ts +1 -1
  98. package/dist/database/transaction-context.d.ts.map +1 -1
  99. package/dist/database/transaction-context.js +1 -1
  100. package/dist/database/upsert-builder.d.ts +9 -3
  101. package/dist/database/upsert-builder.d.ts.map +1 -1
  102. package/dist/database/upsert-builder.js +228 -78
  103. package/dist/entity/entity-manager.d.ts +165 -2
  104. package/dist/entity/entity-manager.d.ts.map +1 -1
  105. package/dist/entity/entity-manager.js +26 -10
  106. package/dist/entity/entity.d.ts +5 -3
  107. package/dist/entity/entity.d.ts.map +1 -1
  108. package/dist/entity/entity.js +153 -54
  109. package/dist/exceptions/error-handler.d.ts +1 -1
  110. package/dist/exceptions/error-handler.d.ts.map +1 -1
  111. package/dist/exceptions/error-handler.js +1 -1
  112. package/dist/exceptions/so-exceptions.d.ts +1 -1
  113. package/dist/exceptions/so-exceptions.d.ts.map +1 -1
  114. package/dist/exceptions/so-exceptions.js +1 -1
  115. package/dist/file-storage/driver.d.ts +1 -1
  116. package/dist/file-storage/driver.d.ts.map +1 -1
  117. package/dist/file-storage/driver.js +1 -1
  118. package/dist/file-storage/file-storage.js +2 -2
  119. package/dist/index.d.ts +18 -11
  120. package/dist/index.d.ts.map +1 -1
  121. package/dist/index.js +19 -13
  122. package/dist/migration/code-generation.d.ts +1 -1
  123. package/dist/migration/code-generation.d.ts.map +1 -1
  124. package/dist/migration/code-generation.js +123 -67
  125. package/dist/migration/migration-set.d.ts +2 -10
  126. package/dist/migration/migration-set.d.ts.map +1 -1
  127. package/dist/migration/migration-set.js +67 -218
  128. package/dist/migration/migrator.d.ts +24 -73
  129. package/dist/migration/migrator.d.ts.map +1 -1
  130. package/dist/migration/migrator.js +121 -301
  131. package/dist/migration/postgresql-schema-reader.d.ts +51 -0
  132. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -0
  133. package/dist/migration/postgresql-schema-reader.js +245 -0
  134. package/dist/migration/types.d.ts +6 -38
  135. package/dist/migration/types.d.ts.map +1 -1
  136. package/dist/migration/types.js +1 -1
  137. package/dist/naite/messaging-types.d.ts +43 -0
  138. package/dist/naite/messaging-types.d.ts.map +1 -0
  139. package/dist/naite/messaging-types.js +7 -0
  140. package/dist/naite/naite-reporter.d.ts +41 -0
  141. package/dist/naite/naite-reporter.d.ts.map +1 -0
  142. package/dist/naite/naite-reporter.js +102 -0
  143. package/dist/naite/naite.d.ts +91 -8
  144. package/dist/naite/naite.d.ts.map +1 -1
  145. package/dist/naite/naite.js +285 -41
  146. package/dist/stream/sse.d.ts +2 -2
  147. package/dist/stream/sse.d.ts.map +1 -1
  148. package/dist/stream/sse.js +1 -1
  149. package/dist/syncer/api-parser.d.ts +3 -13
  150. package/dist/syncer/api-parser.d.ts.map +1 -1
  151. package/dist/syncer/api-parser.js +67 -56
  152. package/dist/syncer/checksum.d.ts +2 -2
  153. package/dist/syncer/checksum.d.ts.map +1 -1
  154. package/dist/syncer/checksum.js +11 -11
  155. package/dist/syncer/code-generator.d.ts +3 -3
  156. package/dist/syncer/code-generator.d.ts.map +1 -1
  157. package/dist/syncer/code-generator.js +37 -17
  158. package/dist/syncer/entity-operations.d.ts +2 -2
  159. package/dist/syncer/entity-operations.d.ts.map +1 -1
  160. package/dist/syncer/entity-operations.js +9 -8
  161. package/dist/syncer/file-patterns.d.ts +1 -1
  162. package/dist/syncer/file-patterns.d.ts.map +1 -1
  163. package/dist/syncer/file-patterns.js +1 -1
  164. package/dist/syncer/index.d.ts +4 -4
  165. package/dist/syncer/index.d.ts.map +1 -1
  166. package/dist/syncer/index.js +5 -5
  167. package/dist/syncer/module-loader.d.ts +4 -4
  168. package/dist/syncer/module-loader.d.ts.map +1 -1
  169. package/dist/syncer/module-loader.js +17 -12
  170. package/dist/syncer/syncer.d.ts +31 -24
  171. package/dist/syncer/syncer.d.ts.map +1 -1
  172. package/dist/syncer/syncer.js +92 -45
  173. package/dist/template/entity-converter.d.ts +1 -1
  174. package/dist/template/entity-converter.d.ts.map +1 -1
  175. package/dist/template/entity-converter.js +15 -8
  176. package/dist/template/helpers.d.ts +2 -2
  177. package/dist/template/helpers.d.ts.map +1 -1
  178. package/dist/template/helpers.js +3 -3
  179. package/dist/template/implementations/entity.template.d.ts +2 -2
  180. package/dist/template/implementations/entity.template.d.ts.map +1 -1
  181. package/dist/template/implementations/entity.template.js +4 -5
  182. package/dist/template/implementations/generated.template.d.ts +2 -3
  183. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  184. package/dist/template/implementations/generated.template.js +46 -29
  185. package/dist/template/implementations/generated_http.template.d.ts +2 -3
  186. package/dist/template/implementations/generated_http.template.d.ts.map +1 -1
  187. package/dist/template/implementations/generated_http.template.js +9 -9
  188. package/dist/template/implementations/generated_sso.template.d.ts +3 -4
  189. package/dist/template/implementations/generated_sso.template.d.ts.map +1 -1
  190. package/dist/template/implementations/generated_sso.template.js +54 -25
  191. package/dist/template/implementations/init_types.template.d.ts +2 -2
  192. package/dist/template/implementations/init_types.template.d.ts.map +1 -1
  193. package/dist/template/implementations/init_types.template.js +2 -2
  194. package/dist/template/implementations/model.template.d.ts +2 -2
  195. package/dist/template/implementations/model.template.d.ts.map +1 -1
  196. package/dist/template/implementations/model.template.js +47 -37
  197. package/dist/template/implementations/model_test.template.d.ts +2 -2
  198. package/dist/template/implementations/model_test.template.d.ts.map +1 -1
  199. package/dist/template/implementations/model_test.template.js +2 -2
  200. package/dist/template/implementations/service.template.d.ts +4 -4
  201. package/dist/template/implementations/service.template.d.ts.map +1 -1
  202. package/dist/template/implementations/service.template.js +24 -16
  203. package/dist/template/implementations/view_enums_buttonset.template.d.ts +2 -2
  204. package/dist/template/implementations/view_enums_buttonset.template.d.ts.map +1 -1
  205. package/dist/template/implementations/view_enums_buttonset.template.js +1 -1
  206. package/dist/template/implementations/view_enums_dropdown.template.d.ts +2 -2
  207. package/dist/template/implementations/view_enums_dropdown.template.d.ts.map +1 -1
  208. package/dist/template/implementations/view_enums_dropdown.template.js +2 -2
  209. package/dist/template/implementations/view_enums_select.template.d.ts +2 -2
  210. package/dist/template/implementations/view_enums_select.template.d.ts.map +1 -1
  211. package/dist/template/implementations/view_enums_select.template.js +2 -2
  212. package/dist/template/implementations/view_form.template.d.ts +2 -2
  213. package/dist/template/implementations/view_form.template.d.ts.map +1 -1
  214. package/dist/template/implementations/view_form.template.js +4 -4
  215. package/dist/template/implementations/view_id_all_select.template.d.ts +2 -2
  216. package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -1
  217. package/dist/template/implementations/view_id_all_select.template.js +1 -1
  218. package/dist/template/implementations/view_id_async_select.template.d.ts +2 -2
  219. package/dist/template/implementations/view_id_async_select.template.d.ts.map +1 -1
  220. package/dist/template/implementations/view_id_async_select.template.js +1 -1
  221. package/dist/template/implementations/view_list.template.d.ts +2 -2
  222. package/dist/template/implementations/view_list.template.d.ts.map +1 -1
  223. package/dist/template/implementations/view_list.template.js +29 -19
  224. package/dist/template/implementations/view_list_columns.template.d.ts +3 -3
  225. package/dist/template/implementations/view_list_columns.template.d.ts.map +1 -1
  226. package/dist/template/implementations/view_list_columns.template.js +1 -1
  227. package/dist/template/implementations/view_search_input.template.d.ts +2 -2
  228. package/dist/template/implementations/view_search_input.template.d.ts.map +1 -1
  229. package/dist/template/implementations/view_search_input.template.js +1 -1
  230. package/dist/template/index.d.ts +4 -2
  231. package/dist/template/index.d.ts.map +1 -1
  232. package/dist/template/index.js +5 -3
  233. package/dist/template/template-manager.d.ts +56 -0
  234. package/dist/template/template-manager.d.ts.map +1 -0
  235. package/dist/template/template-manager.js +125 -0
  236. package/dist/template/template-types.d.ts +16 -0
  237. package/dist/template/template-types.d.ts.map +1 -0
  238. package/dist/template/template-types.js +7 -0
  239. package/dist/template/template.d.ts +12 -2
  240. package/dist/template/template.d.ts.map +1 -1
  241. package/dist/template/template.js +19 -6
  242. package/dist/template/zod-converter.d.ts +40 -7
  243. package/dist/template/zod-converter.d.ts.map +1 -1
  244. package/dist/template/zod-converter.js +341 -58
  245. package/dist/testing/_relation-graph.d.ts +1 -1
  246. package/dist/testing/_relation-graph.d.ts.map +1 -1
  247. package/dist/testing/_relation-graph.js +12 -3
  248. package/dist/testing/fixture-manager.d.ts +42 -11
  249. package/dist/testing/fixture-manager.d.ts.map +1 -1
  250. package/dist/testing/fixture-manager.js +338 -236
  251. package/dist/types/types.d.ts +709 -104
  252. package/dist/types/types.d.ts.map +1 -1
  253. package/dist/types/types.js +309 -52
  254. package/dist/typings/knex.d.js +2 -2
  255. package/dist/utils/async-utils.d.ts.map +1 -1
  256. package/dist/utils/async-utils.js +3 -3
  257. package/dist/utils/console-util.js +1 -1
  258. package/dist/utils/controller.d.ts +1 -0
  259. package/dist/utils/controller.d.ts.map +1 -1
  260. package/dist/utils/controller.js +4 -1
  261. package/dist/utils/esm-utils.d.ts +0 -6
  262. package/dist/utils/esm-utils.d.ts.map +1 -1
  263. package/dist/utils/esm-utils.js +2 -9
  264. package/dist/utils/formatter.d.ts +3 -0
  265. package/dist/utils/formatter.d.ts.map +1 -0
  266. package/dist/utils/formatter.js +110 -0
  267. package/dist/utils/fs-utils.d.ts +1 -1
  268. package/dist/utils/fs-utils.d.ts.map +1 -1
  269. package/dist/utils/fs-utils.js +1 -1
  270. package/dist/utils/lodash-able.d.ts.map +1 -1
  271. package/dist/utils/lodash-able.js +1 -1
  272. package/dist/utils/object-utils.d.ts +44 -0
  273. package/dist/utils/object-utils.d.ts.map +1 -0
  274. package/dist/utils/object-utils.js +191 -0
  275. package/dist/utils/path-utils.d.ts +1 -1
  276. package/dist/utils/path-utils.d.ts.map +1 -1
  277. package/dist/utils/path-utils.js +3 -3
  278. package/dist/utils/process-utils.js +1 -1
  279. package/dist/utils/sql-parser.d.ts +5 -1
  280. package/dist/utils/sql-parser.d.ts.map +1 -1
  281. package/dist/utils/sql-parser.js +14 -3
  282. package/dist/utils/type-utils.d.ts +23 -0
  283. package/dist/utils/type-utils.d.ts.map +1 -0
  284. package/dist/utils/type-utils.js +45 -0
  285. package/dist/utils/utils.d.ts +7 -1
  286. package/dist/utils/utils.d.ts.map +1 -1
  287. package/dist/utils/utils.js +44 -5
  288. package/dist/utils/zod-error.d.ts +1 -1
  289. package/dist/utils/zod-error.d.ts.map +1 -1
  290. package/dist/utils/zod-error.js +1 -1
  291. package/package.json +54 -29
  292. package/src/ai/agents/agent.ts +87 -0
  293. package/src/ai/agents/index.ts +2 -0
  294. package/src/ai/agents/types.ts +47 -0
  295. package/src/ai/index.ts +1 -0
  296. package/src/ai/providers/rtzr/api.ts +37 -0
  297. package/src/ai/providers/rtzr/error.ts +34 -0
  298. package/src/ai/providers/rtzr/index.ts +4 -0
  299. package/src/ai/providers/rtzr/model.ts +201 -0
  300. package/src/ai/providers/rtzr/options.ts +49 -0
  301. package/src/ai/providers/rtzr/provider.ts +91 -0
  302. package/src/ai/providers/rtzr/utils.ts +127 -0
  303. package/src/api/base-frame.ts +4 -2
  304. package/src/api/caster.ts +17 -23
  305. package/src/api/code-converters.ts +176 -533
  306. package/src/api/config.ts +39 -56
  307. package/src/api/context.ts +7 -18
  308. package/src/api/decorators.ts +175 -46
  309. package/src/api/index.ts +2 -2
  310. package/src/api/sonamu.ts +133 -124
  311. package/src/api/validator.ts +83 -0
  312. package/src/bin/build-config.ts +7 -1
  313. package/src/bin/cli.ts +192 -110
  314. package/src/bin/loader-register.ts +38 -0
  315. package/src/database/_batch_update.ts +46 -31
  316. package/src/database/base-model.ts +390 -182
  317. package/src/database/base-model.types.ts +155 -0
  318. package/src/database/code-generator.ts +13 -32
  319. package/src/database/db.ts +36 -50
  320. package/src/database/puri-subset.test-d.ts +471 -0
  321. package/src/database/puri-subset.types.ts +195 -0
  322. package/src/database/puri-wrapper.ts +58 -67
  323. package/src/database/puri.ts +182 -126
  324. package/src/database/puri.types.ts +64 -31
  325. package/src/database/transaction-context.ts +1 -1
  326. package/src/database/upsert-builder.ts +262 -132
  327. package/src/entity/entity-manager.ts +36 -28
  328. package/src/entity/entity.ts +330 -249
  329. package/src/exceptions/error-handler.ts +3 -3
  330. package/src/exceptions/so-exceptions.ts +11 -11
  331. package/src/file-storage/driver.ts +5 -5
  332. package/src/file-storage/file-storage.ts +2 -2
  333. package/src/index.ts +18 -12
  334. package/src/migration/code-generation.ts +185 -172
  335. package/src/migration/migration-set.ts +80 -293
  336. package/src/migration/migrator.ts +182 -425
  337. package/src/migration/mysql-schema-reader.ts.txt +272 -0
  338. package/src/migration/postgresql-schema-reader.ts +310 -0
  339. package/src/migration/types.ts +6 -39
  340. package/src/naite/messaging-types.ts +51 -0
  341. package/src/naite/naite-reporter.ts +128 -0
  342. package/src/naite/naite.ts +378 -33
  343. package/src/shared/web.shared.ts.txt +20 -24
  344. package/src/stream/sse.ts +5 -5
  345. package/src/syncer/api-parser.ts +52 -69
  346. package/src/syncer/checksum.ts +25 -37
  347. package/src/syncer/code-generator.ts +58 -62
  348. package/src/syncer/entity-operations.ts +12 -15
  349. package/src/syncer/file-patterns.ts +2 -2
  350. package/src/syncer/index.ts +4 -4
  351. package/src/syncer/module-loader.ts +28 -25
  352. package/src/syncer/syncer.ts +155 -162
  353. package/src/template/entity-converter.ts +18 -27
  354. package/src/template/helpers.ts +8 -11
  355. package/src/template/implementations/entity.template.ts +6 -6
  356. package/src/template/implementations/generated.template.ts +99 -99
  357. package/src/template/implementations/generated_http.template.ts +21 -54
  358. package/src/template/implementations/generated_sso.template.ts +78 -65
  359. package/src/template/implementations/init_types.template.ts +4 -6
  360. package/src/template/implementations/model.template.ts +47 -38
  361. package/src/template/implementations/model_test.template.ts +3 -3
  362. package/src/template/implementations/service.template.ts +56 -80
  363. package/src/template/implementations/view_enums_buttonset.template.ts +2 -2
  364. package/src/template/implementations/view_enums_dropdown.template.ts +4 -4
  365. package/src/template/implementations/view_enums_select.template.ts +3 -3
  366. package/src/template/implementations/view_form.template.ts +34 -75
  367. package/src/template/implementations/view_id_all_select.template.ts +2 -2
  368. package/src/template/implementations/view_id_async_select.template.ts +9 -23
  369. package/src/template/implementations/view_list.template.ts +54 -95
  370. package/src/template/implementations/view_list_columns.template.ts +4 -10
  371. package/src/template/implementations/view_search_input.template.ts +2 -2
  372. package/src/template/index.ts +4 -2
  373. package/src/template/template-manager.ts +166 -0
  374. package/src/template/template-types.ts +16 -0
  375. package/src/template/template.ts +29 -10
  376. package/src/template/zod-converter.ts +407 -101
  377. package/src/testing/_relation-graph.ts +18 -11
  378. package/src/testing/fixture-manager.ts +468 -362
  379. package/src/types/types.ts +516 -248
  380. package/src/typings/knex.d.ts +7 -9
  381. package/src/utils/async-utils.ts +8 -12
  382. package/src/utils/console-util.ts +1 -1
  383. package/src/utils/controller.ts +3 -0
  384. package/src/utils/esm-utils.ts +8 -18
  385. package/src/utils/formatter.ts +109 -0
  386. package/src/utils/fs-utils.ts +1 -1
  387. package/src/utils/lodash-able.ts +1 -4
  388. package/src/utils/object-utils.ts +217 -0
  389. package/src/utils/path-utils.ts +3 -6
  390. package/src/utils/process-utils.ts +1 -1
  391. package/src/utils/sql-parser.ts +23 -5
  392. package/src/utils/type-utils.ts +83 -0
  393. package/src/utils/utils.ts +58 -9
  394. package/src/utils/zod-error.ts +3 -3
  395. package/dist/bin/cli-wrapper.d.ts +0 -3
  396. package/dist/bin/cli-wrapper.d.ts.map +0 -1
  397. package/dist/bin/cli-wrapper.js +0 -72
  398. package/dist/database/knex-plugins/knex-on-duplicate-update.d.ts +0 -2
  399. package/dist/database/knex-plugins/knex-on-duplicate-update.d.ts.map +0 -1
  400. package/dist/database/knex-plugins/knex-on-duplicate-update.js +0 -39
  401. package/dist/entity/entity-utils.d.ts +0 -61
  402. package/dist/entity/entity-utils.d.ts.map +0 -1
  403. package/dist/entity/entity-utils.js +0 -210
  404. package/src/bin/cli-wrapper.ts +0 -82
  405. package/src/database/knex-plugins/knex-on-duplicate-update.ts +0 -45
  406. package/src/entity/entity-utils.ts +0 -291
@@ -1 +1 @@
1
- {"version":3,"file":"sonamu.d.ts","sourceRoot":"","sources":["../../src/api/sonamu.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAMhD,OAAiB,EAAE,KAAK,SAAS,EAAE,MAAM,UAAU,CAAC;AAEpD,OAAO,KAAK,EAAE,eAAe,EAAE,YAAY,EAAE,cAAc,EAAE,MAAM,SAAS,CAAC;AAC7E,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,EAAE,cAAc,EAAE,MAAM,MAAM,CAAC;AAEpE,OAAO,EAAM,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAMpD,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,wBAAwB,CAAC;AAErD,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAC/C,OAAO,EAAgB,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAMnE,OAAO,KAAK,EAAe,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AACrE,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,cAAc,CAAC;AAEhD,OAAO,EAAc,YAAY,EAAuB,MAAM,UAAU,CAAC;AACzE,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAOnD,MAAM,MAAM,aAAa,GAAG;IAC1B,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAAC;CACvB,CAAC;AACF,cAAM,WAAW;IACR,aAAa,EAAE,OAAO,CAAS;IAC/B,iBAAiB,EAAE,iBAAiB,CAAC;QAC1C,OAAO,EAAE,OAAO,CAAC;KAClB,CAAC,CAA2B;IAEtB,aAAa,EAAE,iBAAiB,CAAC;QACtC,aAAa,EAAE,aAAa,CAAC;KAC9B,CAAC,CAA2B;IAEtB,UAAU,IAAI,OAAO;IAoBrB,gBAAgB,IAAI,aAAa;IAUxC,OAAO,CAAC,YAAY,CAA6B;IACjD,IAAI,WAAW,CAAC,WAAW,EAAE,YAAY,EAExC;IACD,IAAI,WAAW,IAAI,YAAY,CAK9B;IACD,IAAI,WAAW,IAAI,MAAM,CAExB;IAED,OAAO,CAAC,SAAS,CAA+B;IAChD,IAAI,QAAQ,CAAC,QAAQ,EAAE,cAAc,EAEpC;IACD,IAAI,QAAQ,IAAI,cAAc,CAK7B;IAED,OAAO,CAAC,OAAO,CAAuB;IACtC,IAAI,MAAM,CAAC,MAAM,EAAE,MAAM,EAExB;IACD,IAAI,MAAM,IAAI,MAAM,CAKnB;IAED,OAAO,CAAC,OAAO,CAA6B;IAC5C,IAAI,MAAM,CAAC,MAAM,EAAE,YAAY,EAE9B;IACD,IAAI,MAAM,IAAI,YAAY,CAKzB;IAED,OAAO,CAAC,QAAQ,CAA8B;IAC9C,IAAI,OAAO,CAAC,OAAO,EAAE,aAAa,EAEjC;IACD,IAAI,OAAO,IAAI,aAAa,GAAG,IAAI,CAElC;IAED,OAAO,CAAC,QAAQ,CAAuB;IACvC,IAAI,OAAO,CAAC,OAAO,EAAE,MAAM,EAE1B;IACD,IAAI,OAAO,IAAI,MAAM,GAAG,IAAI,CAE3B;IAGM,OAAO,EAAE,SAAS,GAAG,IAAI,CAAQ;IACxC,OAAO,CAAC,YAAY,CAAgB;IACpC,OAAO,CAAC,YAAY,CAAa;IAE1B,MAAM,EAAE,eAAe,GAAG,IAAI,CAAQ;IAEvC,cAAc;IAId,IAAI,CACR,QAAQ,GAAE,OAAe,EACzB,UAAU,GAAE,OAAc,EAC1B,WAAW,CAAC,EAAE,YAAY,EAC1B,UAAU,GAAE,OAAe;IA0DvB,YAAY,CAAC,WAAW,CAAC,EAAE;QAC/B,UAAU,CAAC,EAAE,OAAO,CAAC;QACrB,QAAQ,CAAC,EAAE,OAAO,CAAC;KACpB;IAyCK,WAAW,CACf,MAAM,EAAE,eAAe,CAAC,MAAM,EAAE,eAAe,EAAE,cAAc,CAAC,EAChE,MAAM,EAAE,mBAAmB,EAC3B,OAAO,CAAC,EAAE;QACR,UAAU,CAAC,EAAE,OAAO,CAAC;QACrB,QAAQ,CAAC,EAAE,OAAO,CAAC;KACpB;IA0EH,aAAa,CAAC,GAAG,EAAE,WAAW,EAAE,MAAM,EAAE,mBAAmB,IAEvD,SAAS,cAAc,EACvB,OAAO,YAAY,KAClB,OAAO,CAAC,OAAO,CAAC;IAoHrB,YAAY,IAAI,IAAI;IAkDd,SAAS,CAAC,EAAE,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC;IASvC,OAAO,CAAC,eAAe;YAyCT,YAAY;YAkBZ,IAAI;YAqCJ,gBAAgB;YAwBhB,SAAS;IAUjB,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;CAK/B;AACD,eAAO,MAAM,MAAM,aAAoB,CAAC"}
1
+ {"version":3,"file":"sonamu.d.ts","sourceRoot":"","sources":["../../src/api/sonamu.ts"],"names":[],"mappings":"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"}
@@ -1,29 +1,7 @@
1
+ import assert from "assert";
1
2
  import { AsyncLocalStorage } from "async_hooks";
2
- import chalk from "chalk";
3
- import fastify from "fastify";
4
- import { readFile } from "node:fs/promises";
5
3
  import path from "path";
6
- import { 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";
4
+ import { Naite } from "../naite/naite.js";
27
5
  class SonamuClass {
28
6
  isInitialized = false;
29
7
  asyncLocalStorage = new AsyncLocalStorage();
@@ -40,6 +18,7 @@ class SonamuClass {
40
18
  reply: null,
41
19
  headers: {},
42
20
  createSSE: ()=>{},
21
+ // biome-ignore lint/suspicious/noExplicitAny: 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴
43
22
  naiteStore: new Map()
44
23
  };
45
24
  } else {
@@ -122,18 +101,29 @@ class SonamuClass {
122
101
  if (this.isInitialized) {
123
102
  return;
124
103
  }
125
- !doSilent && console.time(chalk.cyan(`Sonamu.init${forTesting ? " for testing" : ""}`));
104
+ if (!doSilent) {
105
+ const chalk = (await import("chalk")).default;
106
+ console.time(chalk.cyan(`Sonamu.init${forTesting ? " for testing" : ""}`));
107
+ }
126
108
  // API 루트 패스
109
+ const { findApiRootPath } = await import("../utils/utils.js");
127
110
  this.apiRootPath = apiRootPath ?? findApiRootPath();
111
+ const { loadConfig } = await import("./config.js");
128
112
  this.config = await loadConfig(this.apiRootPath);
129
- const secretsPath = path.join(this.apiRootPath, "sonamu.secrets.json");
130
- if (await exists(secretsPath)) {
131
- this.secrets = JSON.parse((await readFile(secretsPath)).toString());
113
+ // sonamu.config.ts 기본값 설정
114
+ this.config.database.database = this.config.database.database ?? "postgresql";
115
+ if (process.env.ANTHROPIC_API_KEY) {
116
+ this.secrets = {
117
+ anthropic_api_key: process.env.ANTHROPIC_API_KEY
118
+ };
132
119
  }
133
120
  // DB 로드
121
+ const { DB } = await import("../database/db.js");
134
122
  this.dbConfig = DB.generateDBConfig(this.config.database);
135
- !doSilent && console.log(chalk.green("DB Config Loaded!"));
136
- attachOnDuplicateUpdate();
123
+ if (!doSilent) {
124
+ const chalk = (await import("chalk")).default;
125
+ console.log(chalk.green("DB Config Loaded!"));
126
+ }
137
127
  // 테스팅인 경우 엔티티 로드 & 싱크 없이 중단
138
128
  if (forTesting) {
139
129
  this.isInitialized = true;
@@ -149,20 +139,31 @@ class SonamuClass {
149
139
  await this.syncer.autoloadTypes();
150
140
  await this.syncer.autoloadModels();
151
141
  await this.syncer.autoloadApis();
152
- await Template.autoload();
142
+ const { TemplateManager } = await import("../template/index.js");
143
+ await TemplateManager.autoload();
144
+ const { isLocal, isTest } = await import("../utils/controller.js");
145
+ if (isLocal()) {
146
+ // 로컬에서는 코드 생성을 위해 Biome 셋업이 필요함 (현재 apiRootPath 전달하여 실행)
147
+ (await import("../utils/formatter.js")).setupBiome(this.apiRootPath);
148
+ }
149
+ const { isHotReloadServer } = await import("../utils/controller.js");
153
150
  if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {
154
151
  await this.syncer.sync();
155
- this.startWatcher();
152
+ await this.startWatcher();
156
153
  this.syncer.syncUI();
157
154
  }
158
155
  this.isInitialized = true;
159
- !doSilent && console.timeEnd(chalk.cyan("Sonamu.init"));
156
+ if (!doSilent) {
157
+ const chalk = (await import("chalk")).default;
158
+ console.timeEnd(chalk.cyan("Sonamu.init"));
159
+ }
160
160
  }
161
161
  async createServer(initOptions) {
162
162
  if (this.isInitialized === false) {
163
163
  await this.init(initOptions?.doSilent, initOptions?.enableSync);
164
164
  }
165
165
  const options = this.config.server;
166
+ const fastify = (await import("fastify")).default;
166
167
  const server = fastify(options.fastify);
167
168
  this.server = server;
168
169
  // Storage 설정 저장
@@ -171,13 +172,13 @@ class SonamuClass {
171
172
  }
172
173
  // 플러그인 등록
173
174
  if (options.plugins) {
174
- this.registerPlugins(server, options.plugins);
175
+ await this.registerPlugins(server, options.plugins);
175
176
  }
176
177
  if (options.auth) {
177
178
  if (!options.plugins?.session) {
178
179
  throw new Error("Auth requires session plugin. Please add plugins.session configuration.");
179
180
  }
180
- this.registerAuth(server, options.auth);
181
+ await this.registerAuth(server, options.auth);
181
182
  }
182
183
  // API 라우팅 설정
183
184
  await this.withFastify(server, options.apiConfig, {
@@ -196,9 +197,16 @@ class SonamuClass {
196
197
  // timezone 설정
197
198
  const timezone = this.config.api.timezone;
198
199
  if (timezone) {
199
- const DATE_FORMAT = "yyyy-MM-dd'T'HH:mm:ssXXX";
200
+ // 타임존에 맞게 응답 날짜 스트링을 변환해주어야 합니다.
201
+ // 가령 timezone이 "Asia/Seoul" 이면
202
+ // "2025-11-21T00:00:00.000Z" 를 "2025-11-21T09:00:00+09:00" 으로 변환해주어야 합니다.
203
+ const { formatInTimeZone } = await import("date-fns-tz");
200
204
  // ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)
201
205
  const ISO_DATE_REGEX = /^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(\.\d{3})?Z$/;
206
+ // T를 둘러싼 작은따옴표가 없다면 "2025-11-19176354618900018:56:29+09:00"와 같은 결과가 나옵니다.
207
+ // 이는 date-fns 특입니다.
208
+ // 이렇게 해도 괜찮습니다. "2025-11-19T18:56:29+09:00" 모양으로 잘 나옵니다.
209
+ const DATE_FORMAT = "yyyy-MM-dd'T'HH:mm:ssXXX";
202
210
  server.setReplySerializer((payload)=>{
203
211
  return JSON.stringify(payload, (_key, value)=>{
204
212
  if (typeof value === "string" && ISO_DATE_REGEX.test(value)) {
@@ -207,7 +215,10 @@ class SonamuClass {
207
215
  return value;
208
216
  });
209
217
  });
210
- !options?.doSilent && console.log(chalk.green(`Timezone set to ${timezone}`));
218
+ if (!options?.doSilent) {
219
+ const chalk = (await import("chalk")).default;
220
+ console.log(chalk.green(`Timezone set to ${timezone}`));
221
+ }
211
222
  }
212
223
  // 전체 라우팅 리스트
213
224
  server.get(`${this.config.api.route.prefix}/routes`, async (_request, _reply)=>{
@@ -218,42 +229,49 @@ class SonamuClass {
218
229
  return "ok";
219
230
  });
220
231
  // API 라우팅 (로컬HMR 상태와 구분)
232
+ const { isLocal } = await import("../utils/controller.js");
221
233
  if (isLocal()) {
222
- server.all("*", (request, reply)=>{
234
+ server.all("*", async (request, reply)=>{
223
235
  const found = this.syncer.apis.find((api)=>this.config.api.route.prefix + api.path === request.url.split("?")[0] && (api.options.httpMethod ?? "GET") === request.method.toUpperCase());
224
236
  if (found) {
225
237
  return this.getApiHandler(found, config)(request, reply);
226
238
  }
239
+ const { NotFoundException } = await import("../exceptions/so-exceptions.js");
227
240
  throw new NotFoundException("존재하지 않는 API 접근입니다.");
228
241
  });
229
242
  } else {
230
- this.syncer.apis.map((api)=>{
243
+ for (const api of this.syncer.apis){
231
244
  // model
232
245
  if (this.syncer.models[api.modelName] === undefined) {
233
246
  throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);
234
247
  }
235
248
  // route
236
249
  server.route({
237
- method: api.options.httpMethod,
250
+ method: api.options.httpMethod ?? "GET",
238
251
  url: this.config.api.route.prefix + api.path,
239
252
  handler: this.getApiHandler(api, config)
240
253
  }); // END server.route
241
- });
254
+ }
242
255
  }
243
256
  }
244
257
  getApiHandler(api, config) {
245
258
  return async (request, reply)=>{
246
259
  (api.options.guards ?? []).every((guard)=>config.guardHandler(guard, request, api));
247
260
  // 파라미터 정보로 zod 스키마 빌드
261
+ const { getZodObjectFromApi } = await import("./code-converters.js");
248
262
  const ReqType = getZodObjectFromApi(api, this.syncer.types);
249
263
  // request 파싱
250
264
  const which = api.options.httpMethod === "GET" ? "query" : "body";
251
265
  let reqBody;
252
266
  try {
267
+ const { fastifyCaster } = await import("./caster.js");
253
268
  reqBody = fastifyCaster(ReqType).parse(request[which] ?? {});
254
269
  } catch (e) {
270
+ const { ZodError } = await import("zod");
255
271
  if (e instanceof ZodError) {
272
+ const { humanizeZodError } = await import("../utils/zod-error.js");
256
273
  const messages = humanizeZodError(e).map((issue)=>issue.message).join(" ");
274
+ const { BadRequestException } = await import("../exceptions/so-exceptions.js");
257
275
  throw new BadRequestException(messages, {
258
276
  zodError: e
259
277
  });
@@ -299,6 +317,7 @@ class SonamuClass {
299
317
  return cachedData;
300
318
  }
301
319
  // createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.
320
+ const { createSSEFactory } = await import("../stream/sse.js");
302
321
  const createSSE = ((_request, _reply, _events)=>createSSEFactory(_request.socket, _reply, _events)).bind(null, request, reply);
303
322
  const context = {
304
323
  ...await Promise.resolve(config.contextProvider({
@@ -306,7 +325,7 @@ class SonamuClass {
306
325
  reply,
307
326
  headers: request.headers,
308
327
  createSSE,
309
- naiteStore: new Map(),
328
+ naiteStore: Naite.createStore(),
310
329
  // auth
311
330
  user: request.user ?? null,
312
331
  passport: {
@@ -319,6 +338,8 @@ class SonamuClass {
319
338
  return this.asyncLocalStorage.run({
320
339
  context
321
340
  }, async ()=>{
341
+ const { ApiParamType } = await import("../types/types.js");
342
+ // biome-ignore lint/suspicious/noExplicitAny: model은 모델 인스턴스이므로 메서드 호출 가능
322
343
  const result = await model[api.methodName].apply(model, api.parameters.map((param)=>{
323
344
  // Context 인젝션
324
345
  if (ApiParamType.isContext(param.type)) {
@@ -336,11 +357,12 @@ class SonamuClass {
336
357
  });
337
358
  };
338
359
  }
339
- startWatcher() {
360
+ async startWatcher() {
340
361
  const watchPath = [
341
362
  path.join(this.apiRootPath, "src"),
342
363
  path.join(this.apiRootPath, "sonamu.config.ts")
343
364
  ];
365
+ const chokidar = (await import("chokidar")).default;
344
366
  this.watcher = chokidar.watch(watchPath, {
345
367
  ignored: (path, stats)=>!!stats?.isFile() && !path.endsWith(".ts") && !path.endsWith(".json"),
346
368
  persistent: true,
@@ -357,6 +379,7 @@ class SonamuClass {
357
379
  const isConfigTs = filePath === path.join(this.apiRootPath, "sonamu.config.ts");
358
380
  if (isConfigTs) {
359
381
  const relativePath = filePath.replace(this.apiRootPath, "api");
382
+ const chalk = (await import("chalk")).default;
360
383
  console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`));
361
384
  process.kill(process.pid, "SIGUSR2");
362
385
  return;
@@ -377,7 +400,7 @@ class SonamuClass {
377
400
  await this.destroy();
378
401
  }
379
402
  }
380
- registerPlugins(server, plugins) {
403
+ async registerPlugins(server, plugins) {
381
404
  if (!plugins) {
382
405
  return;
383
406
  }
@@ -390,23 +413,25 @@ class SonamuClass {
390
413
  static: "@fastify/static",
391
414
  session: "@fastify/secure-session"
392
415
  };
393
- const registerPlugin = (key, pluginName)=>{
416
+ const registerPlugin = async (key, pluginName)=>{
394
417
  const option = plugins[key];
395
418
  if (!option) return;
396
419
  if (option === true) {
397
- server.register(import(pluginName));
420
+ server.register((await import(pluginName)).default);
398
421
  } else {
399
- server.register(import(pluginName), option);
422
+ server.register((await import(pluginName)).default, option);
400
423
  }
401
424
  };
402
- Object.entries(pluginsModules).forEach(([key, pluginName])=>{
403
- registerPlugin(key, pluginName);
404
- });
425
+ for (const [key, pluginName] of Object.entries(pluginsModules)){
426
+ await registerPlugin(key, pluginName);
427
+ }
405
428
  if (plugins.custom) {
406
429
  plugins.custom(server);
407
430
  }
408
431
  }
409
432
  async registerAuth(server, options) {
433
+ // await import("fastify");
434
+ const fastifyPassport = (await import("@fastify/passport")).default;
410
435
  server.register(fastifyPassport.initialize());
411
436
  server.register(fastifyPassport.secureSession());
412
437
  if (typeof options === "boolean") {
@@ -444,6 +469,7 @@ class SonamuClass {
444
469
  }).then(async ()=>{
445
470
  await options.lifecycle?.onStart?.(server);
446
471
  }).catch(async (err)=>{
472
+ const chalk = (await import("chalk")).default;
447
473
  console.error(chalk.red("Failed to start server:", err));
448
474
  await shutdown();
449
475
  });
@@ -455,6 +481,7 @@ class SonamuClass {
455
481
  }
456
482
  this.pendingFiles.push(filePath);
457
483
  const relativePath = path.relative(this.apiRootPath, filePath);
484
+ const chalk = (await import("chalk")).default;
458
485
  console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));
459
486
  await this.syncer.syncFromWatcher(event, filePath);
460
487
  // 처리 완료된 파일을 대기 목록에서 제거
@@ -468,10 +495,15 @@ class SonamuClass {
468
495
  await this.syncer.renewChecksums();
469
496
  const endTime = Date.now();
470
497
  const totalTime = endTime - this.hmrStartTime;
498
+ const [chalk, { centerText }] = await Promise.all([
499
+ (await import("chalk")).default,
500
+ import("../utils/console-util.js")
501
+ ]);
471
502
  const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;
472
503
  console.log(chalk.black.bgGreen(centerText(msg)));
473
504
  }
474
505
  async destroy() {
506
+ const { BaseModel } = await import("../database/base-model.js");
475
507
  await BaseModel.destroy();
476
508
  await this.watcher?.close();
477
509
  this.storage?.destroy();
@@ -479,4 +511,4 @@ class SonamuClass {
479
511
  }
480
512
  export const Sonamu = new SonamuClass();
481
513
 
482
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/api/sonamu.ts"],"sourcesContent":["import { AsyncLocalStorage } from \"async_hooks\";\nimport chalk from \"chalk\";\nimport fastify from \"fastify\";\nimport { readFile } from \"fs/promises\";\nimport path from \"path\";\nimport { exists } from \"../utils/fs-utils\";\nimport chokidar, { type FSWatcher } from \"chokidar\";\nimport { formatInTimeZone } from \"date-fns-tz\";\nimport type { FastifyInstance, FastifyReply, FastifyRequest } from \"fastify\";\nimport type { IncomingMessage, Server, ServerResponse } from \"http\";\nimport { ZodError, ZodObject } from \"zod\";\nimport { DB, SonamuDBConfig } from \"../database/db\";\nimport { attachOnDuplicateUpdate } from \"../database/knex-plugins/knex-on-duplicate-update\";\nimport {\n  BadRequestException,\n  NotFoundException,\n} from \"../exceptions/so-exceptions\";\nimport type { Driver } from \"../file-storage/driver\";\nimport { createSSEFactory } from \"../stream/sse\";\nimport type { Syncer } from \"../syncer/syncer\";\nimport { ApiParamType, SonamuFastifyConfig } from \"../types/types\";\nimport { isLocal, isTest } from \"../utils/controller\";\nimport { findApiRootPath } from \"../utils/utils\";\nimport { humanizeZodError } from \"../utils/zod-error\";\nimport { fastifyCaster } from \"./caster\";\nimport { getZodObjectFromApi } from \"./code-converters\";\nimport type { AuthContext, Context, UploadContext } from \"./context\";\nimport type { ExtendedApi } from \"./decorators\";\nimport fastifyPassport from \"@fastify/passport\";\nimport { loadConfig, SonamuConfig, SonamuServerOptions } from \"./config\";\nimport { AbsolutePath } from \"../utils/path-utils\";\nimport { isHotReloadServer } from \"../utils/esm-utils\";\nimport { Template } from \"../template\";\nimport assert from \"assert\";\nimport { centerText } from \"../utils/console-util\";\nimport { BaseModel } from \"../database/base-model\";\n\nexport type SonamuSecrets = {\n  [key: string]: string;\n};\nclass SonamuClass {\n  public isInitialized: boolean = false;\n  public asyncLocalStorage: AsyncLocalStorage<{\n    context: Context;\n  }> = new AsyncLocalStorage();\n\n  public uploadStorage: AsyncLocalStorage<{\n    uploadContext: UploadContext;\n  }> = new AsyncLocalStorage();\n\n  public getContext(): Context {\n    const store = this.asyncLocalStorage.getStore();\n    if (store?.context) {\n      return store.context;\n    }\n\n    if (process.env.NODE_ENV === \"test\") {\n      // 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n      return {\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: () => {},\n        naiteStore: new Map<string, any>(),\n      } as unknown as Context;\n    } else {\n      throw new Error(\"Sonamu cannot find context\");\n    }\n  }\n\n  public getUploadContext(): UploadContext {\n    const store = this.uploadStorage.getStore();\n    if (store?.uploadContext) {\n      return store.uploadContext;\n    }\n    throw new Error(\n      \"Sonamu cannot find upload context. Did you use @upload decorator?\"\n    );\n  }\n\n  private _apiRootPath: AbsolutePath | null = null;\n  set apiRootPath(apiRootPath: AbsolutePath) {\n    this._apiRootPath = apiRootPath;\n  }\n  get apiRootPath(): AbsolutePath {\n    if (this._apiRootPath === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._apiRootPath!;\n  }\n  get appRootPath(): string {\n    return this.apiRootPath.split(path.sep).slice(0, -1).join(path.sep);\n  }\n\n  private _dbConfig: SonamuDBConfig | null = null;\n  set dbConfig(dbConfig: SonamuDBConfig) {\n    this._dbConfig = dbConfig;\n  }\n  get dbConfig(): SonamuDBConfig {\n    if (this._dbConfig === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._dbConfig!;\n  }\n\n  private _syncer: Syncer | null = null;\n  set syncer(syncer: Syncer) {\n    this._syncer = syncer;\n  }\n  get syncer(): Syncer {\n    if (this._syncer === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._syncer!;\n  }\n\n  private _config: SonamuConfig | null = null;\n  set config(config: SonamuConfig) {\n    this._config = config;\n  }\n  get config(): SonamuConfig {\n    if (this._config === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._config;\n  }\n\n  private _secrets: SonamuSecrets | null = null;\n  set secrets(secrets: SonamuSecrets) {\n    this._secrets = secrets;\n  }\n  get secrets(): SonamuSecrets | null {\n    return this._secrets;\n  }\n\n  private _storage: Driver | null = null;\n  set storage(storage: Driver) {\n    this._storage = storage;\n  }\n  get storage(): Driver | null {\n    return this._storage;\n  }\n\n  // HMR 처리\n  public watcher: FSWatcher | null = null;\n  private pendingFiles: string[] = [];\n  private hmrStartTime: number = 0;\n\n  public server: FastifyInstance | null = null;\n\n  async initForTesting() {\n    await this.init(true, false, undefined, true);\n  }\n\n  async init(\n    doSilent: boolean = false,\n    enableSync: boolean = true,\n    apiRootPath?: AbsolutePath,\n    forTesting: boolean = false\n  ) {\n    if (this.isInitialized) {\n      return;\n    }\n    !doSilent &&\n      console.time(\n        chalk.cyan(`Sonamu.init${forTesting ? \" for testing\" : \"\"}`)\n      );\n\n    // API 루트 패스\n    this.apiRootPath = apiRootPath ?? findApiRootPath();\n    this.config = await loadConfig(this.apiRootPath);\n    const secretsPath = path.join(this.apiRootPath, \"sonamu.secrets.json\");\n    if (await exists(secretsPath)) {\n      this.secrets = JSON.parse(\n        (await readFile(secretsPath)).toString()\n      ) as SonamuSecrets;\n    }\n\n    // DB 로드\n    this.dbConfig = DB.generateDBConfig(this.config.database);\n    !doSilent && console.log(chalk.green(\"DB Config Loaded!\"));\n    attachOnDuplicateUpdate();\n\n    // 테스팅인 경우 엔티티 로드 & 싱크 없이 중단\n    if (forTesting) {\n      this.isInitialized = true;\n      return;\n    }\n\n    // Entity 로드\n    const { EntityManager } = await import(\"../entity/entity-manager\");\n    await EntityManager.autoload(doSilent);\n\n    // Syncer\n    const { Syncer } = await import(\"../syncer/syncer\");\n    this.syncer = new Syncer();\n\n    // Autoload: Models / Types / APIs\n    await this.syncer.autoloadTypes();\n    await this.syncer.autoloadModels();\n    await this.syncer.autoloadApis();\n\n    await Template.autoload();\n\n    if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {\n      await this.syncer.sync();\n\n      this.startWatcher();\n\n      this.syncer.syncUI();\n    }\n\n    this.isInitialized = true;\n    !doSilent && console.timeEnd(chalk.cyan(\"Sonamu.init\"));\n  }\n\n  async createServer(initOptions?: {\n    enableSync?: boolean;\n    doSilent?: boolean;\n  }) {\n    if (this.isInitialized === false) {\n      await this.init(initOptions?.doSilent, initOptions?.enableSync);\n    }\n\n    const options = this.config.server;\n    const server = fastify(options.fastify);\n    this.server = server;\n\n    // Storage 설정 저장\n    if (options.storage) {\n      this.storage = options.storage;\n    }\n\n    // 플러그인 등록\n    if (options.plugins) {\n      this.registerPlugins(server, options.plugins);\n    }\n\n    if (options.auth) {\n      if (!options.plugins?.session) {\n        throw new Error(\n          \"Auth requires session plugin. Please add plugins.session configuration.\"\n        );\n      }\n\n      this.registerAuth(server, options.auth);\n    }\n\n    // API 라우팅 설정\n    await this.withFastify(server, options.apiConfig, {\n      enableSync: initOptions?.enableSync,\n      doSilent: initOptions?.doSilent,\n    });\n\n    // 서버 시작\n    await this.boot(server, options);\n\n    return server;\n  }\n\n  async withFastify(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    options?: {\n      enableSync?: boolean;\n      doSilent?: boolean;\n    }\n  ) {\n    if (this.isInitialized === false) {\n      await this.init(options?.doSilent, options?.enableSync);\n    }\n\n    this.server = server;\n\n    // timezone 설정\n    const timezone = this.config.api.timezone;\n    if (timezone) {\n      const DATE_FORMAT = \"yyyy-MM-dd'T'HH:mm:ssXXX\";\n      // ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)\n      const ISO_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(\\.\\d{3})?Z$/;\n\n      server.setReplySerializer((payload) => {\n        return JSON.stringify(payload, (_key, value) => {\n          if (typeof value === \"string\" && ISO_DATE_REGEX.test(value)) {\n            return formatInTimeZone(new Date(value), timezone, DATE_FORMAT);\n          }\n          return value;\n        });\n      });\n      !options?.doSilent &&\n        console.log(chalk.green(`Timezone set to ${timezone}`));\n    }\n\n    // 전체 라우팅 리스트\n    server.get(\n      `${this.config.api.route.prefix}/routes`,\n      async (_request, _reply): Promise<any> => {\n        return this.syncer.apis;\n      }\n    );\n\n    // Healthcheck API\n    server.get(\n      `${this.config.api.route.prefix}/healthcheck`,\n      async (_request, _reply): Promise<string> => {\n        return \"ok\";\n      }\n    );\n\n    // API 라우팅 (로컬HMR 상태와 구분)\n    if (isLocal()) {\n      server.all(\"*\", (request, reply) => {\n        const found = this.syncer.apis.find(\n          (api) =>\n            this.config.api.route.prefix + api.path ===\n              request.url.split(\"?\")[0] &&\n            (api.options.httpMethod ?? \"GET\") === request.method.toUpperCase()\n        );\n        if (found) {\n          return this.getApiHandler(found, config)(request, reply);\n        }\n        throw new NotFoundException(\"존재하지 않는 API 접근입니다.\");\n      });\n    } else {\n      this.syncer.apis.map((api) => {\n        // model\n        if (this.syncer.models[api.modelName] === undefined) {\n          throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);\n        }\n\n        // route\n        server.route({\n          method: api.options.httpMethod!,\n          url: this.config.api.route.prefix + api.path,\n          handler: this.getApiHandler(api, config),\n        }); // END server.route\n      });\n    }\n  }\n\n  getApiHandler(api: ExtendedApi, config: SonamuFastifyConfig) {\n    return async (\n      request: FastifyRequest,\n      reply: FastifyReply\n    ): Promise<unknown> => {\n      (api.options.guards ?? []).every((guard) =>\n        config.guardHandler(guard, request, api)\n      );\n\n      // 파라미터 정보로 zod 스키마 빌드\n      const ReqType = getZodObjectFromApi(api, this.syncer.types);\n\n      // request 파싱\n      const which = api.options.httpMethod === \"GET\" ? \"query\" : \"body\";\n      let reqBody: {\n        [key: string]: unknown;\n      };\n      try {\n        reqBody = fastifyCaster(ReqType).parse(request[which] ?? {});\n      } catch (e) {\n        if (e instanceof ZodError) {\n          const messages = humanizeZodError(e)\n            .map((issue) => issue.message)\n            .join(\" \");\n          throw new BadRequestException(messages, {\n            zodError: e,\n          });\n        } else {\n          throw e;\n        }\n      }\n\n      // Content-Type\n      reply.type(api.options.contentType ?? \"application/json\");\n\n      // 캐시\n      const { cacheKey, cacheTtl, cachedData } = await (async () => {\n        if (config.cache) {\n          try {\n            const cacheKeyRes = config.cache.resolveKey(api.path, reqBody);\n            if (cacheKeyRes.cache === false) {\n              return { cacheKey: null, cachedData: null };\n            }\n\n            const cacheKey = cacheKeyRes.key;\n            const cacheTtl = cacheKeyRes.ttl;\n            const cachedData = await config.cache.get(cacheKey);\n            return { cacheKey, cacheTtl, cachedData };\n          } catch (e) {\n            console.error(e);\n          }\n          return { cacheKey: null, cachedData: null };\n        }\n        return { cacheKey: null, cachedData: null };\n      })();\n      if (cachedData !== null) {\n        return cachedData;\n      }\n\n      // createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.\n      const createSSE = (<T extends ZodObject>(\n        _request: FastifyRequest,\n        _reply: FastifyReply,\n        _events: T\n      ) => createSSEFactory(_request.socket, _reply, _events)).bind(\n        null,\n        request,\n        reply\n      );\n\n      const context: Context = {\n        ...(await Promise.resolve(\n          config.contextProvider(\n            {\n              request,\n              reply,\n              headers: request.headers,\n              createSSE,\n              naiteStore: new Map<string, any>(),\n              // auth\n              user: request.user ?? null,\n              passport: {\n                login: request.login.bind(\n                  request\n                ) as AuthContext[\"passport\"][\"login\"],\n                logout: request.logout.bind(\n                  request\n                ) as AuthContext[\"passport\"][\"logout\"],\n              },\n            },\n            request,\n            reply\n          )\n        )),\n      };\n\n      const model = this.syncer.models[api.modelName];\n      return this.asyncLocalStorage.run({ context }, async () => {\n        const result = await (model as any)[api.methodName].apply(\n          model,\n          api.parameters.map((param) => {\n            // Context 인젝션\n            if (ApiParamType.isContext(param.type)) {\n              return context;\n            } else {\n              return reqBody[param.name];\n            }\n          })\n        );\n        reply.type(api.options.contentType ?? \"application/json\");\n\n        // 캐시 키 있는 경우 갱신 후 저장\n        if (config.cache && cacheKey) {\n          await config.cache.put(cacheKey, result, cacheTtl);\n        }\n        return result;\n      });\n    };\n  }\n\n  startWatcher(): void {\n    const watchPath = [\n      path.join(this.apiRootPath, \"src\"),\n      path.join(this.apiRootPath, \"sonamu.config.ts\"),\n    ];\n\n    this.watcher = chokidar.watch(watchPath, {\n      ignored: (path, stats) =>\n        !!stats?.isFile() && !path.endsWith(\".ts\") && !path.endsWith(\".json\"),\n      persistent: true,\n      ignoreInitial: true,\n    });\n\n    this.watcher.on(\"all\", async (event: string, filePath: string) => {\n      const absolutePath = filePath as AbsolutePath;\n      assert(\n        absolutePath.startsWith(this.apiRootPath),\n        \"File path is not within the API root path\"\n      );\n\n      if (event !== \"change\" && event !== \"add\") {\n        return;\n      }\n\n      try {\n        // sonamu.config.ts 변경 시 재시작\n        const isConfigTs =\n          filePath === path.join(this.apiRootPath, \"sonamu.config.ts\");\n\n        if (isConfigTs) {\n          const relativePath = filePath.replace(this.apiRootPath, \"api\");\n          console.log(\n            chalk.bold(\n              `Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`\n            )\n          );\n          process.kill(process.pid, \"SIGUSR2\");\n          return;\n        }\n\n        await this.handleFileChange(event, absolutePath);\n      } catch (e) {\n        console.error(e);\n      }\n    });\n  }\n\n  /*\n     A function that automatically handles init and destroy when using Sonamu via scripts.    \n  */\n  async runScript(fn: () => Promise<void>) {\n    await this.init(true, false, undefined, false);\n    try {\n      await fn();\n    } finally {\n      await this.destroy();\n    }\n  }\n\n  private registerPlugins(\n    server: FastifyInstance,\n    plugins: SonamuServerOptions[\"plugins\"]\n  ) {\n    if (!plugins) {\n      return;\n    }\n\n    const pluginsModules = {\n      cors: \"@fastify/cors\",\n      formbody: \"@fastify/formbody\",\n      multipart: \"@fastify/multipart\",\n      qs: \"fastify-qs\",\n      sse: \"fastify-sse-v2\",\n      static: \"@fastify/static\",\n      session: \"@fastify/secure-session\",\n    } as const;\n\n    const registerPlugin = <K extends keyof NonNullable<typeof plugins>>(\n      key: K,\n      pluginName: string\n    ) => {\n      const option = plugins[key];\n      if (!option) return;\n\n      if (option === true) {\n        server.register(import(pluginName));\n      } else {\n        server.register(import(pluginName), option);\n      }\n    };\n\n    Object.entries(pluginsModules).forEach(([key, pluginName]) => {\n      registerPlugin(key as keyof typeof plugins, pluginName);\n    });\n\n    if (plugins.custom) {\n      plugins.custom(server);\n    }\n  }\n\n  private async registerAuth(\n    server: FastifyInstance,\n    options: NonNullable<SonamuServerOptions[\"auth\"]>\n  ) {\n    server.register(fastifyPassport.initialize());\n    server.register(fastifyPassport.secureSession());\n\n    if (typeof options === \"boolean\") {\n      fastifyPassport.registerUserSerializer(async (user, _request) => user);\n      fastifyPassport.registerUserDeserializer(\n        async (serialized, _request) => serialized\n      );\n    } else {\n      fastifyPassport.registerUserSerializer(options.userSerializer);\n      fastifyPassport.registerUserDeserializer(options.userDeserializer);\n    }\n  }\n\n  private async boot(server: FastifyInstance, options: SonamuServerOptions) {\n    const port = options.listen?.port ?? 3000;\n    const host = options.listen?.host ?? \"localhost\";\n\n    server.addHook(\"onClose\", async () => {\n      await options.lifecycle?.onShutdown?.(server);\n      await this.destroy();\n    });\n\n    const shutdown = async () => {\n      try {\n        await server.close();\n        process.exit(0);\n      } catch (err) {\n        console.error(\"Error during shutdown:\", err);\n        process.exit(1);\n      }\n    };\n\n    process.on(\"SIGINT\", shutdown);\n    process.on(\"SIGTERM\", shutdown);\n\n    if (options.lifecycle?.onError) {\n      server.setErrorHandler(options.lifecycle?.onError);\n    }\n\n    server\n      .listen({ port, host })\n      .then(async () => {\n        await options.lifecycle?.onStart?.(server);\n      })\n      .catch(async (err) => {\n        console.error(chalk.red(\"Failed to start server:\", err));\n        await shutdown();\n      });\n  }\n\n  private async handleFileChange(\n    event: string,\n    filePath: AbsolutePath\n  ): Promise<void> {\n    // 첫 번째 파일이면 HMR 시작 시간 기록\n    if (this.pendingFiles.length === 0) {\n      this.hmrStartTime = Date.now();\n    }\n    this.pendingFiles.push(filePath);\n\n    const relativePath = path.relative(this.apiRootPath, filePath);\n    console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));\n\n    await this.syncer.syncFromWatcher(event, filePath);\n\n    // 처리 완료된 파일을 대기 목록에서 제거\n    this.pendingFiles = this.pendingFiles.slice(1);\n\n    // 모든 파일 처리가 완료되면 최종 메시지 출력\n    if (this.pendingFiles.length === 0) {\n      await this.finishHMR();\n    }\n  }\n\n  private async finishHMR(): Promise<void> {\n    await this.syncer.renewChecksums();\n\n    const endTime = Date.now();\n    const totalTime = endTime - this.hmrStartTime;\n    const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;\n\n    console.log(chalk.black.bgGreen(centerText(msg)));\n  }\n\n  async destroy(): Promise<void> {\n    await BaseModel.destroy();\n    await this.watcher?.close();\n    this.storage?.destroy();\n  }\n}\nexport const Sonamu = new SonamuClass();\n"],"names":["AsyncLocalStorage","chalk","fastify","readFile","path","exists","chokidar","formatInTimeZone","ZodError","DB","attachOnDuplicateUpdate","BadRequestException","NotFoundException","createSSEFactory","ApiParamType","isLocal","isTest","findApiRootPath","humanizeZodError","fastifyCaster","getZodObjectFromApi","fastifyPassport","loadConfig","isHotReloadServer","Template","assert","centerText","BaseModel","SonamuClass","isInitialized","asyncLocalStorage","uploadStorage","getContext","store","getStore","context","process","env","NODE_ENV","request","reply","headers","createSSE","naiteStore","Map","Error","getUploadContext","uploadContext","_apiRootPath","apiRootPath","appRootPath","split","sep","slice","join","_dbConfig","dbConfig","_syncer","syncer","_config","config","_secrets","secrets","_storage","storage","watcher","pendingFiles","hmrStartTime","server","initForTesting","init","undefined","doSilent","enableSync","forTesting","console","time","cyan","secretsPath","JSON","parse","toString","generateDBConfig","database","log","green","EntityManager","autoload","Syncer","autoloadTypes","autoloadModels","autoloadApis","sync","startWatcher","syncUI","timeEnd","createServer","initOptions","options","plugins","registerPlugins","auth","session","registerAuth","withFastify","apiConfig","boot","timezone","api","DATE_FORMAT","ISO_DATE_REGEX","setReplySerializer","payload","stringify","_key","value","test","Date","get","route","prefix","_request","_reply","apis","all","found","find","url","httpMethod","method","toUpperCase","getApiHandler","map","models","modelName","handler","guards","every","guard","guardHandler","ReqType","types","which","reqBody","e","messages","issue","message","zodError","type","contentType","cacheKey","cacheTtl","cachedData","cache","cacheKeyRes","resolveKey","key","ttl","error","_events","socket","bind","Promise","resolve","contextProvider","user","passport","login","logout","model","run","result","methodName","apply","parameters","param","isContext","name","put","watchPath","watch","ignored","stats","isFile","endsWith","persistent","ignoreInitial","on","event","filePath","absolutePath","startsWith","isConfigTs","relativePath","replace","bold","blue","kill","pid","handleFileChange","runScript","fn","destroy","pluginsModules","cors","formbody","multipart","qs","sse","static","registerPlugin","pluginName","option","register","Object","entries","forEach","custom","initialize","secureSession","registerUserSerializer","registerUserDeserializer","serialized","userSerializer","userDeserializer","port","listen","host","addHook","lifecycle","onShutdown","shutdown","close","exit","err","onError","setErrorHandler","then","onStart","catch","red","length","now","push","relative","syncFromWatcher","finishHMR","renewChecksums","endTime","totalTime","msg","white","black","bgGreen","Sonamu"],"mappings":"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"}
514
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/api/sonamu.ts"],"sourcesContent":["import assert from \"assert\";\nimport { AsyncLocalStorage } from \"async_hooks\";\nimport type { FSWatcher } from \"chokidar\";\nimport type { FastifyInstance, FastifyReply, FastifyRequest } from \"fastify\";\nimport type { IncomingMessage, Server, ServerResponse } from \"http\";\nimport path from \"path\";\nimport type { ZodObject } from \"zod\";\nimport type { SonamuDBConfig } from \"../database/db\";\nimport type { Driver } from \"../file-storage/driver\";\nimport { Naite } from \"../naite/naite\";\nimport type { Syncer } from \"../syncer/syncer\";\nimport type { SonamuFastifyConfig } from \"../types/types\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\nimport type { SonamuConfig, SonamuServerOptions } from \"./config\";\nimport type { AuthContext, Context, UploadContext } from \"./context\";\nimport type { ExtendedApi } from \"./decorators\";\n\nexport type SonamuSecrets = {\n  anthropic_api_key?: string;\n};\nclass SonamuClass {\n  public isInitialized: boolean = false;\n  public asyncLocalStorage: AsyncLocalStorage<{\n    context: Context;\n  }> = new AsyncLocalStorage();\n\n  public uploadStorage: AsyncLocalStorage<{\n    uploadContext: UploadContext;\n  }> = new AsyncLocalStorage();\n\n  public getContext(): Context {\n    const store = this.asyncLocalStorage.getStore();\n    if (store?.context) {\n      return store.context;\n    }\n\n    if (process.env.NODE_ENV === \"test\") {\n      // 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n      return {\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: () => {},\n        // biome-ignore lint/suspicious/noExplicitAny: 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n        naiteStore: new Map<string, any>(),\n      } as unknown as Context;\n    } else {\n      throw new Error(\"Sonamu cannot find context\");\n    }\n  }\n\n  public getUploadContext(): UploadContext {\n    const store = this.uploadStorage.getStore();\n    if (store?.uploadContext) {\n      return store.uploadContext;\n    }\n    throw new Error(\"Sonamu cannot find upload context. Did you use @upload decorator?\");\n  }\n\n  private _apiRootPath: AbsolutePath | null = null;\n  set apiRootPath(apiRootPath: AbsolutePath) {\n    this._apiRootPath = apiRootPath;\n  }\n  get apiRootPath(): AbsolutePath {\n    if (this._apiRootPath === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._apiRootPath;\n  }\n  get appRootPath(): string {\n    return this.apiRootPath.split(path.sep).slice(0, -1).join(path.sep);\n  }\n\n  private _dbConfig: SonamuDBConfig | null = null;\n  set dbConfig(dbConfig: SonamuDBConfig) {\n    this._dbConfig = dbConfig;\n  }\n  get dbConfig(): SonamuDBConfig {\n    if (this._dbConfig === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._dbConfig;\n  }\n\n  private _syncer: Syncer | null = null;\n  set syncer(syncer: Syncer) {\n    this._syncer = syncer;\n  }\n  get syncer(): Syncer {\n    if (this._syncer === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._syncer;\n  }\n\n  private _config: SonamuConfig | null = null;\n  set config(config: SonamuConfig) {\n    this._config = config;\n  }\n  get config(): SonamuConfig {\n    if (this._config === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._config;\n  }\n\n  private _secrets: SonamuSecrets | null = null;\n  set secrets(secrets: SonamuSecrets) {\n    this._secrets = secrets;\n  }\n  get secrets(): SonamuSecrets | null {\n    return this._secrets;\n  }\n\n  private _storage: Driver | null = null;\n  set storage(storage: Driver) {\n    this._storage = storage;\n  }\n  get storage(): Driver | null {\n    return this._storage;\n  }\n\n  // HMR 처리\n  public watcher: FSWatcher | null = null;\n  private pendingFiles: string[] = [];\n  private hmrStartTime: number = 0;\n\n  public server: FastifyInstance | null = null;\n\n  async initForTesting() {\n    await this.init(true, false, undefined, true);\n  }\n\n  async init(\n    doSilent: boolean = false,\n    enableSync: boolean = true,\n    apiRootPath?: AbsolutePath,\n    forTesting: boolean = false,\n  ) {\n    if (this.isInitialized) {\n      return;\n    }\n\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.time(chalk.cyan(`Sonamu.init${forTesting ? \" for testing\" : \"\"}`));\n    }\n\n    // API 루트 패스\n    const { findApiRootPath } = await import(\"../utils/utils\");\n    this.apiRootPath = apiRootPath ?? findApiRootPath();\n\n    const { loadConfig } = await import(\"./config\");\n    this.config = await loadConfig(this.apiRootPath);\n    // sonamu.config.ts 기본값 설정\n    this.config.database.database = this.config.database.database ?? \"postgresql\";\n\n    if (process.env.ANTHROPIC_API_KEY) {\n      this.secrets = {\n        anthropic_api_key: process.env.ANTHROPIC_API_KEY,\n      };\n    }\n\n    // DB 로드\n    const { DB } = await import(\"../database/db\");\n    this.dbConfig = DB.generateDBConfig(this.config.database);\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.log(chalk.green(\"DB Config Loaded!\"));\n    }\n\n    // 테스팅인 경우 엔티티 로드 & 싱크 없이 중단\n    if (forTesting) {\n      this.isInitialized = true;\n      return;\n    }\n\n    // Entity 로드\n    const { EntityManager } = await import(\"../entity/entity-manager\");\n    await EntityManager.autoload(doSilent);\n\n    // Syncer\n    const { Syncer } = await import(\"../syncer/syncer\");\n    this.syncer = new Syncer();\n\n    // Autoload: Models / Types / APIs\n    await this.syncer.autoloadTypes();\n    await this.syncer.autoloadModels();\n    await this.syncer.autoloadApis();\n\n    const { TemplateManager } = await import(\"../template\");\n    await TemplateManager.autoload();\n\n    const { isLocal, isTest } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      // 로컬에서는 코드 생성을 위해 Biome 셋업이 필요함 (현재 apiRootPath 전달하여 실행)\n      (await import(\"../utils/formatter\")).setupBiome(this.apiRootPath);\n    }\n\n    const { isHotReloadServer } = await import(\"../utils/controller\");\n    if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {\n      await this.syncer.sync();\n\n      await this.startWatcher();\n\n      this.syncer.syncUI();\n    }\n\n    this.isInitialized = true;\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.timeEnd(chalk.cyan(\"Sonamu.init\"));\n    }\n  }\n\n  async createServer(initOptions?: { enableSync?: boolean; doSilent?: boolean }) {\n    if (this.isInitialized === false) {\n      await this.init(initOptions?.doSilent, initOptions?.enableSync);\n    }\n\n    const options = this.config.server;\n    const fastify = (await import(\"fastify\")).default;\n    const server = fastify(options.fastify);\n    this.server = server;\n\n    // Storage 설정 저장\n    if (options.storage) {\n      this.storage = options.storage;\n    }\n\n    // 플러그인 등록\n    if (options.plugins) {\n      await this.registerPlugins(server, options.plugins);\n    }\n\n    if (options.auth) {\n      if (!options.plugins?.session) {\n        throw new Error(\"Auth requires session plugin. Please add plugins.session configuration.\");\n      }\n\n      await this.registerAuth(server, options.auth);\n    }\n\n    // API 라우팅 설정\n    await this.withFastify(server, options.apiConfig, {\n      enableSync: initOptions?.enableSync,\n      doSilent: initOptions?.doSilent,\n    });\n\n    // 서버 시작\n    await this.boot(server, options);\n\n    return server;\n  }\n\n  async withFastify(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    options?: {\n      enableSync?: boolean;\n      doSilent?: boolean;\n    },\n  ) {\n    if (this.isInitialized === false) {\n      await this.init(options?.doSilent, options?.enableSync);\n    }\n\n    this.server = server;\n\n    // timezone 설정\n    const timezone = this.config.api.timezone;\n    if (timezone) {\n      // 타임존에 맞게 응답 날짜 스트링을 변환해주어야 합니다.\n      // 가령 timezone이 \"Asia/Seoul\" 이면\n      // \"2025-11-21T00:00:00.000Z\" 를 \"2025-11-21T09:00:00+09:00\" 으로 변환해주어야 합니다.\n      const { formatInTimeZone } = await import(\"date-fns-tz\");\n\n      // ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)\n      const ISO_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(\\.\\d{3})?Z$/;\n\n      // T를 둘러싼 작은따옴표가 없다면 \"2025-11-19176354618900018:56:29+09:00\"와 같은 결과가 나옵니다.\n      // 이는 date-fns 특입니다.\n      // 이렇게 해도 괜찮습니다. \"2025-11-19T18:56:29+09:00\" 모양으로 잘 나옵니다.\n      const DATE_FORMAT = \"yyyy-MM-dd'T'HH:mm:ssXXX\";\n\n      server.setReplySerializer((payload) => {\n        return JSON.stringify(payload, (_key, value) => {\n          if (typeof value === \"string\" && ISO_DATE_REGEX.test(value)) {\n            return formatInTimeZone(\n              new Date(value),\n              timezone as `${string}/${string}`,\n              DATE_FORMAT,\n            );\n          }\n          return value;\n        });\n      });\n      if (!options?.doSilent) {\n        const chalk = (await import(\"chalk\")).default;\n        console.log(chalk.green(`Timezone set to ${timezone}`));\n      }\n    }\n\n    // 전체 라우팅 리스트\n    server.get(\n      `${this.config.api.route.prefix}/routes`,\n      async (_request, _reply): Promise<typeof this.syncer.apis> => {\n        return this.syncer.apis;\n      },\n    );\n\n    // Healthcheck API\n    server.get(\n      `${this.config.api.route.prefix}/healthcheck`,\n      async (_request, _reply): Promise<string> => {\n        return \"ok\";\n      },\n    );\n\n    // API 라우팅 (로컬HMR 상태와 구분)\n    const { isLocal } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      server.all(\"*\", async (request, reply) => {\n        const found = this.syncer.apis.find(\n          (api) =>\n            this.config.api.route.prefix + api.path === request.url.split(\"?\")[0] &&\n            (api.options.httpMethod ?? \"GET\") === request.method.toUpperCase(),\n        );\n        if (found) {\n          return this.getApiHandler(found, config)(request, reply);\n        }\n        const { NotFoundException } = await import(\"../exceptions/so-exceptions\");\n        throw new NotFoundException(\"존재하지 않는 API 접근입니다.\");\n      });\n    } else {\n      for (const api of this.syncer.apis) {\n        // model\n        if (this.syncer.models[api.modelName] === undefined) {\n          throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);\n        }\n\n        // route\n        server.route({\n          method: api.options.httpMethod ?? \"GET\",\n          url: this.config.api.route.prefix + api.path,\n          handler: this.getApiHandler(api, config),\n        }); // END server.route\n      }\n    }\n  }\n\n  getApiHandler(\n    api: ExtendedApi,\n    config: SonamuFastifyConfig,\n  ): (request: FastifyRequest, reply: FastifyReply) => Promise<unknown> {\n    return async (request: FastifyRequest, reply: FastifyReply): Promise<unknown> => {\n      (api.options.guards ?? []).every((guard) => config.guardHandler(guard, request, api));\n\n      // 파라미터 정보로 zod 스키마 빌드\n      const { getZodObjectFromApi } = await import(\"./code-converters\");\n      const ReqType = getZodObjectFromApi(api, this.syncer.types);\n\n      // request 파싱\n      const which = api.options.httpMethod === \"GET\" ? \"query\" : \"body\";\n      let reqBody: {\n        [key: string]: unknown;\n      };\n      try {\n        const { fastifyCaster } = await import(\"./caster\");\n        reqBody = fastifyCaster(ReqType).parse(request[which] ?? {});\n      } catch (e) {\n        const { ZodError } = await import(\"zod\");\n        if (e instanceof ZodError) {\n          const { humanizeZodError } = await import(\"../utils/zod-error\");\n          const messages = humanizeZodError(e)\n            .map((issue) => issue.message)\n            .join(\" \");\n          const { BadRequestException } = await import(\"../exceptions/so-exceptions\");\n          throw new BadRequestException(messages, {\n            zodError: e,\n          });\n        } else {\n          throw e;\n        }\n      }\n\n      // Content-Type\n      reply.type(api.options.contentType ?? \"application/json\");\n\n      // 캐시\n      const { cacheKey, cacheTtl, cachedData } = await (async () => {\n        if (config.cache) {\n          try {\n            const cacheKeyRes = config.cache.resolveKey(api.path, reqBody);\n            if (cacheKeyRes.cache === false) {\n              return { cacheKey: null, cachedData: null };\n            }\n\n            const cacheKey = cacheKeyRes.key;\n            const cacheTtl = cacheKeyRes.ttl;\n            const cachedData = await config.cache.get(cacheKey);\n            return { cacheKey, cacheTtl, cachedData };\n          } catch (e) {\n            console.error(e);\n          }\n          return { cacheKey: null, cachedData: null };\n        }\n        return { cacheKey: null, cachedData: null };\n      })();\n      if (cachedData !== null) {\n        return cachedData;\n      }\n\n      // createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.\n      const { createSSEFactory } = await import(\"../stream/sse\");\n      const createSSE = (<T extends ZodObject>(\n        _request: FastifyRequest,\n        _reply: FastifyReply,\n        _events: T,\n      ) => createSSEFactory(_request.socket, _reply, _events)).bind(null, request, reply);\n\n      const context: Context = {\n        ...(await Promise.resolve(\n          config.contextProvider(\n            {\n              request,\n              reply,\n              headers: request.headers,\n              createSSE,\n              naiteStore: Naite.createStore(),\n              // auth\n              user: request.user ?? null,\n              passport: {\n                login: request.login.bind(request) as AuthContext[\"passport\"][\"login\"],\n                logout: request.logout.bind(request) as AuthContext[\"passport\"][\"logout\"],\n              },\n            },\n            request,\n            reply,\n          ),\n        )),\n      };\n\n      const model = this.syncer.models[api.modelName];\n      return this.asyncLocalStorage.run({ context }, async () => {\n        const { ApiParamType } = await import(\"../types/types\");\n        // biome-ignore lint/suspicious/noExplicitAny: model은 모델 인스턴스이므로 메서드 호출 가능\n        const result = await (model as any)[api.methodName].apply(\n          model,\n          api.parameters.map((param) => {\n            // Context 인젝션\n            if (ApiParamType.isContext(param.type)) {\n              return context;\n            } else {\n              return reqBody[param.name];\n            }\n          }),\n        );\n        reply.type(api.options.contentType ?? \"application/json\");\n\n        // 캐시 키 있는 경우 갱신 후 저장\n        if (config.cache && cacheKey) {\n          await config.cache.put(cacheKey, result, cacheTtl);\n        }\n        return result;\n      });\n    };\n  }\n\n  async startWatcher(): Promise<void> {\n    const watchPath = [\n      path.join(this.apiRootPath, \"src\"),\n      path.join(this.apiRootPath, \"sonamu.config.ts\"),\n    ];\n\n    const chokidar = (await import(\"chokidar\")).default;\n    this.watcher = chokidar.watch(watchPath, {\n      ignored: (path, stats) =>\n        !!stats?.isFile() && !path.endsWith(\".ts\") && !path.endsWith(\".json\"),\n      persistent: true,\n      ignoreInitial: true,\n    });\n\n    this.watcher.on(\"all\", async (event: string, filePath: string) => {\n      const absolutePath = filePath as AbsolutePath;\n      assert(\n        absolutePath.startsWith(this.apiRootPath),\n        \"File path is not within the API root path\",\n      );\n\n      if (event !== \"change\" && event !== \"add\") {\n        return;\n      }\n\n      try {\n        // sonamu.config.ts 변경 시 재시작\n        const isConfigTs = filePath === path.join(this.apiRootPath, \"sonamu.config.ts\");\n\n        if (isConfigTs) {\n          const relativePath = filePath.replace(this.apiRootPath, \"api\");\n          const chalk = (await import(\"chalk\")).default;\n          console.log(\n            chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`),\n          );\n          process.kill(process.pid, \"SIGUSR2\");\n          return;\n        }\n\n        await this.handleFileChange(event, absolutePath);\n      } catch (e) {\n        console.error(e);\n      }\n    });\n  }\n\n  /*\n     A function that automatically handles init and destroy when using Sonamu via scripts.    \n  */\n  async runScript(fn: () => Promise<void>) {\n    await this.init(true, false, undefined, false);\n    try {\n      await fn();\n    } finally {\n      await this.destroy();\n    }\n  }\n\n  private async registerPlugins(server: FastifyInstance, plugins: SonamuServerOptions[\"plugins\"]) {\n    if (!plugins) {\n      return;\n    }\n\n    const pluginsModules = {\n      cors: \"@fastify/cors\",\n      formbody: \"@fastify/formbody\",\n      multipart: \"@fastify/multipart\",\n      qs: \"fastify-qs\",\n      sse: \"fastify-sse-v2\",\n      static: \"@fastify/static\",\n      session: \"@fastify/secure-session\",\n    } as const;\n\n    const registerPlugin = async <K extends keyof NonNullable<typeof plugins>>(\n      key: K,\n      pluginName: string,\n    ) => {\n      const option = plugins[key];\n      if (!option) return;\n\n      if (option === true) {\n        server.register((await import(pluginName)).default);\n      } else {\n        server.register((await import(pluginName)).default, option);\n      }\n    };\n\n    for (const [key, pluginName] of Object.entries(pluginsModules)) {\n      await registerPlugin(key as keyof typeof plugins, pluginName);\n    }\n\n    if (plugins.custom) {\n      plugins.custom(server);\n    }\n  }\n\n  private async registerAuth(\n    server: FastifyInstance,\n    options: NonNullable<SonamuServerOptions[\"auth\"]>,\n  ) {\n    // await import(\"fastify\");\n    const fastifyPassport = (await import(\"@fastify/passport\")).default;\n    server.register(fastifyPassport.initialize());\n    server.register(fastifyPassport.secureSession());\n\n    if (typeof options === \"boolean\") {\n      fastifyPassport.registerUserSerializer(async (user, _request) => user);\n      fastifyPassport.registerUserDeserializer(async (serialized, _request) => serialized);\n    } else {\n      fastifyPassport.registerUserSerializer(options.userSerializer);\n      fastifyPassport.registerUserDeserializer(options.userDeserializer);\n    }\n  }\n\n  private async boot(server: FastifyInstance, options: SonamuServerOptions) {\n    const port = options.listen?.port ?? 3000;\n    const host = options.listen?.host ?? \"localhost\";\n\n    server.addHook(\"onClose\", async () => {\n      await options.lifecycle?.onShutdown?.(server);\n      await this.destroy();\n    });\n\n    const shutdown = async () => {\n      try {\n        await server.close();\n        process.exit(0);\n      } catch (err) {\n        console.error(\"Error during shutdown:\", err);\n        process.exit(1);\n      }\n    };\n\n    process.on(\"SIGINT\", shutdown);\n    process.on(\"SIGTERM\", shutdown);\n\n    if (options.lifecycle?.onError) {\n      server.setErrorHandler(options.lifecycle?.onError);\n    }\n\n    server\n      .listen({ port, host })\n      .then(async () => {\n        await options.lifecycle?.onStart?.(server);\n      })\n      .catch(async (err) => {\n        const chalk = (await import(\"chalk\")).default;\n        console.error(chalk.red(\"Failed to start server:\", err));\n        await shutdown();\n      });\n  }\n\n  private async handleFileChange(event: string, filePath: AbsolutePath): Promise<void> {\n    // 첫 번째 파일이면 HMR 시작 시간 기록\n    if (this.pendingFiles.length === 0) {\n      this.hmrStartTime = Date.now();\n    }\n    this.pendingFiles.push(filePath);\n\n    const relativePath = path.relative(this.apiRootPath, filePath);\n    const chalk = (await import(\"chalk\")).default;\n    console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));\n\n    await this.syncer.syncFromWatcher(event, filePath);\n\n    // 처리 완료된 파일을 대기 목록에서 제거\n    this.pendingFiles = this.pendingFiles.slice(1);\n\n    // 모든 파일 처리가 완료되면 최종 메시지 출력\n    if (this.pendingFiles.length === 0) {\n      await this.finishHMR();\n    }\n  }\n\n  private async finishHMR(): Promise<void> {\n    await this.syncer.renewChecksums();\n\n    const endTime = Date.now();\n    const totalTime = endTime - this.hmrStartTime;\n    const [chalk, { centerText }] = await Promise.all([\n      (await import(\"chalk\")).default,\n      import(\"../utils/console-util\"),\n    ]);\n    const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;\n\n    console.log(chalk.black.bgGreen(centerText(msg)));\n  }\n\n  async destroy(): Promise<void> {\n    const { BaseModel } = await import(\"../database/base-model\");\n    await BaseModel.destroy();\n    await this.watcher?.close();\n    this.storage?.destroy();\n  }\n}\nexport const Sonamu = new SonamuClass();\n"],"names":["assert","AsyncLocalStorage","path","Naite","SonamuClass","isInitialized","asyncLocalStorage","uploadStorage","getContext","store","getStore","context","process","env","NODE_ENV","request","reply","headers","createSSE","naiteStore","Map","Error","getUploadContext","uploadContext","_apiRootPath","apiRootPath","appRootPath","split","sep","slice","join","_dbConfig","dbConfig","_syncer","syncer","_config","config","_secrets","secrets","_storage","storage","watcher","pendingFiles","hmrStartTime","server","initForTesting","init","undefined","doSilent","enableSync","forTesting","chalk","default","console","time","cyan","findApiRootPath","loadConfig","database","ANTHROPIC_API_KEY","anthropic_api_key","DB","generateDBConfig","log","green","EntityManager","autoload","Syncer","autoloadTypes","autoloadModels","autoloadApis","TemplateManager","isLocal","isTest","setupBiome","isHotReloadServer","sync","startWatcher","syncUI","timeEnd","createServer","initOptions","options","fastify","plugins","registerPlugins","auth","session","registerAuth","withFastify","apiConfig","boot","timezone","api","formatInTimeZone","ISO_DATE_REGEX","DATE_FORMAT","setReplySerializer","payload","JSON","stringify","_key","value","test","Date","get","route","prefix","_request","_reply","apis","all","found","find","url","httpMethod","method","toUpperCase","getApiHandler","NotFoundException","models","modelName","handler","guards","every","guard","guardHandler","getZodObjectFromApi","ReqType","types","which","reqBody","fastifyCaster","parse","e","ZodError","humanizeZodError","messages","map","issue","message","BadRequestException","zodError","type","contentType","cacheKey","cacheTtl","cachedData","cache","cacheKeyRes","resolveKey","key","ttl","error","createSSEFactory","_events","socket","bind","Promise","resolve","contextProvider","createStore","user","passport","login","logout","model","run","ApiParamType","result","methodName","apply","parameters","param","isContext","name","put","watchPath","chokidar","watch","ignored","stats","isFile","endsWith","persistent","ignoreInitial","on","event","filePath","absolutePath","startsWith","isConfigTs","relativePath","replace","bold","blue","kill","pid","handleFileChange","runScript","fn","destroy","pluginsModules","cors","formbody","multipart","qs","sse","static","registerPlugin","pluginName","option","register","Object","entries","custom","fastifyPassport","initialize","secureSession","registerUserSerializer","registerUserDeserializer","serialized","userSerializer","userDeserializer","port","listen","host","addHook","lifecycle","onShutdown","shutdown","close","exit","err","onError","setErrorHandler","then","onStart","catch","red","length","now","push","relative","syncFromWatcher","finishHMR","renewChecksums","endTime","totalTime","centerText","msg","white","black","bgGreen","BaseModel","Sonamu"],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ export declare const RESERVED_KEYWORDS: Set<string>;
2
+ export declare class ApiValidationError extends Error {
3
+ constructor(message: string);
4
+ }
5
+ export declare function validateMethodName(methodName: string): void;
6
+ //# sourceMappingURL=validator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validator.d.ts","sourceRoot":"","sources":["../../src/api/validator.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,iBAAiB,aAmE5B,CAAC;AAEH,qBAAa,kBAAmB,SAAQ,KAAK;gBAC/B,OAAO,EAAE,MAAM;CAI5B;AAED,wBAAgB,kBAAkB,CAAC,UAAU,EAAE,MAAM,QAMpD"}