sonamu 0.6.0 → 0.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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 +227 -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 +386 -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 +55 -30
  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 +261 -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 +459 -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,16 +1,30 @@
1
- import { DateTime } from "luxon";
2
- import { chunk, groupBy, isObject, omit, set, uniq } from "lodash-es";
3
- import { DB } from "./db.js";
4
- import { isCustomJoinClause } from "../types/types.js";
1
+ import assert from "assert";
5
2
  import inflection from "inflection";
6
- import chalk from "chalk";
7
- import { UpsertBuilder } from "./upsert-builder.js";
8
- import SqlParser from "node-sql-parser";
9
- import { getTableName, getTableNamesFromWhere } from "../utils/sql-parser.js";
3
+ import { group, isObject, omit, set, unique } from "radashi";
4
+ import { Sonamu } from "../api/index.js";
5
+ import { isCustomJoinClause } from "../types/types.js";
6
+ import { getJoinTables, getTableNamesFromWhere } from "../utils/sql-parser.js";
7
+ import { chunk } from "../utils/utils.js";
8
+ import { DB } from "./db.js";
9
+ import { Puri } from "./puri.js";
10
10
  import { PuriWrapper } from "./puri-wrapper.js";
11
- export class BaseModelClass {
11
+ import { UpsertBuilder } from "./upsert-builder.js";
12
+ /**
13
+ * 모든 Model 클래스의 기본 클래스
14
+ *
15
+ * @template TSubsetKey - 서브셋 키 유니온 (예: "A" | "P" | "SS")
16
+ * @template TSubsetMapping - 서브셋별 최종 결과 타입 매핑
17
+ * @template TSubsetQueries - 서브셋 쿼리 함수 객체
18
+ * @template TLoaderQueries - 서브셋별 로더 쿼리 배열 객체
19
+ */ export class BaseModelClass {
20
+ subsetQueries;
21
+ loaderQueries;
12
22
  modelName = "Unknown";
13
- /* DB 인스턴스 get, destroy */ getDB(which) {
23
+ constructor(subsetQueries, loaderQueries){
24
+ this.subsetQueries = subsetQueries;
25
+ this.loaderQueries = loaderQueries;
26
+ }
27
+ getDB(which) {
14
28
  return DB.getDB(which);
15
29
  }
16
30
  getPuri(which) {
@@ -26,16 +40,13 @@ export class BaseModelClass {
26
40
  async destroy() {
27
41
  return DB.destroy();
28
42
  }
29
- myNow(timestamp) {
30
- const dt = timestamp === undefined ? DateTime.local() : DateTime.fromSeconds(timestamp);
31
- return dt.toFormat("yyyy-MM-dd HH:mm:ss");
32
- }
33
43
  async getInsertedIds(wdb, rows, tableName, unqKeyFields, chunkSize = 500) {
34
44
  if (!wdb) {
35
45
  wdb = this.getDB("w");
36
46
  }
37
47
  let unqKeys;
38
- let whereInField, selectField;
48
+ let whereInField;
49
+ let selectField;
39
50
  if (unqKeyFields.length > 1) {
40
51
  whereInField = wdb.raw(`CONCAT_WS('_', '${unqKeyFields.join(",")}')`);
41
52
  selectField = `${whereInField} as tmpUid`;
@@ -45,96 +56,175 @@ export class BaseModelClass {
45
56
  selectField = unqKeyFields[0];
46
57
  unqKeys = rows.map((row)=>row[unqKeyFields[0]]);
47
58
  }
48
- const chunks = chunk(unqKeys, chunkSize);
49
59
  let resultIds = [];
50
- for (let chunk of chunks){
51
- const dbRows = await wdb(tableName).select("id", wdb.raw(selectField)).whereIn(whereInField, chunk);
52
- resultIds = resultIds.concat(dbRows.map((dbRow)=>parseInt(dbRow.id)));
60
+ for (const items of chunk(unqKeys, chunkSize)){
61
+ const dbRows = await wdb(tableName).select("id", wdb.raw(selectField)).whereIn(whereInField, items);
62
+ resultIds = resultIds.concat(dbRows.map((dbRow)=>parseInt(String(dbRow.id))));
53
63
  }
54
64
  return resultIds;
55
65
  }
56
- async useLoaders(db, rows, loaders) {
57
- if (loaders.length === 0) {
58
- return rows;
66
+ /**
67
+ * 특정 서브셋에 대한 쿼리 빌더 획득
68
+ *
69
+ * @returns qb - 쿼리 빌더 (조건 추가용)
70
+ * @returns onSubset - 특정 서브셋 전용 타입이 필요할 때 사용
71
+ */ getSubsetQueries(subset) {
72
+ if (!this.subsetQueries) {
73
+ throw new Error("subsetQueries is not defined");
59
74
  }
60
- for (let loader of loaders){
61
- let subQ;
62
- let subRows;
63
- let toCol;
64
- const fromIds = rows.map((row)=>row[loader.manyJoin.idField]);
65
- if (loader.manyJoin.through === undefined) {
66
- // HasMany
67
- const idColumn = `${loader.manyJoin.toTable}.${loader.manyJoin.toCol}`;
68
- subQ = db(loader.manyJoin.toTable).whereIn(idColumn, fromIds).select([
69
- ...loader.select,
70
- idColumn
71
- ]);
72
- // HasMany에서 OneJoin이 있는 경우
73
- loader.oneJoins.map((join)=>{
74
- if (join.join == "inner") {
75
- subQ.innerJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
76
- } else if (join.join == "outer") {
77
- subQ.leftOuterJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
78
- }
79
- });
80
- toCol = loader.manyJoin.toCol;
81
- } else {
82
- // ManyToMany
83
- const idColumn = `${loader.manyJoin.through.table}.${loader.manyJoin.through.fromCol}`;
84
- subQ = db(loader.manyJoin.through.table).join(loader.manyJoin.toTable, `${loader.manyJoin.through.table}.${loader.manyJoin.through.toCol}`, `${loader.manyJoin.toTable}.${loader.manyJoin.toCol}`).whereIn(idColumn, fromIds).select(uniq([
85
- ...loader.select,
86
- idColumn
87
- ]));
88
- // ManyToMany에서 OneJoin이 있는 경우
89
- loader.oneJoins.map((join)=>{
90
- if (join.join == "inner") {
91
- subQ.innerJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
92
- } else if (join.join == "outer") {
93
- subQ.leftOuterJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
94
- }
95
- });
96
- toCol = loader.manyJoin.through.fromCol;
75
+ const puriWrapper = new PuriWrapper(this.getDB("r"), new UpsertBuilder());
76
+ const qb = this.subsetQueries[subset]?.(puriWrapper);
77
+ return {
78
+ qb: qb,
79
+ onSubset: (_subset)=>qb
80
+ };
81
+ }
82
+ /**
83
+ * Enhancer 객체 생성 헬퍼
84
+ * 타입 검증 및 추론을 도와줌
85
+ */ createEnhancers(enhancers) {
86
+ return enhancers;
87
+ }
88
+ /**
89
+ * 서브셋 쿼리 실행
90
+ *
91
+ * 1. 쿼리 실행 (pagination 적용)
92
+ * 2. 로더 실행 (1:N, N:M 관계 데이터 로딩)
93
+ * 3. Hydrate (flat → 중첩 객체)
94
+ * 4. Enhancer 적용 (virtual 필드 계산)
95
+ */ async executeSubsetQuery(params) {
96
+ const { subset, qb, params: queryParams, debug = false, optimizeCountQuery = false } = params;
97
+ if (!this.loaderQueries) {
98
+ throw new Error("loaderQueries is not defined");
99
+ }
100
+ if (!queryParams.num || !queryParams.page) {
101
+ throw new Error("num and page are required");
102
+ }
103
+ const { num, page } = queryParams;
104
+ // COUNT 쿼리 실행
105
+ const total = await this.executeCountQuery(qb, queryParams, debug, optimizeCountQuery);
106
+ // LIST 쿼리 실행
107
+ const computedRows = await this.executeListQuery(subset, qb, queryParams, num, page, debug);
108
+ // Enhancer 적용
109
+ const enhancer = params.enhancers?.[subset];
110
+ const rows = await Promise.all(computedRows.map((row)=>enhancer?.(row) ?? row));
111
+ return {
112
+ rows,
113
+ total
114
+ };
115
+ }
116
+ /**
117
+ * COUNT 쿼리 실행 (내부 메서드)
118
+ */ async executeCountQuery(qb, params, debug, optimizeCountQuery) {
119
+ if (params.queryMode === "list") {
120
+ return 0;
121
+ }
122
+ const countPuri = qb.clone().clear("order").clear("limit").clear("offset");
123
+ if (optimizeCountQuery) {
124
+ const { default: SqlParser } = await import("node-sql-parser");
125
+ const parser = new SqlParser.Parser();
126
+ const parsedQuery = parser.astify(countPuri.toQuery(), {
127
+ database: Sonamu.config.database.database
128
+ });
129
+ const leftJoinTables = getJoinTables(parsedQuery, [
130
+ "LEFT JOIN"
131
+ ]);
132
+ const whereTables = getTableNamesFromWhere(parsedQuery);
133
+ const tablesToRemove = leftJoinTables.filter((j)=>!whereTables.includes(j));
134
+ tablesToRemove.forEach((table)=>{
135
+ countPuri.clearJoin(table);
136
+ });
137
+ }
138
+ // COUNT(*)로 전체 레코드 수를 계산
139
+ // TODO: qb의 DISTINCT가 있는 경우 처리해야 함
140
+ const countResult = await countPuri.clear("select").select({
141
+ total: Puri.rawNumber(`COUNT(*)::integer`)
142
+ }).first();
143
+ if (debug) {
144
+ countPuri.debug();
145
+ }
146
+ return countResult?.total ?? 0;
147
+ }
148
+ /**
149
+ * LIST 쿼리 실행 (내부 메서드)
150
+ */ async executeListQuery(subset, qb, params, num, page, debug) {
151
+ if (params.queryMode === "count") {
152
+ return [];
153
+ }
154
+ let unloadedRows = await qb.limit(num).offset(num * (page - 1));
155
+ if (debug) {
156
+ qb.debug();
157
+ }
158
+ // 로더 처리
159
+ const loaders = this.loaderQueries[subset];
160
+ if (loaders && Array.isArray(loaders)) {
161
+ unloadedRows = await this.processLoaders(unloadedRows, loaders, debug);
162
+ }
163
+ return this.hydrate(unloadedRows);
164
+ }
165
+ /**
166
+ * 재귀적 로더 처리
167
+ */ async processLoaders(rows, loaders, debug) {
168
+ for (const resolveLoader of loaders){
169
+ const { as, refId, qb: resolveLoaderQbFn, loaders: nestedLoaders } = resolveLoader;
170
+ const resolveLoaderQb = resolveLoaderQbFn(new PuriWrapper(this.getDB("r"), new UpsertBuilder()), rows.map((row)=>row[refId]));
171
+ if (debug) {
172
+ resolveLoaderQb.debug();
97
173
  }
98
- subRows = await subQ;
99
- if (loader.loaders) {
100
- // 추가 -Many 케이스가 있는 경우 recursion 처리
101
- subRows = await this.useLoaders(db, subRows, loader.loaders);
174
+ let loadedRows = await resolveLoaderQb;
175
+ // 중첩 loaders 있으면 재귀 처리
176
+ if (nestedLoaders && nestedLoaders.length > 0) {
177
+ loadedRows = await this.processLoaders(loadedRows, nestedLoaders, debug);
102
178
  }
103
- // 불러온 row들을 참조ID 기준으로 분류 배치
104
- const subRowGroups = groupBy(subRows, toCol);
179
+ const subRowGroups = group(loadedRows, (row)=>row.refId);
105
180
  rows = rows.map((row)=>{
106
- row[loader.as] = (subRowGroups[row[loader.manyJoin.idField]] ?? []).map((r)=>omit(r, toCol));
181
+ row[as] = (subRowGroups[row[refId]] ?? []).map((r)=>omit(r, [
182
+ "refId"
183
+ ]));
107
184
  return row;
108
185
  });
109
186
  }
110
187
  return rows;
111
188
  }
112
- hydrate(rows) {
189
+ /**
190
+ * Flat 레코드를 중첩 객체로 변환
191
+ *
192
+ * - `user__name` → `{ user: { name } }`
193
+ * - nullable relation의 경우 모든 필드가 null이면 객체 자체를 null로
194
+ */ hydrate(rows) {
113
195
  return rows.map((row)=>{
114
- // nullable relation 경우 관련된 필드가 전부 null 생성되는 것 방지하는 코드
196
+ // nullable relation 처리: 관련 필드가 전부 null 경우 방지
115
197
  const nestedKeys = Object.keys(row).filter((key)=>key.includes("__"));
116
- const groups = groupBy(nestedKeys, (key)=>key.split("__")[0]);
117
- const nullKeys = Object.keys(groups).filter((key)=>groups[key].length > 1 && groups[key].every((field)=>row[field] === null || Array.isArray(row[field]) && row[field].length === 0));
198
+ const groups = Object.groupBy(nestedKeys, (key)=>key.split("__")[0]);
199
+ const nullKeys = Object.entries(groups).filter(([_, data])=>data && data.length > 1 && data.every((field)=>row[field] === null || Array.isArray(row[field]) && row[field].length === 0)).map(([key])=>key);
118
200
  const hydrated = Object.keys(row).reduce((r, field)=>{
119
201
  if (!field.includes("__")) {
202
+ // 일반 필드: 배열 내 객체면 재귀 hydrate
120
203
  if (Array.isArray(row[field]) && isObject(row[field][0])) {
121
204
  r[field] = this.hydrate(row[field]);
122
- return r;
123
205
  } else {
124
206
  r[field] = row[field];
125
- return r;
126
207
  }
208
+ return r;
127
209
  }
210
+ // 중첩 필드 처리: user__name → user[name]
128
211
  const parts = field.split("__");
129
212
  const objPath = parts[0] + parts.slice(1).map((part)=>`[${part}]`).join("");
130
- set(r, objPath, row[field] && Array.isArray(row[field]) && isObject(row[field][0]) ? this.hydrate(row[field]) : row[field]);
213
+ r = set(r, objPath, row[field] && Array.isArray(row[field]) && isObject(row[field][0]) ? this.hydrate(row[field]) : row[field]);
131
214
  return r;
132
215
  }, {});
133
- nullKeys.map((nullKey)=>hydrated[nullKey] = null);
216
+ // null relation 처리
217
+ nullKeys.forEach((nullKey)=>{
218
+ hydrated[nullKey] = null;
219
+ });
134
220
  return hydrated;
135
221
  });
136
222
  }
223
+ // Legacy SubsetQuery 실행 (Puri 도입 전 호환용)
137
224
  async runSubsetQuery({ params, baseTable, subset, subsetQuery, build, afterBuild, debug, db: _db, optimizeCountQuery }) {
225
+ const chalk = (await import("chalk")).default;
226
+ const SqlParser = (await import("node-sql-parser")).default;
227
+ const { getTableName, getTableNamesFromWhere } = await import("../utils/sql-parser.js");
138
228
  const db = _db ?? this.getDB(subset.startsWith("A") ? "w" : "r");
139
229
  baseTable = baseTable ?? inflection.pluralize(inflection.underscore(this.modelName));
140
230
  const queryMode = params.queryMode ?? (params.id !== undefined ? "list" : "both");
@@ -147,10 +237,10 @@ export class BaseModelClass {
147
237
  virtual
148
238
  });
149
239
  const applyJoinClause = (qb, joins)=>{
150
- joins.map((join)=>{
151
- if (join.join == "inner") {
240
+ joins.forEach((join)=>{
241
+ if (join.join === "inner") {
152
242
  qb.innerJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
153
- } else if (join.join == "outer") {
243
+ } else if (join.join === "outer") {
154
244
  qb.leftOuterJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
155
245
  }
156
246
  });
@@ -162,12 +252,12 @@ export class BaseModelClass {
162
252
  }
163
253
  const clonedQb = qb.clone().clear("order").clear("offset").clear("limit");
164
254
  const parser = new SqlParser.Parser();
165
- // optmizeCountQuery가 true인 경우 다른 clause에 영향을 주지 않는 모든 join을 제외함
166
255
  if (optimizeCountQuery) {
167
- const parsedQuery = parser.astify(clonedQb.toQuery());
256
+ const parsedQuery = parser.astify(clonedQb.toQuery(), {
257
+ database: Sonamu.config.database.database
258
+ });
168
259
  const tables = getTableNamesFromWhere(parsedQuery);
169
- // where절에 사용되는 테이블의 조인을 위해 사용되는 테이블
170
- const needToJoin = uniq(tables.flatMap((table)=>table.split("__").map((t)=>inflection.pluralize(t))));
260
+ const needToJoin = unique(tables.flatMap((table)=>table.split("__").map((t)=>inflection.pluralize(t))));
171
261
  applyJoinClause(clonedQb, joins.filter((j)=>needToJoin.includes(j.table)));
172
262
  } else {
173
263
  applyJoinClause(clonedQb, joins);
@@ -179,7 +269,9 @@ export class BaseModelClass {
179
269
  joins,
180
270
  virtual
181
271
  }) ?? clonedQb;
182
- const parsedQuery = parser.astify(processedQb.toQuery());
272
+ const parsedQuery = parser.astify(processedQb.toQuery(), {
273
+ database: Sonamu.config.database.database
274
+ });
183
275
  const q = Array.isArray(parsedQuery) ? parsedQuery[0] : parsedQuery;
184
276
  if (q.type !== "select") {
185
277
  throw new Error("Invalid query");
@@ -188,7 +280,6 @@ export class BaseModelClass {
188
280
  as: "total"
189
281
  }).first();
190
282
  const countRow = await countQuery;
191
- // debug: countQuery
192
283
  if (debug === true || debug === "count") {
193
284
  console.debug("DEBUG: count query", chalk.blue(countQuery.toQuery().toString()));
194
285
  }
@@ -199,14 +290,12 @@ export class BaseModelClass {
199
290
  if (queryMode === "count") {
200
291
  return [];
201
292
  }
202
- // limit, offset
203
293
  if (params.num !== 0) {
294
+ assert(params.num);
204
295
  qb.limit(params.num);
205
- qb.offset(params.num * (params.page - 1));
296
+ qb.offset(params.num * ((params.page ?? 1) - 1));
206
297
  }
207
- // select, rows
208
298
  const clonedQb = qb.clone().select(select);
209
- // join
210
299
  applyJoinClause(clonedQb, joins);
211
300
  const listQuery = afterBuild?.({
212
301
  qb: clonedQb,
@@ -216,7 +305,6 @@ export class BaseModelClass {
216
305
  virtual
217
306
  }) ?? clonedQb;
218
307
  let rows = await listQuery;
219
- // debug: listQuery
220
308
  if (debug === true || debug === "list") {
221
309
  console.debug("DEBUG: list query", chalk.blue(listQuery.toQuery().toString()));
222
310
  }
@@ -231,6 +319,61 @@ export class BaseModelClass {
231
319
  qb
232
320
  };
233
321
  }
322
+ // Legacy Loader 처리 (Puri 도입 전 호환용)
323
+ async useLoaders(db, rows, loaders) {
324
+ if (loaders.length === 0) {
325
+ return rows;
326
+ }
327
+ for (const loader of loaders){
328
+ let subQ;
329
+ let subRows;
330
+ let toCol;
331
+ const fromIds = rows.map((row)=>row[loader.manyJoin.idField]);
332
+ if (loader.manyJoin.through === undefined) {
333
+ // HasMany
334
+ const idColumn = `${loader.manyJoin.toTable}.${loader.manyJoin.toCol}`;
335
+ subQ = db(loader.manyJoin.toTable).whereIn(idColumn, fromIds).select([
336
+ ...loader.select,
337
+ idColumn
338
+ ]);
339
+ loader.oneJoins.forEach((join)=>{
340
+ if (join.join === "inner") {
341
+ subQ.innerJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
342
+ } else if (join.join === "outer") {
343
+ subQ.leftOuterJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
344
+ }
345
+ });
346
+ toCol = loader.manyJoin.toCol;
347
+ } else {
348
+ // ManyToMany
349
+ const idColumn = `${loader.manyJoin.through.table}.${loader.manyJoin.through.fromCol}`;
350
+ subQ = db(loader.manyJoin.through.table).join(loader.manyJoin.toTable, `${loader.manyJoin.through.table}.${loader.manyJoin.through.toCol}`, `${loader.manyJoin.toTable}.${loader.manyJoin.toCol}`).whereIn(idColumn, fromIds).select(unique([
351
+ ...loader.select,
352
+ idColumn
353
+ ]));
354
+ loader.oneJoins.forEach((join)=>{
355
+ if (join.join === "inner") {
356
+ subQ.innerJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
357
+ } else if (join.join === "outer") {
358
+ subQ.leftOuterJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));
359
+ }
360
+ });
361
+ toCol = loader.manyJoin.through.fromCol;
362
+ }
363
+ subRows = await subQ;
364
+ if (loader.loaders) {
365
+ subRows = await this.useLoaders(db, subRows, loader.loaders);
366
+ }
367
+ const subRowGroups = group(subRows, (row)=>row[toCol]);
368
+ rows = rows.map((row)=>{
369
+ row[loader.as] = (subRowGroups[row[loader.manyJoin.idField]] ?? []).map((r)=>omit(r, [
370
+ toCol
371
+ ]));
372
+ return row;
373
+ });
374
+ }
375
+ return rows;
376
+ }
234
377
  getJoinClause(db, join) {
235
378
  if (!isCustomJoinClause(join)) {
236
379
  return db.raw(`${join.from} = ${join.to}`);
@@ -244,4 +387,4 @@ export class BaseModelClass {
244
387
  }
245
388
  export const BaseModel = new BaseModelClass();
246
389
 
247
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/base-model.ts"],"sourcesContent":["import { DateTime } from \"luxon\";\nimport { Knex } from \"knex\";\nimport { chunk, groupBy, isObject, omit, set, uniq } from \"lodash-es\";\nimport { DBPreset, DB } from \"./db\";\nimport { isCustomJoinClause, type SubsetQuery } from \"../types/types\";\nimport type { BaseListParams } from \"../utils/model\";\nimport inflection from \"inflection\";\nimport chalk from \"chalk\";\nimport { UpsertBuilder } from \"./upsert-builder\";\nimport SqlParser from \"node-sql-parser\";\nimport { getTableName, getTableNamesFromWhere } from \"../utils/sql-parser\";\nimport { PuriWrapper } from \"./puri-wrapper\";\n\nexport class BaseModelClass {\n  public modelName: string = \"Unknown\";\n\n  /* DB 인스턴스 get, destroy */\n  getDB(which: DBPreset): Knex {\n    return DB.getDB(which);\n  }\n\n  getPuri(which: DBPreset): PuriWrapper {\n    // 트랜잭션 컨텍스트에서 트랜잭션 획득\n    const trx = DB.getTransactionContext().getTransaction(which);\n    if (trx) {\n      return trx;\n    }\n\n    // 트랜잭션이 없으면 새로운 PuriWrapper 반환\n    const db = this.getDB(which);\n    return new PuriWrapper(db, this.getUpsertBuilder());\n  }\n\n  async destroy() {\n    return DB.destroy();\n  }\n\n  myNow(timestamp?: number): string {\n    const dt: DateTime =\n      timestamp === undefined\n        ? DateTime.local()\n        : DateTime.fromSeconds(timestamp);\n    return dt.toFormat(\"yyyy-MM-dd HH:mm:ss\");\n  }\n\n  async getInsertedIds(\n    wdb: Knex,\n    rows: any[],\n    tableName: string,\n    unqKeyFields: string[],\n    chunkSize: number = 500\n  ) {\n    if (!wdb) {\n      wdb = this.getDB(\"w\");\n    }\n\n    let unqKeys: string[];\n    let whereInField: any, selectField: string;\n    if (unqKeyFields.length > 1) {\n      whereInField = wdb.raw(`CONCAT_WS('_', '${unqKeyFields.join(\",\")}')`);\n      selectField = `${whereInField} as tmpUid`;\n      unqKeys = rows.map((row) =>\n        unqKeyFields.map((field) => row[field]).join(\"_\")\n      );\n    } else {\n      whereInField = unqKeyFields[0];\n      selectField = unqKeyFields[0];\n      unqKeys = rows.map((row) => row[unqKeyFields[0]]);\n    }\n    const chunks = chunk(unqKeys, chunkSize);\n\n    let resultIds: number[] = [];\n    for (let chunk of chunks) {\n      const dbRows = await wdb(tableName)\n        .select(\"id\", wdb.raw(selectField))\n        .whereIn(whereInField, chunk);\n      resultIds = resultIds.concat(\n        dbRows.map((dbRow: any) => parseInt(dbRow.id))\n      );\n    }\n\n    return resultIds;\n  }\n\n  async useLoaders(db: Knex, rows: any[], loaders: SubsetQuery[\"loaders\"]) {\n    if (loaders.length === 0) {\n      return rows;\n    }\n\n    for (let loader of loaders) {\n      let subQ: any;\n      let subRows: any[];\n      let toCol: string;\n\n      const fromIds = rows.map((row) => row[loader.manyJoin.idField]);\n\n      if (loader.manyJoin.through === undefined) {\n        // HasMany\n        const idColumn = `${loader.manyJoin.toTable}.${loader.manyJoin.toCol}`;\n        subQ = db(loader.manyJoin.toTable)\n          .whereIn(idColumn, fromIds)\n          .select([...loader.select, idColumn]);\n\n        // HasMany에서 OneJoin이 있는 경우\n        loader.oneJoins.map((join) => {\n          if (join.join == \"inner\") {\n            subQ.innerJoin(\n              `${join.table} as ${join.as}`,\n              this.getJoinClause(db, join)\n            );\n          } else if (join.join == \"outer\") {\n            subQ.leftOuterJoin(\n              `${join.table} as ${join.as}`,\n              this.getJoinClause(db, join)\n            );\n          }\n        });\n        toCol = loader.manyJoin.toCol;\n      } else {\n        // ManyToMany\n        const idColumn = `${loader.manyJoin.through.table}.${loader.manyJoin.through.fromCol}`;\n        subQ = db(loader.manyJoin.through.table)\n          .join(\n            loader.manyJoin.toTable,\n            `${loader.manyJoin.through.table}.${loader.manyJoin.through.toCol}`,\n            `${loader.manyJoin.toTable}.${loader.manyJoin.toCol}`\n          )\n          .whereIn(idColumn, fromIds)\n          .select(uniq([...loader.select, idColumn]));\n\n        // ManyToMany에서 OneJoin이 있는 경우\n        loader.oneJoins.map((join) => {\n          if (join.join == \"inner\") {\n            subQ.innerJoin(\n              `${join.table} as ${join.as}`,\n              this.getJoinClause(db, join)\n            );\n          } else if (join.join == \"outer\") {\n            subQ.leftOuterJoin(\n              `${join.table} as ${join.as}`,\n              this.getJoinClause(db, join)\n            );\n          }\n        });\n        toCol = loader.manyJoin.through.fromCol;\n      }\n      subRows = await subQ;\n\n      if (loader.loaders) {\n        // 추가 -Many 케이스가 있는 경우 recursion 처리\n        subRows = await this.useLoaders(db, subRows, loader.loaders);\n      }\n\n      // 불러온 row들을 참조ID 기준으로 분류 배치\n      const subRowGroups = groupBy(subRows, toCol);\n      rows = rows.map((row) => {\n        row[loader.as] = (subRowGroups[row[loader.manyJoin.idField]] ?? []).map(\n          (r) => omit(r, toCol)\n        );\n        return row;\n      });\n    }\n    return rows;\n  }\n\n  hydrate<T>(rows: T[]): T[] {\n    return rows.map((row: any) => {\n      // nullable relation인 경우 관련된 필드가 전부 null로 생성되는 것 방지하는 코드\n      const nestedKeys = Object.keys(row).filter((key) => key.includes(\"__\"));\n      const groups = groupBy(nestedKeys, (key) => key.split(\"__\")[0]);\n      const nullKeys = Object.keys(groups).filter(\n        (key) =>\n          groups[key].length > 1 &&\n          groups[key].every(\n            (field) =>\n              row[field] === null ||\n              (Array.isArray(row[field]) && row[field].length === 0)\n          )\n      );\n\n      const hydrated = Object.keys(row).reduce((r, field) => {\n        if (!field.includes(\"__\")) {\n          if (Array.isArray(row[field]) && isObject(row[field][0])) {\n            r[field] = this.hydrate(row[field]);\n            return r;\n          } else {\n            r[field] = row[field];\n            return r;\n          }\n        }\n\n        const parts = field.split(\"__\");\n        const objPath =\n          parts[0] +\n          parts\n            .slice(1)\n            .map((part) => `[${part}]`)\n            .join(\"\");\n        set(\n          r,\n          objPath,\n          row[field] && Array.isArray(row[field]) && isObject(row[field][0])\n            ? this.hydrate(row[field])\n            : row[field]\n        );\n\n        return r;\n      }, {} as any);\n      nullKeys.map((nullKey) => (hydrated[nullKey] = null));\n\n      return hydrated;\n    });\n  }\n\n  async runSubsetQuery<T extends BaseListParams, U extends string>({\n    params,\n    baseTable,\n    subset,\n    subsetQuery,\n    build,\n    afterBuild,\n    debug,\n    db: _db,\n    optimizeCountQuery,\n  }: {\n    subset: U;\n    params: T;\n    subsetQuery: SubsetQuery;\n    build: (buildParams: {\n      qb: Knex.QueryBuilder;\n      db: Knex;\n      select: (string | Knex.Raw)[];\n      joins: SubsetQuery[\"joins\"];\n      virtual: string[];\n    }) => Knex.QueryBuilder;\n    afterBuild?: (buildParams: {\n      qb: Knex.QueryBuilder;\n      db: Knex;\n      select: (string | Knex.Raw)[];\n      joins: SubsetQuery[\"joins\"];\n      virtual: string[];\n    }) => Knex.QueryBuilder;\n    baseTable?: string;\n    debug?: boolean | \"list\" | \"count\";\n    db?: Knex;\n    optimizeCountQuery?: boolean;\n  }): Promise<{\n    rows: any[];\n    total?: number | undefined;\n    subsetQuery: SubsetQuery;\n    qb: Knex.QueryBuilder;\n  }> {\n    const db = _db ?? this.getDB(subset.startsWith(\"A\") ? \"w\" : \"r\");\n    baseTable =\n      baseTable ?? inflection.pluralize(inflection.underscore(this.modelName));\n    const queryMode =\n      params.queryMode ?? (params.id !== undefined ? \"list\" : \"both\");\n\n    const { select, virtual, joins, loaders } = subsetQuery;\n    const qb = build({\n      qb: db.from(baseTable),\n      db,\n      select,\n      joins,\n      virtual,\n    });\n\n    const applyJoinClause = (\n      qb: Knex.QueryBuilder,\n      joins: SubsetQuery[\"joins\"]\n    ) => {\n      joins.map((join) => {\n        if (join.join == \"inner\") {\n          qb.innerJoin(\n            `${join.table} as ${join.as}`,\n            this.getJoinClause(db, join)\n          );\n        } else if (join.join == \"outer\") {\n          qb.leftOuterJoin(\n            `${join.table} as ${join.as}`,\n            this.getJoinClause(db, join)\n          );\n        }\n      });\n    };\n\n    // countQuery\n    const total = await (async () => {\n      if (queryMode === \"list\") {\n        return undefined;\n      }\n\n      const clonedQb = qb.clone().clear(\"order\").clear(\"offset\").clear(\"limit\");\n      const parser = new SqlParser.Parser();\n\n      // optmizeCountQuery가 true인 경우 다른 clause에 영향을 주지 않는 모든 join을 제외함\n      if (optimizeCountQuery) {\n        const parsedQuery = parser.astify(clonedQb.toQuery());\n        const tables = getTableNamesFromWhere(parsedQuery);\n        // where절에 사용되는 테이블의 조인을 위해 사용되는 테이블\n        const needToJoin = uniq(\n          tables.flatMap((table) =>\n            table.split(\"__\").map((t) => inflection.pluralize(t))\n          )\n        );\n        applyJoinClause(\n          clonedQb,\n          joins.filter((j) => needToJoin.includes(j.table))\n        );\n      } else {\n        applyJoinClause(clonedQb, joins);\n      }\n\n      const processedQb =\n        afterBuild?.({\n          qb: clonedQb,\n          db,\n          select,\n          joins,\n          virtual,\n        }) ?? clonedQb;\n\n      const parsedQuery = parser.astify(processedQb.toQuery());\n      const q = Array.isArray(parsedQuery) ? parsedQuery[0] : parsedQuery;\n      if (q.type !== \"select\") {\n        throw new Error(\"Invalid query\");\n      }\n\n      const countQuery =\n        q.distinct !== null\n          ? clonedQb\n              .clear(\"select\")\n              .select(\n                db.raw(\n                  `COUNT(DISTINCT \\`${getTableName(q.columns[0].expr)}\\`.\\`${q.columns[0].expr.column}\\`) as total`\n                )\n              )\n              .first()\n          : clonedQb.clear(\"select\").count(\"*\", { as: \"total\" }).first();\n      const countRow: { total?: number } = await countQuery;\n\n      // debug: countQuery\n      if (debug === true || debug === \"count\") {\n        console.debug(\n          \"DEBUG: count query\",\n          chalk.blue(countQuery.toQuery().toString())\n        );\n      }\n\n      return countRow?.total ?? 0;\n    })();\n\n    // listQuery\n    const rows = await (async () => {\n      if (queryMode === \"count\") {\n        return [];\n      }\n\n      // limit, offset\n      if (params.num !== 0) {\n        qb.limit(params.num!);\n        qb.offset(params.num! * (params.page! - 1));\n      }\n\n      // select, rows\n      const clonedQb = qb.clone().select(select);\n\n      // join\n      applyJoinClause(clonedQb, joins);\n\n      const listQuery =\n        afterBuild?.({\n          qb: clonedQb,\n          db,\n          select,\n          joins,\n          virtual,\n        }) ?? clonedQb;\n\n      let rows = await listQuery;\n      // debug: listQuery\n      if (debug === true || debug === \"list\") {\n        console.debug(\n          \"DEBUG: list query\",\n          chalk.blue(listQuery.toQuery().toString())\n        );\n      }\n\n      rows = await this.useLoaders(db, rows, loaders);\n      rows = this.hydrate(rows);\n      return rows;\n    })();\n\n    return { rows, total, subsetQuery, qb };\n  }\n\n  getJoinClause(\n    db: Knex<any, unknown>,\n    join: SubsetQuery[\"joins\"][number]\n  ): Knex.Raw<any> {\n    if (!isCustomJoinClause(join)) {\n      return db.raw(`${join.from} = ${join.to}`);\n    } else {\n      return db.raw(join.custom);\n    }\n  }\n\n  getUpsertBuilder(): UpsertBuilder {\n    return new UpsertBuilder();\n  }\n}\nexport const BaseModel = new BaseModelClass();\n"],"names":["DateTime","chunk","groupBy","isObject","omit","set","uniq","DB","isCustomJoinClause","inflection","chalk","UpsertBuilder","SqlParser","getTableName","getTableNamesFromWhere","PuriWrapper","BaseModelClass","modelName","getDB","which","getPuri","trx","getTransactionContext","getTransaction","db","getUpsertBuilder","destroy","myNow","timestamp","dt","undefined","local","fromSeconds","toFormat","getInsertedIds","wdb","rows","tableName","unqKeyFields","chunkSize","unqKeys","whereInField","selectField","length","raw","join","map","row","field","chunks","resultIds","dbRows","select","whereIn","concat","dbRow","parseInt","id","useLoaders","loaders","loader","subQ","subRows","toCol","fromIds","manyJoin","idField","through","idColumn","toTable","oneJoins","innerJoin","table","as","getJoinClause","leftOuterJoin","fromCol","subRowGroups","r","hydrate","nestedKeys","Object","keys","filter","key","includes","groups","split","nullKeys","every","Array","isArray","hydrated","reduce","parts","objPath","slice","part","nullKey","runSubsetQuery","params","baseTable","subset","subsetQuery","build","afterBuild","debug","_db","optimizeCountQuery","startsWith","pluralize","underscore","queryMode","virtual","joins","qb","from","applyJoinClause","total","clonedQb","clone","clear","parser","Parser","parsedQuery","astify","toQuery","tables","needToJoin","flatMap","t","j","processedQb","q","type","Error","countQuery","distinct","columns","expr","column","first","count","countRow","console","blue","toString","num","limit","offset","page","listQuery","to","custom","BaseModel"],"mappings":"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"}
390
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/base-model.ts"],"sourcesContent":["import assert from \"assert\";\nimport inflection from \"inflection\";\nimport type { Knex } from \"knex\";\nimport { group, isObject, omit, set, unique } from \"radashi\";\nimport { Sonamu } from \"../api\";\nimport { type DatabaseSchemaExtend, isCustomJoinClause, type SubsetQuery } from \"../types/types\";\nimport type { BaseListParams } from \"../utils/model\";\nimport { getJoinTables, getTableNamesFromWhere } from \"../utils/sql-parser\";\nimport { chunk } from \"../utils/utils\";\nimport type {\n  EnhancerMap,\n  ExecuteSubsetQueryResult,\n  ResolveSubsetIntersection,\n  UnionExtractedTTables,\n} from \"./base-model.types\";\nimport type { DBPreset } from \"./db\";\nimport { DB } from \"./db\";\nimport { Puri } from \"./puri\";\nimport type { InferAllSubsets, PuriLoaderQueries, PuriSubsetFn } from \"./puri-subset.types\";\nimport { PuriWrapper } from \"./puri-wrapper\";\nimport { UpsertBuilder } from \"./upsert-builder\";\n\ntype UnknownDBRecord = Record<string, unknown>;\n\n/**\n * 모든 Model 클래스의 기본 클래스\n *\n * @template TSubsetKey - 서브셋 키 유니온 (예: \"A\" | \"P\" | \"SS\")\n * @template TSubsetMapping - 서브셋별 최종 결과 타입 매핑\n * @template TSubsetQueries - 서브셋 쿼리 함수 객체\n * @template TLoaderQueries - 서브셋별 로더 쿼리 배열 객체\n */\nexport class BaseModelClass<\n  TSubsetKey extends string = never,\n  TSubsetMapping extends Record<string, any> = never,\n  TSubsetQueries extends Record<TSubsetKey, PuriSubsetFn> = never,\n  TLoaderQueries extends PuriLoaderQueries<TSubsetKey> = never,\n> {\n  public modelName: string = \"Unknown\";\n\n  constructor(\n    protected subsetQueries?: TSubsetQueries,\n    protected loaderQueries?: TLoaderQueries,\n  ) {}\n\n  getDB(which: DBPreset): Knex {\n    return DB.getDB(which);\n  }\n\n  getPuri(which: DBPreset): PuriWrapper {\n    // 트랜잭션 컨텍스트에서 트랜잭션 획득\n    const trx = DB.getTransactionContext().getTransaction(which);\n    if (trx) {\n      return trx;\n    }\n\n    // 트랜잭션이 없으면 새로운 PuriWrapper 반환\n    const db = this.getDB(which);\n    return new PuriWrapper(db, this.getUpsertBuilder());\n  }\n\n  async destroy() {\n    return DB.destroy();\n  }\n\n  async getInsertedIds(\n    wdb: Knex,\n    rows: UnknownDBRecord[],\n    tableName: string,\n    unqKeyFields: string[],\n    chunkSize: number = 500,\n  ) {\n    if (!wdb) {\n      wdb = this.getDB(\"w\");\n    }\n\n    let unqKeys: string[];\n    let whereInField: string | Knex.Raw;\n    let selectField: string;\n\n    if (unqKeyFields.length > 1) {\n      whereInField = wdb.raw(`CONCAT_WS('_', '${unqKeyFields.join(\",\")}')`);\n      selectField = `${whereInField} as tmpUid`;\n      unqKeys = rows.map((row) => unqKeyFields.map((field) => row[field]).join(\"_\"));\n    } else {\n      whereInField = unqKeyFields[0];\n      selectField = unqKeyFields[0];\n      unqKeys = rows.map((row) => row[unqKeyFields[0]] as string);\n    }\n\n    let resultIds: number[] = [];\n    for (const items of chunk(unqKeys, chunkSize)) {\n      const dbRows = await wdb(tableName)\n        .select(\"id\", wdb.raw(selectField))\n        .whereIn(whereInField as string, items);\n      resultIds = resultIds.concat(\n        dbRows.map((dbRow: UnknownDBRecord) => parseInt(String(dbRow.id))),\n      );\n    }\n\n    return resultIds;\n  }\n\n  /**\n   * 특정 서브셋에 대한 쿼리 빌더 획득\n   *\n   * @returns qb - 쿼리 빌더 (조건 추가용)\n   * @returns onSubset - 특정 서브셋 전용 타입이 필요할 때 사용\n   */\n  getSubsetQueries<T extends TSubsetKey>(subset: T) {\n    if (!this.subsetQueries) {\n      throw new Error(\"subsetQueries is not defined\");\n    }\n\n    const puriWrapper = new PuriWrapper(this.getDB(\"r\"), new UpsertBuilder());\n    const qb = this.subsetQueries[subset]?.(puriWrapper);\n\n    // NonAllowedAsSingleTable: 단일 테이블 컬럼 접근 방지용 마커\n    type QBTables = UnionExtractedTTables<TSubsetKey, TSubsetQueries> & {\n      NonAllowedAsSingleTable: { __fulltext__: true };\n    };\n\n    return {\n      qb: qb as unknown as Puri<DatabaseSchemaExtend, QBTables, {}>,\n      onSubset: ((_subset: TSubsetKey | readonly TSubsetKey[]) => qb) as {\n        // 단일 키\n        <S extends TSubsetKey>(subset: S): ReturnType<TSubsetQueries[S]>;\n        // 키 배열 -> 교집합 반환\n        <Arr extends readonly TSubsetKey[]>(\n          subsets: [...Arr],\n        ): ResolveSubsetIntersection<Arr, TSubsetQueries>;\n      },\n    };\n  }\n\n  /**\n   * Enhancer 객체 생성 헬퍼\n   * 타입 검증 및 추론을 도와줌\n   */\n  createEnhancers<T extends TSubsetKey>(\n    enhancers: EnhancerMap<T, InferAllSubsets<TSubsetQueries, TLoaderQueries>, TSubsetMapping>,\n  ) {\n    return enhancers;\n  }\n\n  /**\n   * 서브셋 쿼리 실행\n   *\n   * 1. 쿼리 실행 (pagination 적용)\n   * 2. 로더 실행 (1:N, N:M 관계 데이터 로딩)\n   * 3. Hydrate (flat → 중첩 객체)\n   * 4. Enhancer 적용 (virtual 필드 계산)\n   */\n  async executeSubsetQuery<\n    T extends TSubsetKey,\n    TComputedResults extends InferAllSubsets<TSubsetQueries, TLoaderQueries>,\n  >(\n    params: {\n      subset: T;\n      qb: Puri<any, any, any>;\n      params: {\n        num?: number;\n        page?: number;\n        queryMode?: \"list\" | \"count\" | \"both\";\n      };\n      debug?: boolean;\n      optimizeCountQuery?: boolean;\n    } & EnhancerParam<TSubsetKey, TComputedResults, TSubsetMapping>,\n  ): Promise<ExecuteSubsetQueryResult<TSubsetMapping, T>> {\n    const { subset, qb, params: queryParams, debug = false, optimizeCountQuery = false } = params;\n\n    if (!this.loaderQueries) {\n      throw new Error(\"loaderQueries is not defined\");\n    }\n\n    if (!queryParams.num || !queryParams.page) {\n      throw new Error(\"num and page are required\");\n    }\n\n    const { num, page } = queryParams;\n\n    // COUNT 쿼리 실행\n    const total = await this.executeCountQuery(qb, queryParams, debug, optimizeCountQuery);\n\n    // LIST 쿼리 실행\n    const computedRows = await this.executeListQuery(subset, qb, queryParams, num, page, debug);\n\n    // Enhancer 적용\n    const enhancer = (params as any).enhancers?.[subset];\n    const rows = (await Promise.all(\n      computedRows.map((row) => enhancer?.(row) ?? row),\n    )) as TSubsetMapping[T][];\n\n    return { rows, total };\n  }\n\n  /**\n   * COUNT 쿼리 실행 (내부 메서드)\n   */\n  private async executeCountQuery(\n    qb: Puri<any, any, any>,\n    params: { queryMode?: \"list\" | \"count\" | \"both\" },\n    debug: boolean,\n    optimizeCountQuery: boolean,\n  ): Promise<number> {\n    if (params.queryMode === \"list\") {\n      return 0;\n    }\n\n    const countPuri = qb.clone().clear(\"order\").clear(\"limit\").clear(\"offset\");\n\n    if (optimizeCountQuery) {\n      const { default: SqlParser } = await import(\"node-sql-parser\");\n      const parser = new SqlParser.Parser();\n      const parsedQuery = parser.astify(countPuri.toQuery(), {\n        database: Sonamu.config.database.database,\n      });\n\n      const leftJoinTables = getJoinTables(parsedQuery, [\"LEFT JOIN\"]);\n      const whereTables = getTableNamesFromWhere(parsedQuery);\n\n      const tablesToRemove = leftJoinTables.filter((j) => !whereTables.includes(j));\n      tablesToRemove.forEach((table) => {\n        countPuri.clearJoin(table);\n      });\n    }\n\n    // COUNT(*)로 전체 레코드 수를 계산\n    // TODO: qb의 DISTINCT가 있는 경우 처리해야 함\n    const countResult: { total?: number } = await countPuri\n      .clear(\"select\")\n      .select({ total: Puri.rawNumber(`COUNT(*)::integer`) })\n      .first();\n\n    if (debug) {\n      countPuri.debug();\n    }\n\n    return countResult?.total ?? 0;\n  }\n\n  /**\n   * LIST 쿼리 실행 (내부 메서드)\n   */\n  private async executeListQuery<T extends TSubsetKey>(\n    subset: T,\n    qb: Puri<any, any, any>,\n    params: { queryMode?: \"list\" | \"count\" | \"both\" },\n    num: number,\n    page: number,\n    debug: boolean,\n  ): Promise<any[]> {\n    if (params.queryMode === \"count\") {\n      return [];\n    }\n\n    let unloadedRows = (await qb.limit(num).offset(num * (page - 1))) as any[];\n\n    if (debug) {\n      qb.debug();\n    }\n\n    // 로더 처리\n    const loaders = (this.loaderQueries as any)[subset];\n    if (loaders && Array.isArray(loaders)) {\n      unloadedRows = await this.processLoaders(unloadedRows, loaders, debug);\n    }\n\n    return this.hydrate(unloadedRows);\n  }\n\n  /**\n   * 재귀적 로더 처리\n   */\n  private async processLoaders(rows: any[], loaders: any[], debug: boolean): Promise<any[]> {\n    for (const resolveLoader of loaders) {\n      const { as, refId, qb: resolveLoaderQbFn, loaders: nestedLoaders } = resolveLoader;\n\n      const resolveLoaderQb = resolveLoaderQbFn(\n        new PuriWrapper(this.getDB(\"r\"), new UpsertBuilder()),\n        rows.map((row) => row[refId]),\n      );\n\n      if (debug) {\n        resolveLoaderQb.debug();\n      }\n\n      let loadedRows = (await resolveLoaderQb) as any[];\n\n      // 중첩 loaders가 있으면 재귀 처리\n      if (nestedLoaders && nestedLoaders.length > 0) {\n        loadedRows = await this.processLoaders(loadedRows, nestedLoaders, debug);\n      }\n\n      const subRowGroups = group(loadedRows, (row) => row.refId);\n\n      rows = rows.map((row) => {\n        row[as] = (subRowGroups[row[refId]] ?? []).map((r) => omit(r, [\"refId\"]));\n        return row;\n      });\n    }\n\n    return rows;\n  }\n\n  /**\n   * Flat 레코드를 중첩 객체로 변환\n   *\n   * - `user__name` → `{ user: { name } }`\n   * - nullable relation의 경우 모든 필드가 null이면 객체 자체를 null로\n   */\n  hydrate<T extends UnknownDBRecord>(rows: T[]): T[] {\n    return rows.map((row: T) => {\n      // nullable relation 처리: 관련 필드가 전부 null인 경우 방지\n      const nestedKeys = Object.keys(row).filter((key) => key.includes(\"__\"));\n      const groups = Object.groupBy(nestedKeys, (key) => key.split(\"__\")[0]);\n      const nullKeys = Object.entries(groups)\n        .filter(\n          ([_, data]) =>\n            data &&\n            data.length > 1 &&\n            data.every(\n              (field) =>\n                row[field] === null || (Array.isArray(row[field]) && row[field].length === 0),\n            ),\n        )\n        .map(([key]) => key);\n\n      const hydrated = Object.keys(row).reduce((r, field) => {\n        if (!field.includes(\"__\")) {\n          // 일반 필드: 배열 내 객체면 재귀 hydrate\n          if (Array.isArray(row[field]) && isObject(row[field][0])) {\n            r[field] = this.hydrate(row[field]);\n          } else {\n            r[field] = row[field];\n          }\n          return r;\n        }\n\n        // 중첩 필드 처리: user__name → user[name]\n        const parts = field.split(\"__\");\n        const objPath =\n          parts[0] +\n          parts\n            .slice(1)\n            .map((part) => `[${part}]`)\n            .join(\"\");\n\n        r = set(\n          r,\n          objPath,\n          row[field] && Array.isArray(row[field]) && isObject(row[field][0])\n            ? this.hydrate(row[field])\n            : row[field],\n        );\n\n        return r;\n      }, {} as UnknownDBRecord);\n\n      // null relation 처리\n      nullKeys.forEach((nullKey) => {\n        hydrated[nullKey] = null;\n      });\n\n      return hydrated;\n    }) as T[];\n  }\n\n  // Legacy SubsetQuery 실행 (Puri 도입 전 호환용)\n  async runSubsetQuery<T extends BaseListParams, U extends string>({\n    params,\n    baseTable,\n    subset,\n    subsetQuery,\n    build,\n    afterBuild,\n    debug,\n    db: _db,\n    optimizeCountQuery,\n  }: {\n    subset: U;\n    params: T;\n    subsetQuery: SubsetQuery;\n    build: (buildParams: {\n      qb: Knex.QueryBuilder;\n      db: Knex;\n      select: (string | Knex.Raw)[];\n      joins: SubsetQuery[\"joins\"];\n      virtual: string[];\n    }) => Knex.QueryBuilder;\n    afterBuild?: (buildParams: {\n      qb: Knex.QueryBuilder;\n      db: Knex;\n      select: (string | Knex.Raw)[];\n      joins: SubsetQuery[\"joins\"];\n      virtual: string[];\n    }) => Knex.QueryBuilder;\n    baseTable?: string;\n    debug?: boolean | \"list\" | \"count\";\n    db?: Knex;\n    optimizeCountQuery?: boolean;\n  }): Promise<{\n    // biome-ignore lint/suspicious/noExplicitAny: Puri 도입 전까지 any로 유지\n    rows: any[];\n    total?: number | undefined;\n    subsetQuery: SubsetQuery;\n    qb: Knex.QueryBuilder;\n  }> {\n    const chalk = (await import(\"chalk\")).default;\n    const SqlParser = (await import(\"node-sql-parser\")).default;\n    const { getTableName, getTableNamesFromWhere } = await import(\"../utils/sql-parser\");\n\n    const db = _db ?? this.getDB(subset.startsWith(\"A\") ? \"w\" : \"r\");\n    baseTable = baseTable ?? inflection.pluralize(inflection.underscore(this.modelName));\n    const queryMode = params.queryMode ?? (params.id !== undefined ? \"list\" : \"both\");\n\n    const { select, virtual, joins, loaders } = subsetQuery;\n    const qb = build({\n      qb: db.from(baseTable),\n      db,\n      select,\n      joins,\n      virtual,\n    });\n\n    const applyJoinClause = (qb: Knex.QueryBuilder, joins: SubsetQuery[\"joins\"]) => {\n      joins.forEach((join) => {\n        if (join.join === \"inner\") {\n          qb.innerJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));\n        } else if (join.join === \"outer\") {\n          qb.leftOuterJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));\n        }\n      });\n    };\n\n    // countQuery\n    const total = await (async () => {\n      if (queryMode === \"list\") {\n        return undefined;\n      }\n\n      const clonedQb = qb.clone().clear(\"order\").clear(\"offset\").clear(\"limit\");\n      const parser = new SqlParser.Parser();\n\n      if (optimizeCountQuery) {\n        const parsedQuery = parser.astify(clonedQb.toQuery(), {\n          database: Sonamu.config.database.database,\n        });\n        const tables = getTableNamesFromWhere(parsedQuery);\n        const needToJoin = unique(\n          tables.flatMap((table) => table.split(\"__\").map((t) => inflection.pluralize(t))),\n        );\n        applyJoinClause(\n          clonedQb,\n          joins.filter((j) => needToJoin.includes(j.table)),\n        );\n      } else {\n        applyJoinClause(clonedQb, joins);\n      }\n\n      const processedQb = afterBuild?.({ qb: clonedQb, db, select, joins, virtual }) ?? clonedQb;\n\n      const parsedQuery = parser.astify(processedQb.toQuery(), {\n        database: Sonamu.config.database.database,\n      });\n      const q = Array.isArray(parsedQuery) ? parsedQuery[0] : parsedQuery;\n      if (q.type !== \"select\") {\n        throw new Error(\"Invalid query\");\n      }\n\n      const countQuery =\n        q.distinct !== null\n          ? clonedQb\n              .clear(\"select\")\n              .select(\n                db.raw(\n                  `COUNT(DISTINCT \\`${getTableName(q.columns[0].expr)}\\`.\\`${q.columns[0].expr.column}\\`) as total`,\n                ),\n              )\n              .first()\n          : clonedQb.clear(\"select\").count(\"*\", { as: \"total\" }).first();\n      const countRow: { total?: number } = await countQuery;\n\n      if (debug === true || debug === \"count\") {\n        console.debug(\"DEBUG: count query\", chalk.blue(countQuery.toQuery().toString()));\n      }\n\n      return countRow?.total ?? 0;\n    })();\n\n    // listQuery\n    const rows = await (async () => {\n      if (queryMode === \"count\") {\n        return [];\n      }\n\n      if (params.num !== 0) {\n        assert(params.num);\n        qb.limit(params.num);\n        qb.offset(params.num * ((params.page ?? 1) - 1));\n      }\n\n      const clonedQb = qb.clone().select(select);\n      applyJoinClause(clonedQb, joins);\n\n      const listQuery = afterBuild?.({ qb: clonedQb, db, select, joins, virtual }) ?? clonedQb;\n\n      let rows = await listQuery;\n      if (debug === true || debug === \"list\") {\n        console.debug(\"DEBUG: list query\", chalk.blue(listQuery.toQuery().toString()));\n      }\n\n      rows = await this.useLoaders(db, rows, loaders);\n      rows = this.hydrate(rows);\n      return rows;\n    })();\n\n    return { rows, total, subsetQuery, qb };\n  }\n\n  // Legacy Loader 처리 (Puri 도입 전 호환용)\n  async useLoaders(db: Knex, rows: UnknownDBRecord[], loaders: SubsetQuery[\"loaders\"]) {\n    if (loaders.length === 0) {\n      return rows;\n    }\n\n    for (const loader of loaders) {\n      let subQ: Knex.QueryBuilder;\n      let subRows: UnknownDBRecord[];\n      let toCol: string;\n\n      const fromIds = rows.map((row) => row[loader.manyJoin.idField]);\n\n      if (loader.manyJoin.through === undefined) {\n        // HasMany\n        const idColumn = `${loader.manyJoin.toTable}.${loader.manyJoin.toCol}`;\n        subQ = db(loader.manyJoin.toTable)\n          .whereIn(idColumn as string, fromIds as string[])\n          .select([...loader.select, idColumn]);\n\n        loader.oneJoins.forEach((join) => {\n          if (join.join === \"inner\") {\n            subQ.innerJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));\n          } else if (join.join === \"outer\") {\n            subQ.leftOuterJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));\n          }\n        });\n        toCol = loader.manyJoin.toCol;\n      } else {\n        // ManyToMany\n        const idColumn = `${loader.manyJoin.through.table}.${loader.manyJoin.through.fromCol}`;\n        subQ = db(loader.manyJoin.through.table)\n          .join(\n            loader.manyJoin.toTable,\n            `${loader.manyJoin.through.table}.${loader.manyJoin.through.toCol}`,\n            `${loader.manyJoin.toTable}.${loader.manyJoin.toCol}`,\n          )\n          .whereIn(idColumn as string, fromIds as string[])\n          .select(unique([...loader.select, idColumn]));\n\n        loader.oneJoins.forEach((join) => {\n          if (join.join === \"inner\") {\n            subQ.innerJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));\n          } else if (join.join === \"outer\") {\n            subQ.leftOuterJoin(`${join.table} as ${join.as}`, this.getJoinClause(db, join));\n          }\n        });\n        toCol = loader.manyJoin.through.fromCol;\n      }\n      subRows = await subQ;\n\n      if (loader.loaders) {\n        subRows = await this.useLoaders(db, subRows, loader.loaders);\n      }\n\n      const subRowGroups = group(subRows, (row) => row[toCol] as string);\n      rows = rows.map((row) => {\n        row[loader.as] = (subRowGroups[row[loader.manyJoin.idField] as string] ?? []).map((r) =>\n          omit(r, [toCol]),\n        );\n        return row;\n      });\n    }\n    return rows;\n  }\n\n  getJoinClause(db: Knex<any, unknown>, join: SubsetQuery[\"joins\"][number]): Knex.Raw<any> {\n    if (!isCustomJoinClause(join)) {\n      return db.raw(`${join.from} = ${join.to}`);\n    } else {\n      return db.raw(join.custom);\n    }\n  }\n\n  getUpsertBuilder(): UpsertBuilder {\n    return new UpsertBuilder();\n  }\n}\n\n/**\n * Enhancer 파라미터 조건부 타입\n * RequiredEnhancerKeys가 없으면 enhancers 선택적, 있으면 필수\n */\ntype EnhancerParam<\n  TSubsetKey extends string,\n  TComputedResults extends Record<TSubsetKey, any>,\n  TSubsetMapping extends Record<TSubsetKey, any>,\n> = [RequiredEnhancerKeys<TSubsetKey, TComputedResults, TSubsetMapping>] extends [never]\n  ? { enhancers?: EnhancerMap<TSubsetKey, TComputedResults, TSubsetMapping> }\n  : { enhancers: EnhancerMap<TSubsetKey, TComputedResults, TSubsetMapping> };\n\ntype RequiredEnhancerKeys<\n  TSubsetKey extends string,\n  TComputedResults extends Record<TSubsetKey, any>,\n  TSubsetMapping extends Record<TSubsetKey, any>,\n> = {\n  [K in TSubsetKey]: TComputedResults[K] extends TSubsetMapping[K] ? never : K;\n}[TSubsetKey];\n\nexport const BaseModel = new BaseModelClass();\n"],"names":["assert","inflection","group","isObject","omit","set","unique","Sonamu","isCustomJoinClause","getJoinTables","getTableNamesFromWhere","chunk","DB","Puri","PuriWrapper","UpsertBuilder","BaseModelClass","modelName","subsetQueries","loaderQueries","getDB","which","getPuri","trx","getTransactionContext","getTransaction","db","getUpsertBuilder","destroy","getInsertedIds","wdb","rows","tableName","unqKeyFields","chunkSize","unqKeys","whereInField","selectField","length","raw","join","map","row","field","resultIds","items","dbRows","select","whereIn","concat","dbRow","parseInt","String","id","getSubsetQueries","subset","Error","puriWrapper","qb","onSubset","_subset","createEnhancers","enhancers","executeSubsetQuery","params","queryParams","debug","optimizeCountQuery","num","page","total","executeCountQuery","computedRows","executeListQuery","enhancer","Promise","all","queryMode","countPuri","clone","clear","default","SqlParser","parser","Parser","parsedQuery","astify","toQuery","database","config","leftJoinTables","whereTables","tablesToRemove","filter","j","includes","forEach","table","clearJoin","countResult","rawNumber","first","unloadedRows","limit","offset","loaders","Array","isArray","processLoaders","hydrate","resolveLoader","as","refId","resolveLoaderQbFn","nestedLoaders","resolveLoaderQb","loadedRows","subRowGroups","r","nestedKeys","Object","keys","key","groups","groupBy","split","nullKeys","entries","_","data","every","hydrated","reduce","parts","objPath","slice","part","nullKey","runSubsetQuery","baseTable","subsetQuery","build","afterBuild","_db","chalk","getTableName","startsWith","pluralize","underscore","undefined","virtual","joins","from","applyJoinClause","innerJoin","getJoinClause","leftOuterJoin","clonedQb","tables","needToJoin","flatMap","t","processedQb","q","type","countQuery","distinct","columns","expr","column","count","countRow","console","blue","toString","listQuery","useLoaders","loader","subQ","subRows","toCol","fromIds","manyJoin","idField","through","idColumn","toTable","oneJoins","fromCol","to","custom","BaseModel"],"mappings":"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"}