sonamu 0.5.7 → 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 (529) 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 +13 -2
  39. package/dist/api/caster.d.ts.map +1 -1
  40. package/dist/api/caster.js +71 -2
  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 +258 -2
  44. package/dist/api/config.d.ts +90 -0
  45. package/dist/api/config.d.ts.map +1 -0
  46. package/dist/api/config.js +25 -0
  47. package/dist/api/context.d.ts +4 -2
  48. package/dist/api/context.d.ts.map +1 -1
  49. package/dist/api/context.js +3 -2
  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 +235 -2
  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 +9 -2
  56. package/dist/api/sonamu.d.ts +10 -24
  57. package/dist/api/sonamu.d.ts.map +1 -1
  58. package/dist/api/sonamu.js +514 -2
  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 +6 -1
  63. package/dist/bin/build-config.d.ts.map +1 -1
  64. package/dist/bin/build-config.js +15 -2
  65. package/dist/bin/cli.js +519 -2
  66. package/dist/bin/hot-hook-register.d.ts +11 -0
  67. package/dist/bin/hot-hook-register.d.ts.map +1 -0
  68. package/dist/bin/hot-hook-register.js +21 -0
  69. package/dist/bin/loader-register.d.ts +2 -0
  70. package/dist/bin/loader-register.d.ts.map +1 -0
  71. package/dist/bin/loader-register.js +34 -0
  72. package/dist/database/_batch_update.d.ts +5 -3
  73. package/dist/database/_batch_update.d.ts.map +1 -1
  74. package/dist/database/_batch_update.js +95 -2
  75. package/dist/database/base-model.d.ts +96 -10
  76. package/dist/database/base-model.d.ts.map +1 -1
  77. package/dist/database/base-model.js +390 -2
  78. package/dist/database/base-model.types.d.ts +93 -0
  79. package/dist/database/base-model.types.d.ts.map +1 -0
  80. package/dist/database/base-model.types.js +10 -0
  81. package/dist/database/code-generator.d.ts +1 -1
  82. package/dist/database/code-generator.d.ts.map +1 -1
  83. package/dist/database/code-generator.js +54 -2
  84. package/dist/database/db.d.ts +6 -21
  85. package/dist/database/db.d.ts.map +1 -1
  86. package/dist/database/db.js +129 -2
  87. package/dist/database/puri-subset.test-d.js +81 -0
  88. package/dist/database/puri-subset.types.d.ts +123 -0
  89. package/dist/database/puri-subset.types.d.ts.map +1 -0
  90. package/dist/database/puri-subset.types.js +16 -0
  91. package/dist/database/puri-wrapper.d.ts +13 -11
  92. package/dist/database/puri-wrapper.d.ts.map +1 -1
  93. package/dist/database/puri-wrapper.js +109 -2
  94. package/dist/database/puri.d.ts +41 -23
  95. package/dist/database/puri.d.ts.map +1 -1
  96. package/dist/database/puri.js +601 -2
  97. package/dist/database/puri.types.d.ts +25 -6
  98. package/dist/database/puri.types.d.ts.map +1 -1
  99. package/dist/database/puri.types.js +6 -2
  100. package/dist/database/transaction-context.d.ts +1 -1
  101. package/dist/database/transaction-context.d.ts.map +1 -1
  102. package/dist/database/transaction-context.js +14 -2
  103. package/dist/database/upsert-builder.d.ts +9 -3
  104. package/dist/database/upsert-builder.d.ts.map +1 -1
  105. package/dist/database/upsert-builder.js +365 -2
  106. package/dist/entity/entity-manager.d.ts +167 -2
  107. package/dist/entity/entity-manager.d.ts.map +1 -1
  108. package/dist/entity/entity-manager.js +130 -2
  109. package/dist/entity/entity.d.ts +5 -3
  110. package/dist/entity/entity.d.ts.map +1 -1
  111. package/dist/entity/entity.js +750 -2
  112. package/dist/exceptions/error-handler.d.ts +1 -1
  113. package/dist/exceptions/error-handler.d.ts.map +1 -1
  114. package/dist/exceptions/error-handler.js +29 -2
  115. package/dist/exceptions/so-exceptions.d.ts +1 -1
  116. package/dist/exceptions/so-exceptions.d.ts.map +1 -1
  117. package/dist/exceptions/so-exceptions.js +85 -2
  118. package/dist/file-storage/driver.d.ts +1 -1
  119. package/dist/file-storage/driver.d.ts.map +1 -1
  120. package/dist/file-storage/driver.js +79 -2
  121. package/dist/file-storage/file-storage.js +75 -2
  122. package/dist/index.d.ts +18 -9
  123. package/dist/index.d.ts.map +1 -1
  124. package/dist/index.js +34 -2
  125. package/dist/migration/code-generation.d.ts +1 -1
  126. package/dist/migration/code-generation.d.ts.map +1 -1
  127. package/dist/migration/code-generation.js +614 -2
  128. package/dist/migration/migration-set.d.ts +2 -10
  129. package/dist/migration/migration-set.d.ts.map +1 -1
  130. package/dist/migration/migration-set.js +213 -2
  131. package/dist/migration/migrator.d.ts +24 -82
  132. package/dist/migration/migrator.d.ts.map +1 -1
  133. package/dist/migration/migrator.js +330 -2
  134. package/dist/migration/postgresql-schema-reader.d.ts +51 -0
  135. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -0
  136. package/dist/migration/postgresql-schema-reader.js +245 -0
  137. package/dist/migration/types.d.ts +6 -38
  138. package/dist/migration/types.d.ts.map +1 -1
  139. package/dist/migration/types.js +3 -2
  140. package/dist/naite/messaging-types.d.ts +43 -0
  141. package/dist/naite/messaging-types.d.ts.map +1 -0
  142. package/dist/naite/messaging-types.js +7 -0
  143. package/dist/naite/naite-reporter.d.ts +41 -0
  144. package/dist/naite/naite-reporter.d.ts.map +1 -0
  145. package/dist/naite/naite-reporter.js +102 -0
  146. package/dist/naite/naite.d.ts +95 -0
  147. package/dist/naite/naite.d.ts.map +1 -0
  148. package/dist/naite/naite.js +316 -0
  149. package/dist/stream/index.js +3 -2
  150. package/dist/stream/sse.d.ts +2 -2
  151. package/dist/stream/sse.d.ts.map +1 -1
  152. package/dist/stream/sse.js +38 -2
  153. package/dist/syncer/api-parser.d.ts +10 -0
  154. package/dist/syncer/api-parser.d.ts.map +1 -0
  155. package/dist/syncer/api-parser.js +240 -0
  156. package/dist/syncer/checksum.d.ts +21 -0
  157. package/dist/syncer/checksum.d.ts.map +1 -0
  158. package/dist/syncer/checksum.js +98 -0
  159. package/dist/syncer/code-generator.d.ts +20 -0
  160. package/dist/syncer/code-generator.d.ts.map +1 -0
  161. package/dist/syncer/code-generator.js +161 -0
  162. package/dist/syncer/entity-operations.d.ts +17 -0
  163. package/dist/syncer/entity-operations.d.ts.map +1 -0
  164. package/dist/syncer/entity-operations.js +59 -0
  165. package/dist/syncer/file-patterns.d.ts +29 -0
  166. package/dist/syncer/file-patterns.d.ts.map +1 -0
  167. package/dist/syncer/file-patterns.js +38 -0
  168. package/dist/syncer/index.d.ts +6 -0
  169. package/dist/syncer/index.d.ts.map +1 -1
  170. package/dist/syncer/index.js +9 -2
  171. package/dist/syncer/module-loader.d.ts +35 -0
  172. package/dist/syncer/module-loader.d.ts.map +1 -0
  173. package/dist/syncer/module-loader.js +87 -0
  174. package/dist/syncer/syncer.d.ts +98 -106
  175. package/dist/syncer/syncer.d.ts.map +1 -1
  176. package/dist/syncer/syncer.js +422 -2
  177. package/dist/template/entity-converter.d.ts +14 -0
  178. package/dist/template/entity-converter.d.ts.map +1 -0
  179. package/dist/template/entity-converter.js +108 -0
  180. package/dist/template/helpers.d.ts +23 -0
  181. package/dist/template/helpers.d.ts.map +1 -0
  182. package/dist/template/helpers.js +64 -0
  183. package/dist/{templates → template/implementations}/entity.template.d.ts +3 -3
  184. package/dist/template/implementations/entity.template.d.ts.map +1 -0
  185. package/dist/template/implementations/entity.template.js +86 -0
  186. package/dist/{templates → template/implementations}/generated.template.d.ts +3 -4
  187. package/dist/template/implementations/generated.template.d.ts.map +1 -0
  188. package/dist/template/implementations/generated.template.js +249 -0
  189. package/dist/{templates → template/implementations}/generated_http.template.d.ts +3 -4
  190. package/dist/template/implementations/generated_http.template.d.ts.map +1 -0
  191. package/dist/template/implementations/generated_http.template.js +131 -0
  192. package/dist/{templates → template/implementations}/generated_sso.template.d.ts +4 -5
  193. package/dist/template/implementations/generated_sso.template.d.ts.map +1 -0
  194. package/dist/template/implementations/generated_sso.template.js +134 -0
  195. package/dist/{templates → template/implementations}/init_types.template.d.ts +3 -3
  196. package/dist/template/implementations/init_types.template.d.ts.map +1 -0
  197. package/dist/template/implementations/init_types.template.js +38 -0
  198. package/dist/template/implementations/model.template.d.ts +17 -0
  199. package/dist/template/implementations/model.template.d.ts.map +1 -0
  200. package/dist/template/implementations/model.template.js +181 -0
  201. package/dist/{templates → template/implementations}/model_test.template.d.ts +3 -3
  202. package/dist/template/implementations/model_test.template.d.ts.map +1 -0
  203. package/dist/template/implementations/model_test.template.js +35 -0
  204. package/dist/{templates → template/implementations}/service.template.d.ts +6 -6
  205. package/dist/template/implementations/service.template.d.ts.map +1 -0
  206. package/dist/template/implementations/service.template.js +201 -0
  207. package/dist/{templates → template/implementations}/view_enums_buttonset.template.d.ts +3 -3
  208. package/dist/template/implementations/view_enums_buttonset.template.d.ts.map +1 -0
  209. package/dist/template/implementations/view_enums_buttonset.template.js +31 -0
  210. package/dist/{templates → template/implementations}/view_enums_dropdown.template.d.ts +3 -4
  211. package/dist/template/implementations/view_enums_dropdown.template.d.ts.map +1 -0
  212. package/dist/template/implementations/view_enums_dropdown.template.js +50 -0
  213. package/dist/{templates → template/implementations}/view_enums_select.template.d.ts +3 -3
  214. package/dist/template/implementations/view_enums_select.template.d.ts.map +1 -0
  215. package/dist/template/implementations/view_enums_select.template.js +55 -0
  216. package/dist/{templates → template/implementations}/view_form.template.d.ts +5 -5
  217. package/dist/template/implementations/view_form.template.d.ts.map +1 -0
  218. package/dist/template/implementations/view_form.template.js +337 -0
  219. package/dist/{templates → template/implementations}/view_id_all_select.template.d.ts +3 -3
  220. package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -0
  221. package/dist/template/implementations/view_id_all_select.template.js +31 -0
  222. package/dist/{templates → template/implementations}/view_id_async_select.template.d.ts +3 -3
  223. package/dist/template/implementations/view_id_async_select.template.d.ts.map +1 -0
  224. package/dist/template/implementations/view_id_async_select.template.js +105 -0
  225. package/dist/{templates → template/implementations}/view_list.template.d.ts +5 -13
  226. package/dist/template/implementations/view_list.template.d.ts.map +1 -0
  227. package/dist/template/implementations/view_list.template.js +475 -0
  228. package/dist/template/implementations/view_list_columns.template.d.ts +17 -0
  229. package/dist/template/implementations/view_list_columns.template.d.ts.map +1 -0
  230. package/dist/template/implementations/view_list_columns.template.js +49 -0
  231. package/dist/{templates → template/implementations}/view_search_input.template.d.ts +3 -3
  232. package/dist/template/implementations/view_search_input.template.d.ts.map +1 -0
  233. package/dist/template/implementations/view_search_input.template.js +64 -0
  234. package/dist/template/index.d.ts +7 -0
  235. package/dist/template/index.d.ts.map +1 -0
  236. package/dist/template/index.js +8 -0
  237. package/dist/template/template-manager.d.ts +56 -0
  238. package/dist/template/template-manager.d.ts.map +1 -0
  239. package/dist/template/template-manager.js +125 -0
  240. package/dist/template/template-types.d.ts +16 -0
  241. package/dist/template/template-types.d.ts.map +1 -0
  242. package/dist/template/template-types.js +7 -0
  243. package/dist/template/template.d.ts +49 -0
  244. package/dist/template/template.d.ts.map +1 -0
  245. package/dist/template/template.js +60 -0
  246. package/dist/template/zod-converter.d.ts +51 -0
  247. package/dist/template/zod-converter.d.ts.map +1 -0
  248. package/dist/template/zod-converter.js +449 -0
  249. package/dist/testing/_relation-graph.d.ts +1 -1
  250. package/dist/testing/_relation-graph.d.ts.map +1 -1
  251. package/dist/testing/_relation-graph.js +89 -2
  252. package/dist/testing/fixture-manager.d.ts +42 -11
  253. package/dist/testing/fixture-manager.d.ts.map +1 -1
  254. package/dist/testing/fixture-manager.js +623 -2
  255. package/dist/types/types.d.ts +747 -143
  256. package/dist/types/types.d.ts.map +1 -1
  257. package/dist/types/types.js +546 -2
  258. package/dist/typings/knex.d.js +3 -2
  259. package/dist/utils/async-utils.d.ts +7 -0
  260. package/dist/utils/async-utils.d.ts.map +1 -1
  261. package/dist/utils/async-utils.js +57 -2
  262. package/dist/utils/console-util.d.ts +2 -0
  263. package/dist/utils/console-util.d.ts.map +1 -0
  264. package/dist/utils/console-util.js +6 -0
  265. package/dist/utils/controller.d.ts +1 -0
  266. package/dist/utils/controller.d.ts.map +1 -1
  267. package/dist/utils/controller.js +29 -2
  268. package/dist/utils/esm-utils.d.ts +39 -0
  269. package/dist/utils/esm-utils.d.ts.map +1 -0
  270. package/dist/utils/esm-utils.js +49 -0
  271. package/dist/utils/formatter.d.ts +3 -0
  272. package/dist/utils/formatter.d.ts.map +1 -0
  273. package/dist/utils/formatter.js +110 -0
  274. package/dist/utils/fs-utils.d.ts +1 -1
  275. package/dist/utils/fs-utils.d.ts.map +1 -1
  276. package/dist/utils/fs-utils.js +17 -2
  277. package/dist/utils/lodash-able.d.ts.map +1 -1
  278. package/dist/utils/lodash-able.js +6 -2
  279. package/dist/utils/model.js +22 -2
  280. package/dist/utils/object-utils.d.ts +44 -0
  281. package/dist/utils/object-utils.d.ts.map +1 -0
  282. package/dist/utils/object-utils.js +191 -0
  283. package/dist/utils/path-utils.d.ts +89 -0
  284. package/dist/utils/path-utils.d.ts.map +1 -0
  285. package/dist/utils/path-utils.js +60 -0
  286. package/dist/utils/process-utils.d.ts +13 -0
  287. package/dist/utils/process-utils.d.ts.map +1 -0
  288. package/dist/utils/process-utils.js +36 -0
  289. package/dist/utils/sql-parser.d.ts +5 -1
  290. package/dist/utils/sql-parser.d.ts.map +1 -1
  291. package/dist/utils/sql-parser.js +46 -2
  292. package/dist/utils/type-utils.d.ts +23 -0
  293. package/dist/utils/type-utils.d.ts.map +1 -0
  294. package/dist/utils/type-utils.js +45 -0
  295. package/dist/utils/utils.d.ts +10 -7
  296. package/dist/utils/utils.d.ts.map +1 -1
  297. package/dist/utils/utils.js +72 -2
  298. package/dist/utils/zod-error.d.ts +1 -1
  299. package/dist/utils/zod-error.d.ts.map +1 -1
  300. package/dist/utils/zod-error.js +19 -2
  301. package/package.json +65 -27
  302. package/src/ai/agents/agent.ts +87 -0
  303. package/src/ai/agents/index.ts +2 -0
  304. package/src/ai/agents/types.ts +47 -0
  305. package/src/ai/index.ts +1 -0
  306. package/src/ai/providers/rtzr/api.ts +37 -0
  307. package/src/ai/providers/rtzr/error.ts +34 -0
  308. package/src/ai/providers/rtzr/index.ts +4 -0
  309. package/src/ai/providers/rtzr/model.ts +201 -0
  310. package/src/ai/providers/rtzr/options.ts +49 -0
  311. package/src/ai/providers/rtzr/provider.ts +91 -0
  312. package/src/ai/providers/rtzr/utils.ts +127 -0
  313. package/src/api/base-frame.ts +4 -2
  314. package/src/api/caster.ts +17 -23
  315. package/src/api/code-converters.ts +178 -535
  316. package/src/api/config.ts +125 -0
  317. package/src/api/context.ts +7 -17
  318. package/src/api/decorators.ts +176 -46
  319. package/src/api/index.ts +2 -2
  320. package/src/api/sonamu.ts +190 -167
  321. package/src/api/validator.ts +83 -0
  322. package/src/bin/build-config.ts +8 -1
  323. package/src/bin/cli.ts +258 -124
  324. package/src/bin/hot-hook-register.ts +22 -0
  325. package/src/bin/loader-register.ts +38 -0
  326. package/src/database/_batch_update.ts +46 -31
  327. package/src/database/base-model.ts +390 -182
  328. package/src/database/base-model.types.ts +155 -0
  329. package/src/database/code-generator.ts +13 -32
  330. package/src/database/db.ts +40 -96
  331. package/src/database/puri-subset.test-d.ts +471 -0
  332. package/src/database/puri-subset.types.ts +195 -0
  333. package/src/database/puri-wrapper.ts +58 -67
  334. package/src/database/puri.ts +229 -148
  335. package/src/database/puri.types.ts +76 -30
  336. package/src/database/transaction-context.ts +1 -1
  337. package/src/database/upsert-builder.ts +262 -132
  338. package/src/entity/entity-manager.ts +48 -36
  339. package/src/entity/entity.ts +330 -248
  340. package/src/exceptions/error-handler.ts +3 -3
  341. package/src/exceptions/so-exceptions.ts +11 -11
  342. package/src/file-storage/driver.ts +5 -5
  343. package/src/file-storage/file-storage.ts +2 -2
  344. package/src/index.ts +18 -10
  345. package/src/migration/code-generation.ts +185 -172
  346. package/src/migration/migration-set.ts +80 -293
  347. package/src/migration/migrator.ts +199 -571
  348. package/src/migration/mysql-schema-reader.ts.txt +272 -0
  349. package/src/migration/postgresql-schema-reader.ts +310 -0
  350. package/src/migration/types.ts +6 -39
  351. package/src/naite/messaging-types.ts +51 -0
  352. package/src/naite/naite-reporter.ts +128 -0
  353. package/src/naite/naite.ts +415 -0
  354. package/src/shared/web.shared.ts.txt +20 -24
  355. package/src/stream/sse.ts +5 -5
  356. package/src/syncer/api-parser.ts +282 -0
  357. package/src/syncer/checksum.ts +140 -0
  358. package/src/syncer/code-generator.ts +198 -0
  359. package/src/syncer/entity-operations.ts +65 -0
  360. package/src/syncer/file-patterns.ts +56 -0
  361. package/src/syncer/index.ts +6 -0
  362. package/src/syncer/module-loader.ts +128 -0
  363. package/src/syncer/syncer.ts +389 -1453
  364. package/src/template/entity-converter.ts +114 -0
  365. package/src/template/helpers.ts +81 -0
  366. package/src/{templates → template/implementations}/entity.template.ts +7 -7
  367. package/src/{templates → template/implementations}/generated.template.ts +101 -101
  368. package/src/{templates → template/implementations}/generated_http.template.ts +27 -57
  369. package/src/template/implementations/generated_sso.template.ts +151 -0
  370. package/src/{templates → template/implementations}/init_types.template.ts +5 -7
  371. package/src/{templates → template/implementations}/model.template.ts +52 -43
  372. package/src/{templates → template/implementations}/model_test.template.ts +5 -5
  373. package/src/{templates → template/implementations}/service.template.ts +66 -82
  374. package/src/{templates → template/implementations}/view_enums_buttonset.template.ts +3 -3
  375. package/src/{templates → template/implementations}/view_enums_dropdown.template.ts +4 -20
  376. package/src/{templates → template/implementations}/view_enums_select.template.ts +4 -4
  377. package/src/{templates → template/implementations}/view_form.template.ts +40 -83
  378. package/src/{templates → template/implementations}/view_id_all_select.template.ts +3 -3
  379. package/src/{templates → template/implementations}/view_id_async_select.template.ts +10 -24
  380. package/src/{templates → template/implementations}/view_list.template.ts +60 -152
  381. package/src/{templates → template/implementations}/view_list_columns.template.ts +5 -11
  382. package/src/{templates → template/implementations}/view_search_input.template.ts +3 -3
  383. package/src/template/index.ts +6 -0
  384. package/src/template/template-manager.ts +166 -0
  385. package/src/template/template-types.ts +16 -0
  386. package/src/template/template.ts +105 -0
  387. package/src/template/zod-converter.ts +525 -0
  388. package/src/testing/_relation-graph.ts +18 -11
  389. package/src/testing/fixture-manager.ts +472 -359
  390. package/src/types/types.ts +553 -308
  391. package/src/typings/knex.d.ts +7 -9
  392. package/src/utils/async-utils.ts +23 -10
  393. package/src/utils/console-util.ts +4 -0
  394. package/src/utils/controller.ts +3 -0
  395. package/src/utils/esm-utils.ts +59 -0
  396. package/src/utils/formatter.ts +109 -0
  397. package/src/utils/fs-utils.ts +1 -1
  398. package/src/utils/lodash-able.ts +1 -4
  399. package/src/utils/object-utils.ts +217 -0
  400. package/src/utils/path-utils.ts +99 -0
  401. package/src/utils/process-utils.ts +46 -0
  402. package/src/utils/sql-parser.ts +23 -5
  403. package/src/utils/type-utils.ts +83 -0
  404. package/src/utils/utils.ts +66 -43
  405. package/src/utils/zod-error.ts +3 -4
  406. package/dist/api/base-frame.js.map +0 -1
  407. package/dist/api/caster.js.map +0 -1
  408. package/dist/api/code-converters.js.map +0 -1
  409. package/dist/api/context.js.map +0 -1
  410. package/dist/api/decorators.js.map +0 -1
  411. package/dist/api/index.js.map +0 -1
  412. package/dist/api/sonamu.js.map +0 -1
  413. package/dist/bin/build-config.js.map +0 -1
  414. package/dist/bin/cli-wrapper.d.ts +0 -3
  415. package/dist/bin/cli-wrapper.d.ts.map +0 -1
  416. package/dist/bin/cli-wrapper.js +0 -3
  417. package/dist/bin/cli-wrapper.js.map +0 -1
  418. package/dist/bin/cli.js.map +0 -1
  419. package/dist/database/_batch_update.js.map +0 -1
  420. package/dist/database/base-model.js.map +0 -1
  421. package/dist/database/code-generator.js.map +0 -1
  422. package/dist/database/db.js.map +0 -1
  423. package/dist/database/knex-plugins/knex-on-duplicate-update.d.ts +0 -2
  424. package/dist/database/knex-plugins/knex-on-duplicate-update.d.ts.map +0 -1
  425. package/dist/database/knex-plugins/knex-on-duplicate-update.js +0 -2
  426. package/dist/database/knex-plugins/knex-on-duplicate-update.js.map +0 -1
  427. package/dist/database/puri-wrapper.js.map +0 -1
  428. package/dist/database/puri.js.map +0 -1
  429. package/dist/database/puri.types.js.map +0 -1
  430. package/dist/database/transaction-context.js.map +0 -1
  431. package/dist/database/upsert-builder.js.map +0 -1
  432. package/dist/entity/entity-manager.js.map +0 -1
  433. package/dist/entity/entity-utils.d.ts +0 -61
  434. package/dist/entity/entity-utils.d.ts.map +0 -1
  435. package/dist/entity/entity-utils.js +0 -2
  436. package/dist/entity/entity-utils.js.map +0 -1
  437. package/dist/entity/entity.js.map +0 -1
  438. package/dist/exceptions/error-handler.js.map +0 -1
  439. package/dist/exceptions/so-exceptions.js.map +0 -1
  440. package/dist/file-storage/driver.js.map +0 -1
  441. package/dist/file-storage/file-storage.js.map +0 -1
  442. package/dist/index.js.map +0 -1
  443. package/dist/migration/code-generation.js.map +0 -1
  444. package/dist/migration/migration-set.js.map +0 -1
  445. package/dist/migration/migrator.js.map +0 -1
  446. package/dist/migration/types.js.map +0 -1
  447. package/dist/stream/index.js.map +0 -1
  448. package/dist/stream/sse.js.map +0 -1
  449. package/dist/syncer/index.js.map +0 -1
  450. package/dist/syncer/syncer.js.map +0 -1
  451. package/dist/templates/base-template.d.ts +0 -13
  452. package/dist/templates/base-template.d.ts.map +0 -1
  453. package/dist/templates/base-template.js +0 -2
  454. package/dist/templates/base-template.js.map +0 -1
  455. package/dist/templates/entity.template.d.ts.map +0 -1
  456. package/dist/templates/entity.template.js +0 -2
  457. package/dist/templates/entity.template.js.map +0 -1
  458. package/dist/templates/generated.template.d.ts.map +0 -1
  459. package/dist/templates/generated.template.js +0 -2
  460. package/dist/templates/generated.template.js.map +0 -1
  461. package/dist/templates/generated_http.template.d.ts.map +0 -1
  462. package/dist/templates/generated_http.template.js +0 -2
  463. package/dist/templates/generated_http.template.js.map +0 -1
  464. package/dist/templates/generated_sso.template.d.ts.map +0 -1
  465. package/dist/templates/generated_sso.template.js +0 -2
  466. package/dist/templates/generated_sso.template.js.map +0 -1
  467. package/dist/templates/index.d.ts +0 -2
  468. package/dist/templates/index.d.ts.map +0 -1
  469. package/dist/templates/index.js +0 -2
  470. package/dist/templates/index.js.map +0 -1
  471. package/dist/templates/init_types.template.d.ts.map +0 -1
  472. package/dist/templates/init_types.template.js +0 -2
  473. package/dist/templates/init_types.template.js.map +0 -1
  474. package/dist/templates/model.template.d.ts +0 -17
  475. package/dist/templates/model.template.d.ts.map +0 -1
  476. package/dist/templates/model.template.js +0 -2
  477. package/dist/templates/model.template.js.map +0 -1
  478. package/dist/templates/model_test.template.d.ts.map +0 -1
  479. package/dist/templates/model_test.template.js +0 -2
  480. package/dist/templates/model_test.template.js.map +0 -1
  481. package/dist/templates/service.template.d.ts.map +0 -1
  482. package/dist/templates/service.template.js +0 -2
  483. package/dist/templates/service.template.js.map +0 -1
  484. package/dist/templates/view_enums_buttonset.template.d.ts.map +0 -1
  485. package/dist/templates/view_enums_buttonset.template.js +0 -2
  486. package/dist/templates/view_enums_buttonset.template.js.map +0 -1
  487. package/dist/templates/view_enums_dropdown.template.d.ts.map +0 -1
  488. package/dist/templates/view_enums_dropdown.template.js +0 -2
  489. package/dist/templates/view_enums_dropdown.template.js.map +0 -1
  490. package/dist/templates/view_enums_select.template.d.ts.map +0 -1
  491. package/dist/templates/view_enums_select.template.js +0 -2
  492. package/dist/templates/view_enums_select.template.js.map +0 -1
  493. package/dist/templates/view_form.template.d.ts.map +0 -1
  494. package/dist/templates/view_form.template.js +0 -2
  495. package/dist/templates/view_form.template.js.map +0 -1
  496. package/dist/templates/view_id_all_select.template.d.ts.map +0 -1
  497. package/dist/templates/view_id_all_select.template.js +0 -2
  498. package/dist/templates/view_id_all_select.template.js.map +0 -1
  499. package/dist/templates/view_id_async_select.template.d.ts.map +0 -1
  500. package/dist/templates/view_id_async_select.template.js +0 -2
  501. package/dist/templates/view_id_async_select.template.js.map +0 -1
  502. package/dist/templates/view_list.template.d.ts.map +0 -1
  503. package/dist/templates/view_list.template.js +0 -2
  504. package/dist/templates/view_list.template.js.map +0 -1
  505. package/dist/templates/view_list_columns.template.d.ts +0 -17
  506. package/dist/templates/view_list_columns.template.d.ts.map +0 -1
  507. package/dist/templates/view_list_columns.template.js +0 -2
  508. package/dist/templates/view_list_columns.template.js.map +0 -1
  509. package/dist/templates/view_search_input.template.d.ts.map +0 -1
  510. package/dist/templates/view_search_input.template.js +0 -2
  511. package/dist/templates/view_search_input.template.js.map +0 -1
  512. package/dist/testing/_relation-graph.js.map +0 -1
  513. package/dist/testing/fixture-manager.js.map +0 -1
  514. package/dist/types/types.js.map +0 -1
  515. package/dist/typings/knex.d.js.map +0 -1
  516. package/dist/utils/async-utils.js.map +0 -1
  517. package/dist/utils/controller.js.map +0 -1
  518. package/dist/utils/fs-utils.js.map +0 -1
  519. package/dist/utils/lodash-able.js.map +0 -1
  520. package/dist/utils/model.js.map +0 -1
  521. package/dist/utils/sql-parser.js.map +0 -1
  522. package/dist/utils/utils.js.map +0 -1
  523. package/dist/utils/zod-error.js.map +0 -1
  524. package/src/bin/cli-wrapper.ts +0 -75
  525. package/src/database/knex-plugins/knex-on-duplicate-update.ts +0 -45
  526. package/src/entity/entity-utils.ts +0 -291
  527. package/src/templates/base-template.ts +0 -19
  528. package/src/templates/generated_sso.template.ts +0 -138
  529. package/src/templates/index.ts +0 -1
@@ -0,0 +1,471 @@
1
+ import { describe, expectTypeOf, it } from "vitest";
2
+ import type { DatabaseSchemaExtend } from "../types/types";
3
+ import type { Puri } from "./puri";
4
+ import type { Hydrate, InferAllSubsets, LoadersResult } from "./puri-subset.types";
5
+ import type { PuriWrapper } from "./puri-wrapper";
6
+
7
+ type MockPuri<T> = Puri<DatabaseSchemaExtend, any, T>;
8
+
9
+ describe("Hydrate", () => {
10
+ it("flat 객체를 그대로 유지한다 (__ 없는 경우)", () => {
11
+ type Input = { id: number; name: string };
12
+ type Result = Hydrate<Input>;
13
+
14
+ const result = {} as Result;
15
+ expectTypeOf(result).toEqualTypeOf<Hydrate<Input>>();
16
+ });
17
+
18
+ it("단일 depth의 __ 키를 중첩 객체로 변환한다", () => {
19
+ type Input = { id: number; user__name: string; user__email: string };
20
+ type Result = Hydrate<Input>;
21
+
22
+ const result = {} as Result;
23
+ expectTypeOf(result).toEqualTypeOf<{
24
+ id: number;
25
+ user: { name: string; email: string };
26
+ }>();
27
+ });
28
+
29
+ it("다중 depth의 __ 키를 재귀적으로 중첩 객체로 변환한다", () => {
30
+ type Input = {
31
+ id: number;
32
+ user__profile__bio: string;
33
+ user__profile__avatar: string;
34
+ user__name: string;
35
+ };
36
+ type Result = Hydrate<Input>;
37
+
38
+ const result = {} as Result;
39
+ expectTypeOf(result).toEqualTypeOf<{
40
+ id: number;
41
+ user: {
42
+ name: string;
43
+ profile: { bio: string; avatar: string };
44
+ };
45
+ }>();
46
+ });
47
+
48
+ it("여러 관계를 동시에 처리한다", () => {
49
+ type Input = {
50
+ id: number;
51
+ user__id: number;
52
+ user__name: string;
53
+ post__id: number;
54
+ post__title: string;
55
+ };
56
+ type Result = Hydrate<Input>;
57
+
58
+ const result = {} as Result;
59
+ expectTypeOf(result).toEqualTypeOf<{
60
+ id: number;
61
+ user: { id: number; name: string };
62
+ post: { id: number; title: string };
63
+ }>();
64
+ });
65
+
66
+ it("빈 객체를 처리한다", () => {
67
+ type Input = {};
68
+ type Result = Hydrate<Input>;
69
+
70
+ const result = {} as Result;
71
+ expectTypeOf(result).toEqualTypeOf<{}>();
72
+ });
73
+
74
+ it("nullable 필드가 있는 중첩 객체를 처리한다", () => {
75
+ type Input = {
76
+ id: number;
77
+ user__id: number | null;
78
+ user__name: string | null;
79
+ };
80
+ type Result = Hydrate<Input>;
81
+
82
+ const result = {} as Result;
83
+ expectTypeOf(result).toEqualTypeOf<{
84
+ id: number;
85
+ user: { id: number | null; name: string | null };
86
+ }>();
87
+ });
88
+
89
+ it("동일한 prefix를 가진 여러 필드를 올바르게 그룹화한다", () => {
90
+ type Input = {
91
+ id: number;
92
+ user__id: number;
93
+ user__name: string;
94
+ user_count: number;
95
+ };
96
+ type Result = Hydrate<Input>;
97
+
98
+ const result = {} as Result;
99
+ expectTypeOf(result).toEqualTypeOf<{
100
+ id: number;
101
+ user: { id: number; name: string };
102
+ user_count: number;
103
+ }>();
104
+ });
105
+ });
106
+
107
+ describe("LoadersResult", () => {
108
+ it("단일 로더의 결과 타입을 생성한다", () => {
109
+ type MockLoaderQb = (
110
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
111
+ fromIds: number[],
112
+ ) => MockPuri<{ id: number; title: string; refId: number }>;
113
+
114
+ type Loaders = [
115
+ {
116
+ as: "posts";
117
+ refId: "id";
118
+ qb: MockLoaderQb;
119
+ },
120
+ ];
121
+ type Result = LoadersResult<Loaders>;
122
+
123
+ const result = {} as Result;
124
+ expectTypeOf(result).toHaveProperty("posts");
125
+ expectTypeOf(result).toEqualTypeOf<{ posts: { id: number; title: string }[] }>();
126
+ });
127
+
128
+ it("중첩 로더를 처리한다", () => {
129
+ type CommentLoaderQb = (
130
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
131
+ fromIds: number[],
132
+ ) => MockPuri<{ id: number; content: string; refId: number }>;
133
+
134
+ type PostLoaderQb = (
135
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
136
+ fromIds: number[],
137
+ ) => MockPuri<{ id: number; title: string; refId: number }>;
138
+
139
+ type Loaders = [
140
+ {
141
+ as: "posts";
142
+ refId: "id";
143
+ qb: PostLoaderQb;
144
+ loaders: [
145
+ {
146
+ as: "comments";
147
+ refId: "id";
148
+ qb: CommentLoaderQb;
149
+ },
150
+ ];
151
+ },
152
+ ];
153
+ type Result = LoadersResult<Loaders>;
154
+
155
+ const result = {} as Result;
156
+ expectTypeOf(result).toHaveProperty("posts");
157
+ expectTypeOf(result).toEqualTypeOf<{
158
+ posts: { id: number; title: string; comments: { id: number; content: string }[] }[];
159
+ }>();
160
+ });
161
+
162
+ it("여러 로더를 처리한다", () => {
163
+ type ProjectLoader = {
164
+ as: "projects";
165
+ refId: "id";
166
+ qb: (
167
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
168
+ fromIds: number[],
169
+ ) => MockPuri<{ id: number; name: string; refId: number }>;
170
+ };
171
+ type DepartmentLoader = {
172
+ as: "department";
173
+ refId: "id";
174
+ qb: (
175
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
176
+ fromIds: number[],
177
+ ) => MockPuri<{ id: number; name: string; company_name: string; refId: number }>;
178
+ };
179
+ type EmployeeLoader = {
180
+ as: "employees";
181
+ refId: "id";
182
+ qb: (
183
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
184
+ fromIds: number[],
185
+ ) => MockPuri<{ id: number; employee_number: string; refId: number }>;
186
+ loaders: [
187
+ {
188
+ as: "user";
189
+ refId: "id";
190
+ qb: (
191
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
192
+ fromIds: number[],
193
+ ) => MockPuri<{ id: number; name: string; email: string; refId: number }>;
194
+ },
195
+ ];
196
+ };
197
+
198
+ type Loaders = [ProjectLoader, DepartmentLoader, EmployeeLoader];
199
+ type Result = LoadersResult<Loaders>;
200
+
201
+ const result = {} as Result;
202
+ expectTypeOf(result).toHaveProperty("projects");
203
+ expectTypeOf(result).toHaveProperty("employees");
204
+ expectTypeOf(result).toHaveProperty("department");
205
+ expectTypeOf(result).toEqualTypeOf<{
206
+ projects: { id: number; name: string }[];
207
+ department: { id: number; name: string; company_name: string }[];
208
+ employees: {
209
+ id: number;
210
+ employee_number: string;
211
+ user: { id: number; name: string; email: string }[];
212
+ }[];
213
+ }>();
214
+ });
215
+ });
216
+
217
+ describe("InferAllSubsets", () => {
218
+ describe("서브셋이 하나인 경우", () => {
219
+ it("로더 없이 기본 Subset 결과를 추론한다", () => {
220
+ // SubsetQuery
221
+ type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{
222
+ id: number;
223
+ user__name: string;
224
+ user__email: string;
225
+ department__name: string;
226
+ department__company__name: string;
227
+ }>;
228
+ type SubsetQueries = {
229
+ A: SubsetFnA;
230
+ };
231
+
232
+ // LoaderQuery
233
+ type LoaderQueries = {
234
+ A: [];
235
+ };
236
+
237
+ // Result
238
+ type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;
239
+
240
+ const result = {} as Result;
241
+ expectTypeOf(result).toHaveProperty("A");
242
+ expectTypeOf(result.A).toEqualTypeOf<{
243
+ id: number;
244
+ user: { name: string; email: string };
245
+ department: { name: string; company: { name: string } };
246
+ }>();
247
+ expectTypeOf(result.A.department.company).toEqualTypeOf<{ name: string }>();
248
+ });
249
+
250
+ it("로더를 포함한 Subset 결과를 추론한다", () => {
251
+ // SubsetQuery
252
+ type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{
253
+ id: number;
254
+ company__id: number;
255
+ company__name: string;
256
+ department__id: number;
257
+ department__name: string;
258
+ }>;
259
+ type SubsetQueries = {
260
+ A: SubsetFnA;
261
+ };
262
+
263
+ // LoaderQuery
264
+ type CompanyDepartmentsLoader = {
265
+ as: "company__departments";
266
+ refId: "company__id";
267
+ qb: (
268
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
269
+ fromIds: number[],
270
+ ) => MockPuri<{ id: number; name: string; refId: number }>;
271
+ };
272
+ type DepartmentProjectsLoader = {
273
+ as: "department__projects";
274
+ refId: "department__id";
275
+ qb: (
276
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
277
+ fromIds: number[],
278
+ ) => MockPuri<{ id: number; name: string; status: string; refId: number }>;
279
+ };
280
+ type LoaderQueries = {
281
+ A: [CompanyDepartmentsLoader, DepartmentProjectsLoader];
282
+ };
283
+
284
+ // Result
285
+ type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;
286
+
287
+ const result = {} as Result;
288
+ expectTypeOf(result).toHaveProperty("A");
289
+ expectTypeOf(result).toEqualTypeOf<{
290
+ A: {
291
+ id: number;
292
+ company: { id: number; name: string; departments: { id: number; name: string }[] };
293
+ department: {
294
+ id: number;
295
+ name: string;
296
+ projects: { id: number; name: string; status: string }[];
297
+ };
298
+ };
299
+ }>();
300
+ });
301
+
302
+ it("중첩 로더를 포함한 Subset 결과를 추론한다", () => {
303
+ // SubsetQuery
304
+ type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{
305
+ id: number;
306
+ company__name: string;
307
+ }>;
308
+ type SubsetQueries = {
309
+ A: SubsetFnA;
310
+ };
311
+
312
+ // LoaderQuery
313
+ type CompanyDepartmentsLoader = {
314
+ as: "company__departments";
315
+ refId: "company__id";
316
+ qb: (
317
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
318
+ fromIds: number[],
319
+ ) => MockPuri<{ id: number; name: string; refId: number }>;
320
+ loaders: [
321
+ {
322
+ as: "projects";
323
+ refId: "id";
324
+ qb: (
325
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
326
+ fromIds: number[],
327
+ ) => MockPuri<{ id: number; name: string; status: string; refId: number }>;
328
+ },
329
+ ];
330
+ };
331
+ type LoaderQueries = {
332
+ A: [CompanyDepartmentsLoader];
333
+ };
334
+
335
+ // Result
336
+ type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;
337
+
338
+ const result = {} as Result;
339
+ expectTypeOf(result).toEqualTypeOf<{
340
+ A: {
341
+ id: number;
342
+ company: {
343
+ name: string;
344
+ departments: {
345
+ id: number;
346
+ name: string;
347
+ projects: { id: number; name: string; status: string }[];
348
+ }[];
349
+ };
350
+ };
351
+ }>();
352
+ });
353
+ });
354
+
355
+ describe("서브셋이 여러 개인 경우", () => {
356
+ it("로더 없이 기본 Subset 결과를 추론한다", () => {
357
+ // SubsetQuery
358
+ type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{
359
+ id: number;
360
+ name: string;
361
+ email: string;
362
+ department__name: string;
363
+ }>;
364
+ type SubsetFnP = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{
365
+ id: number;
366
+ name: string;
367
+ department__name: string;
368
+ }>;
369
+ type SubsetFnSS = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{
370
+ id: number;
371
+ name: string;
372
+ email: string;
373
+ password: string;
374
+ }>;
375
+ type SubsetQueries = {
376
+ A: SubsetFnA;
377
+ P: SubsetFnP;
378
+ SS: SubsetFnSS;
379
+ };
380
+
381
+ // LoaderQuery
382
+ type LoaderQueries = {
383
+ A: [];
384
+ P: [];
385
+ SS: [];
386
+ };
387
+
388
+ // Result
389
+ type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;
390
+
391
+ const result = {} as Result;
392
+ expectTypeOf(result).toEqualTypeOf<{
393
+ A: { id: number; name: string; email: string; department: { name: string } };
394
+ P: { id: number; name: string; department: { name: string } };
395
+ SS: { id: number; name: string; email: string; password: string };
396
+ }>();
397
+ });
398
+
399
+ it("로더를 포함한 Subset 결과를 추론한다", () => {
400
+ // SubsetQuery
401
+ type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{
402
+ id: number;
403
+ name: string;
404
+ email: string;
405
+ department__name: string;
406
+ }>;
407
+ type SubsetFnP = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{
408
+ id: number;
409
+ name: string;
410
+ department__name: string;
411
+ }>;
412
+ type SubsetFnSS = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{
413
+ id: number;
414
+ name: string;
415
+ email: string;
416
+ password: string;
417
+ }>;
418
+ type SubsetQueries = {
419
+ A: SubsetFnA;
420
+ P: SubsetFnP;
421
+ SS: SubsetFnSS;
422
+ };
423
+
424
+ // LoaderQuery
425
+ type ProjectLoader = {
426
+ as: "projects";
427
+ refId: "id";
428
+ qb: (
429
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
430
+ fromIds: number[],
431
+ ) => MockPuri<{ id: number; name: string; status: string; refId: number }>;
432
+ loaders: [
433
+ {
434
+ as: "tags";
435
+ refId: "id";
436
+ qb: (
437
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
438
+ fromIds: number[],
439
+ ) => MockPuri<{ id: number; name: string; refId: number }>;
440
+ },
441
+ ];
442
+ };
443
+ type LoaderQueries = {
444
+ A: [ProjectLoader];
445
+ P: [];
446
+ SS: [];
447
+ };
448
+
449
+ // Result
450
+ type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;
451
+
452
+ const result = {} as Result;
453
+ expectTypeOf(result).toEqualTypeOf<{
454
+ A: {
455
+ id: number;
456
+ name: string;
457
+ email: string;
458
+ department: { name: string };
459
+ projects: {
460
+ id: number;
461
+ name: string;
462
+ status: string;
463
+ tags: { id: number; name: string }[];
464
+ }[];
465
+ };
466
+ P: { id: number; name: string; department: { name: string } };
467
+ SS: { id: number; name: string; email: string; password: string };
468
+ }>();
469
+ });
470
+ });
471
+ });
@@ -0,0 +1,195 @@
1
+ /**
2
+ * Puri Subset 타입 시스템
3
+ *
4
+ * SubsetQuery와 LoaderQuery를 기반으로 최종 결과 타입을 추론하는 타입 유틸리티들.
5
+ * 핵심 개념:
6
+ * - SubsetQuery: 기본 쿼리 (join, select)
7
+ * - LoaderQuery: 1:N, N:M 관계 데이터 로딩
8
+ * - Hydrate: flat한 결과를 중첩 객체로 변환 (예: user__name → { user: { name } })
9
+ */
10
+
11
+ import type { DatabaseSchemaExtend } from "../types/types";
12
+ import type { Puri } from "./puri";
13
+ import type { Expand } from "./puri.types";
14
+ import type { PuriWrapper } from "./puri-wrapper";
15
+
16
+ // ============================================================================
17
+ // 기본 타입 정의
18
+ // ============================================================================
19
+
20
+ /**
21
+ * SubsetQuery 함수 시그니처
22
+ * PuriWrapper를 받아 Puri 쿼리 빌더를 반환
23
+ */
24
+ export type PuriSubsetFn = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => Puri<any, any, any>;
25
+
26
+ /**
27
+ * Puri 인스턴스에서 TResult 타입 추출
28
+ */
29
+ export type ExtractPuriResult<T> = T extends Puri<any, any, infer R> ? R : never;
30
+
31
+ // ============================================================================
32
+ // Loader 관련 타입
33
+ // ============================================================================
34
+
35
+ /**
36
+ * Loader 쿼리 빌더 함수 시그니처
37
+ * @param qbWrapper - Puri 래퍼
38
+ * @param fromIds - 부모 레코드 ID 배열
39
+ */
40
+ export type PuriLoaderQbFn = (
41
+ qbWrapper: PuriWrapper<DatabaseSchemaExtend>,
42
+ fromIds: number[],
43
+ ) => Puri<any, any, any>;
44
+
45
+ /**
46
+ * 단일 Loader 정의
47
+ * 1:N 또는 N:M 관계 데이터를 로드하는 설정
48
+ */
49
+ export type GenericPuriLoader = {
50
+ /** 결과 객체에서 사용할 필드명 */
51
+ as: string;
52
+ /** 부모 레코드와 연결할 참조 필드명 */
53
+ refId: string;
54
+ /** 데이터 로딩 쿼리 빌더 */
55
+ qb: PuriLoaderQbFn;
56
+ /** 중첩 로더 (재귀적 로딩 지원) */
57
+ loaders?: GenericPuriLoader[];
58
+ };
59
+
60
+ /**
61
+ * 모델별 Loader 쿼리 컬렉션
62
+ * 각 SubsetKey에 대해 Loader 배열을 정의
63
+ */
64
+ export type PuriLoaderQueries<TSubsetKey extends string> = Record<TSubsetKey, GenericPuriLoader[]>;
65
+
66
+ // ============================================================================
67
+ // Hydrate 타입 시스템
68
+ // ============================================================================
69
+
70
+ /**
71
+ * 구분자(__) 앞부분 추출
72
+ * @example ExtractHead<"user__name"> = "user"
73
+ */
74
+ type ExtractHead<K extends string> = K extends `${infer Head}__${string}` ? Head : never;
75
+
76
+ /**
77
+ * 구분자(__) 뒷부분 추출
78
+ * @example ExtractTail<"user__profile__name", "user"> = "profile__name"
79
+ */
80
+ type ExtractTail<K extends string, Head extends string> = K extends `${Head}__${infer Tail}`
81
+ ? Tail
82
+ : never;
83
+
84
+ /**
85
+ * Flat 객체를 중첩 객체로 변환
86
+ *
87
+ * 런타임 hydrate 함수와 동일한 동작을 타입 레벨에서 구현.
88
+ * `__`로 구분된 키를 중첩 객체 구조로 변환.
89
+ *
90
+ * @example
91
+ * type Input = { id: number; user__name: string; user__profile__bio: string }
92
+ * type Output = Hydrate<Input>
93
+ * // { id: number; user: { name: string; profile: { bio: string } } }
94
+ */
95
+ export type Hydrate<T> = Expand<HydrateInner<T>>;
96
+ type HydrateInner<T> = {
97
+ [K in keyof T as K extends `${string}__${string}` ? never : K]: T[K];
98
+ } & {
99
+ [K in ExtractHead<keyof T & string>]: Expand<
100
+ HydrateInner<{
101
+ [P in keyof T as P extends `${K}__${string}` ? ExtractTail<P & string, K> : never]: T[P];
102
+ }>
103
+ >;
104
+ };
105
+
106
+ // ============================================================================
107
+ // Loader 결과 타입 추론
108
+ // ============================================================================
109
+
110
+ /**
111
+ * Loader 쿼리 함수에서 결과 타입 추출 (refId 제외, Hydrate 적용)
112
+ */
113
+ type ExtractLoaderResult<TLoaderQb> = TLoaderQb extends PuriLoaderQbFn
114
+ ? Expand<Hydrate<Omit<ExtractPuriResult<ReturnType<TLoaderQb>>, "refId">>>
115
+ : never;
116
+
117
+ /**
118
+ * Loader 배열에서 결과 객체 타입 빌드 (재귀적으로 중첩 loader 처리)
119
+ *
120
+ * @example
121
+ * type Loaders = [
122
+ * { as: "employees"; refId: "id"; qb: ...; loaders: [{ as: "projects"; ... }] }
123
+ * ]
124
+ * type Result = LoadersResult<Loaders>
125
+ * // { employees: Array<{ ...employee fields; projects: Array<...> }> }
126
+ */
127
+ export type LoadersResult<TLoaders extends GenericPuriLoader[]> = Expand<{
128
+ [L in TLoaders[number] as L["as"]]: WithLoaders<ExtractLoaderResult<L["qb"]>, L["loaders"]>[];
129
+ }>;
130
+
131
+ /**
132
+ * 기본 결과와 Loader 결과 병합
133
+ */
134
+ type WithLoaders<TBase, TLoaders> = TLoaders extends GenericPuriLoader[]
135
+ ? Expand<TBase & LoadersResult<TLoaders>>
136
+ : TBase;
137
+
138
+ // ============================================================================
139
+ // Subset 결과 타입 추론
140
+ // ============================================================================
141
+
142
+ /**
143
+ * 단일 Subset 함수 + Loader에서 최종 결과 타입 추론
144
+ */
145
+ type InferSubsetWithLoaders<
146
+ TSubsetFn extends (...args: any) => Puri<any, any, any>,
147
+ TLoaders extends GenericPuriLoader[] | undefined = undefined,
148
+ > = Expand<Hydrate<WithLoaders<ExtractPuriResult<ReturnType<TSubsetFn>>, TLoaders>>>;
149
+
150
+ /**
151
+ * 전체 SubsetQueries + LoaderQueries 객체에서 전체 결과 맵 생성
152
+ *
153
+ * @template TSubsetMap - Subset 함수들의 모음 객체
154
+ * @template TLoaderMap - Loader 배열들의 모음 객체
155
+ *
156
+ * @example
157
+ * type Result = InferAllSubsets<
158
+ * { A: () => Puri<...>, P: () => Puri<...> },
159
+ * { A: [...loaders], P: [] }
160
+ * >
161
+ * // { A: InferredTypeA; P: InferredTypeP }
162
+ */
163
+ export type InferAllSubsets<
164
+ TSubsetMap extends Record<string, (...args: any) => any>,
165
+ TLoaderMap extends Partial<Record<string, GenericPuriLoader[]>>,
166
+ > = {
167
+ [K in keyof TSubsetMap]: InferSubsetWithLoaders<
168
+ TSubsetMap[K],
169
+ K extends keyof TLoaderMap ? TLoaderMap[K] : undefined
170
+ >;
171
+ };
172
+
173
+ // ============================================================================
174
+ // Clear 문
175
+ // ============================================================================
176
+
177
+ /**
178
+ * Knex QueryBuilder에서 clear 가능한 statement 목록
179
+ */
180
+ export type ClearStatements =
181
+ | "with"
182
+ | "select"
183
+ | "columns"
184
+ | "hintComments"
185
+ | "where"
186
+ | "union"
187
+ | "using"
188
+ | "join"
189
+ | "group"
190
+ | "order"
191
+ | "having"
192
+ | "limit"
193
+ | "offset"
194
+ | "counter"
195
+ | "counters";