sonamu 0.6.0 → 0.7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (406) hide show
  1. package/.swcrc.project-default +18 -0
  2. package/bin/cli.js +24 -0
  3. package/dist/ai/agents/agent.d.ts +11 -0
  4. package/dist/ai/agents/agent.d.ts.map +1 -0
  5. package/dist/ai/agents/agent.js +65 -0
  6. package/dist/ai/agents/index.d.ts +3 -0
  7. package/dist/ai/agents/index.d.ts.map +1 -0
  8. package/dist/ai/agents/index.js +4 -0
  9. package/dist/ai/agents/types.d.ts +43 -0
  10. package/dist/ai/agents/types.d.ts.map +1 -0
  11. package/dist/ai/agents/types.js +3 -0
  12. package/dist/ai/index.d.ts +2 -0
  13. package/dist/ai/index.d.ts.map +1 -0
  14. package/dist/ai/index.js +3 -0
  15. package/dist/ai/providers/rtzr/api.d.ts +22 -0
  16. package/dist/ai/providers/rtzr/api.d.ts.map +1 -0
  17. package/dist/ai/providers/rtzr/api.js +28 -0
  18. package/dist/ai/providers/rtzr/error.d.ts +18 -0
  19. package/dist/ai/providers/rtzr/error.d.ts.map +1 -0
  20. package/dist/ai/providers/rtzr/error.js +29 -0
  21. package/dist/ai/providers/rtzr/index.d.ts +5 -0
  22. package/dist/ai/providers/rtzr/index.d.ts.map +1 -0
  23. package/dist/ai/providers/rtzr/index.js +6 -0
  24. package/dist/ai/providers/rtzr/model.d.ts +52 -0
  25. package/dist/ai/providers/rtzr/model.d.ts.map +1 -0
  26. package/dist/ai/providers/rtzr/model.js +137 -0
  27. package/dist/ai/providers/rtzr/options.d.ts +7 -0
  28. package/dist/ai/providers/rtzr/options.d.ts.map +1 -0
  29. package/dist/ai/providers/rtzr/options.js +47 -0
  30. package/dist/ai/providers/rtzr/provider.d.ts +18 -0
  31. package/dist/ai/providers/rtzr/provider.d.ts.map +1 -0
  32. package/dist/ai/providers/rtzr/provider.js +54 -0
  33. package/dist/ai/providers/rtzr/utils.d.ts +19 -0
  34. package/dist/ai/providers/rtzr/utils.d.ts.map +1 -0
  35. package/dist/ai/providers/rtzr/utils.js +88 -0
  36. package/dist/api/base-frame.d.ts +2 -2
  37. package/dist/api/base-frame.d.ts.map +1 -1
  38. package/dist/api/base-frame.js +2 -1
  39. package/dist/api/caster.d.ts.map +1 -1
  40. package/dist/api/caster.js +6 -1
  41. package/dist/api/code-converters.d.ts +58 -14
  42. package/dist/api/code-converters.d.ts.map +1 -1
  43. package/dist/api/code-converters.js +178 -409
  44. package/dist/api/config.d.ts +27 -13
  45. package/dist/api/config.d.ts.map +1 -1
  46. package/dist/api/config.js +19 -26
  47. package/dist/api/context.d.ts +4 -3
  48. package/dist/api/context.d.ts.map +1 -1
  49. package/dist/api/context.js +1 -1
  50. package/dist/api/decorators.d.ts +20 -6
  51. package/dist/api/decorators.d.ts.map +1 -1
  52. package/dist/api/decorators.js +111 -18
  53. package/dist/api/index.d.ts +2 -2
  54. package/dist/api/index.d.ts.map +1 -1
  55. package/dist/api/index.js +3 -3
  56. package/dist/api/sonamu.d.ts +7 -7
  57. package/dist/api/sonamu.d.ts.map +1 -1
  58. package/dist/api/sonamu.js +83 -51
  59. package/dist/api/validator.d.ts +6 -0
  60. package/dist/api/validator.d.ts.map +1 -0
  61. package/dist/api/validator.js +81 -0
  62. package/dist/bin/build-config.d.ts +5 -1
  63. package/dist/bin/build-config.d.ts.map +1 -1
  64. package/dist/bin/build-config.js +5 -2
  65. package/dist/bin/cli.js +165 -64
  66. package/dist/bin/loader-register.d.ts +2 -0
  67. package/dist/bin/loader-register.d.ts.map +1 -0
  68. package/dist/bin/loader-register.js +34 -0
  69. package/dist/database/_batch_update.d.ts +5 -3
  70. package/dist/database/_batch_update.d.ts.map +1 -1
  71. package/dist/database/_batch_update.js +30 -13
  72. package/dist/database/base-model.d.ts +96 -10
  73. package/dist/database/base-model.d.ts.map +1 -1
  74. package/dist/database/base-model.js +232 -89
  75. package/dist/database/base-model.types.d.ts +93 -0
  76. package/dist/database/base-model.types.d.ts.map +1 -0
  77. package/dist/database/base-model.types.js +10 -0
  78. package/dist/database/code-generator.d.ts +1 -1
  79. package/dist/database/code-generator.d.ts.map +1 -1
  80. package/dist/database/code-generator.js +11 -10
  81. package/dist/database/db.d.ts +5 -6
  82. package/dist/database/db.d.ts.map +1 -1
  83. package/dist/database/db.js +22 -25
  84. package/dist/database/puri-subset.test-d.js +81 -0
  85. package/dist/database/puri-subset.types.d.ts +123 -0
  86. package/dist/database/puri-subset.types.d.ts.map +1 -0
  87. package/dist/database/puri-subset.types.js +16 -0
  88. package/dist/database/puri-wrapper.d.ts +13 -11
  89. package/dist/database/puri-wrapper.d.ts.map +1 -1
  90. package/dist/database/puri-wrapper.js +2 -2
  91. package/dist/database/puri.d.ts +25 -14
  92. package/dist/database/puri.d.ts.map +1 -1
  93. package/dist/database/puri.js +83 -21
  94. package/dist/database/puri.types.d.ts +21 -7
  95. package/dist/database/puri.types.d.ts.map +1 -1
  96. package/dist/database/puri.types.js +4 -1
  97. package/dist/database/transaction-context.d.ts +1 -1
  98. package/dist/database/transaction-context.d.ts.map +1 -1
  99. package/dist/database/transaction-context.js +1 -1
  100. package/dist/database/upsert-builder.d.ts +9 -3
  101. package/dist/database/upsert-builder.d.ts.map +1 -1
  102. package/dist/database/upsert-builder.js +228 -78
  103. package/dist/entity/entity-manager.d.ts +165 -2
  104. package/dist/entity/entity-manager.d.ts.map +1 -1
  105. package/dist/entity/entity-manager.js +26 -10
  106. package/dist/entity/entity.d.ts +5 -3
  107. package/dist/entity/entity.d.ts.map +1 -1
  108. package/dist/entity/entity.js +153 -54
  109. package/dist/exceptions/error-handler.d.ts +1 -1
  110. package/dist/exceptions/error-handler.d.ts.map +1 -1
  111. package/dist/exceptions/error-handler.js +1 -1
  112. package/dist/exceptions/so-exceptions.d.ts +1 -1
  113. package/dist/exceptions/so-exceptions.d.ts.map +1 -1
  114. package/dist/exceptions/so-exceptions.js +1 -1
  115. package/dist/file-storage/driver.d.ts +1 -1
  116. package/dist/file-storage/driver.d.ts.map +1 -1
  117. package/dist/file-storage/driver.js +1 -1
  118. package/dist/file-storage/file-storage.js +2 -2
  119. package/dist/index.d.ts +18 -11
  120. package/dist/index.d.ts.map +1 -1
  121. package/dist/index.js +19 -13
  122. package/dist/migration/code-generation.d.ts +1 -1
  123. package/dist/migration/code-generation.d.ts.map +1 -1
  124. package/dist/migration/code-generation.js +123 -67
  125. package/dist/migration/migration-set.d.ts +2 -10
  126. package/dist/migration/migration-set.d.ts.map +1 -1
  127. package/dist/migration/migration-set.js +67 -218
  128. package/dist/migration/migrator.d.ts +24 -73
  129. package/dist/migration/migrator.d.ts.map +1 -1
  130. package/dist/migration/migrator.js +121 -301
  131. package/dist/migration/postgresql-schema-reader.d.ts +51 -0
  132. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -0
  133. package/dist/migration/postgresql-schema-reader.js +245 -0
  134. package/dist/migration/types.d.ts +6 -38
  135. package/dist/migration/types.d.ts.map +1 -1
  136. package/dist/migration/types.js +1 -1
  137. package/dist/naite/messaging-types.d.ts +43 -0
  138. package/dist/naite/messaging-types.d.ts.map +1 -0
  139. package/dist/naite/messaging-types.js +7 -0
  140. package/dist/naite/naite-reporter.d.ts +41 -0
  141. package/dist/naite/naite-reporter.d.ts.map +1 -0
  142. package/dist/naite/naite-reporter.js +102 -0
  143. package/dist/naite/naite.d.ts +91 -8
  144. package/dist/naite/naite.d.ts.map +1 -1
  145. package/dist/naite/naite.js +285 -41
  146. package/dist/stream/sse.d.ts +2 -2
  147. package/dist/stream/sse.d.ts.map +1 -1
  148. package/dist/stream/sse.js +1 -1
  149. package/dist/syncer/api-parser.d.ts +3 -13
  150. package/dist/syncer/api-parser.d.ts.map +1 -1
  151. package/dist/syncer/api-parser.js +67 -56
  152. package/dist/syncer/checksum.d.ts +2 -2
  153. package/dist/syncer/checksum.d.ts.map +1 -1
  154. package/dist/syncer/checksum.js +11 -11
  155. package/dist/syncer/code-generator.d.ts +3 -3
  156. package/dist/syncer/code-generator.d.ts.map +1 -1
  157. package/dist/syncer/code-generator.js +37 -17
  158. package/dist/syncer/entity-operations.d.ts +2 -2
  159. package/dist/syncer/entity-operations.d.ts.map +1 -1
  160. package/dist/syncer/entity-operations.js +9 -8
  161. package/dist/syncer/file-patterns.d.ts +1 -1
  162. package/dist/syncer/file-patterns.d.ts.map +1 -1
  163. package/dist/syncer/file-patterns.js +1 -1
  164. package/dist/syncer/index.d.ts +4 -4
  165. package/dist/syncer/index.d.ts.map +1 -1
  166. package/dist/syncer/index.js +5 -5
  167. package/dist/syncer/module-loader.d.ts +4 -4
  168. package/dist/syncer/module-loader.d.ts.map +1 -1
  169. package/dist/syncer/module-loader.js +17 -12
  170. package/dist/syncer/syncer.d.ts +31 -24
  171. package/dist/syncer/syncer.d.ts.map +1 -1
  172. package/dist/syncer/syncer.js +92 -45
  173. package/dist/template/entity-converter.d.ts +1 -1
  174. package/dist/template/entity-converter.d.ts.map +1 -1
  175. package/dist/template/entity-converter.js +15 -8
  176. package/dist/template/helpers.d.ts +2 -2
  177. package/dist/template/helpers.d.ts.map +1 -1
  178. package/dist/template/helpers.js +3 -3
  179. package/dist/template/implementations/entity.template.d.ts +2 -2
  180. package/dist/template/implementations/entity.template.d.ts.map +1 -1
  181. package/dist/template/implementations/entity.template.js +4 -5
  182. package/dist/template/implementations/generated.template.d.ts +2 -3
  183. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  184. package/dist/template/implementations/generated.template.js +46 -29
  185. package/dist/template/implementations/generated_http.template.d.ts +2 -3
  186. package/dist/template/implementations/generated_http.template.d.ts.map +1 -1
  187. package/dist/template/implementations/generated_http.template.js +9 -9
  188. package/dist/template/implementations/generated_sso.template.d.ts +3 -4
  189. package/dist/template/implementations/generated_sso.template.d.ts.map +1 -1
  190. package/dist/template/implementations/generated_sso.template.js +54 -25
  191. package/dist/template/implementations/init_types.template.d.ts +2 -2
  192. package/dist/template/implementations/init_types.template.d.ts.map +1 -1
  193. package/dist/template/implementations/init_types.template.js +2 -2
  194. package/dist/template/implementations/model.template.d.ts +2 -2
  195. package/dist/template/implementations/model.template.d.ts.map +1 -1
  196. package/dist/template/implementations/model.template.js +47 -37
  197. package/dist/template/implementations/model_test.template.d.ts +2 -2
  198. package/dist/template/implementations/model_test.template.d.ts.map +1 -1
  199. package/dist/template/implementations/model_test.template.js +2 -2
  200. package/dist/template/implementations/service.template.d.ts +4 -4
  201. package/dist/template/implementations/service.template.d.ts.map +1 -1
  202. package/dist/template/implementations/service.template.js +24 -16
  203. package/dist/template/implementations/view_enums_buttonset.template.d.ts +2 -2
  204. package/dist/template/implementations/view_enums_buttonset.template.d.ts.map +1 -1
  205. package/dist/template/implementations/view_enums_buttonset.template.js +1 -1
  206. package/dist/template/implementations/view_enums_dropdown.template.d.ts +2 -2
  207. package/dist/template/implementations/view_enums_dropdown.template.d.ts.map +1 -1
  208. package/dist/template/implementations/view_enums_dropdown.template.js +2 -2
  209. package/dist/template/implementations/view_enums_select.template.d.ts +2 -2
  210. package/dist/template/implementations/view_enums_select.template.d.ts.map +1 -1
  211. package/dist/template/implementations/view_enums_select.template.js +2 -2
  212. package/dist/template/implementations/view_form.template.d.ts +2 -2
  213. package/dist/template/implementations/view_form.template.d.ts.map +1 -1
  214. package/dist/template/implementations/view_form.template.js +4 -4
  215. package/dist/template/implementations/view_id_all_select.template.d.ts +2 -2
  216. package/dist/template/implementations/view_id_all_select.template.d.ts.map +1 -1
  217. package/dist/template/implementations/view_id_all_select.template.js +1 -1
  218. package/dist/template/implementations/view_id_async_select.template.d.ts +2 -2
  219. package/dist/template/implementations/view_id_async_select.template.d.ts.map +1 -1
  220. package/dist/template/implementations/view_id_async_select.template.js +1 -1
  221. package/dist/template/implementations/view_list.template.d.ts +2 -2
  222. package/dist/template/implementations/view_list.template.d.ts.map +1 -1
  223. package/dist/template/implementations/view_list.template.js +29 -19
  224. package/dist/template/implementations/view_list_columns.template.d.ts +3 -3
  225. package/dist/template/implementations/view_list_columns.template.d.ts.map +1 -1
  226. package/dist/template/implementations/view_list_columns.template.js +1 -1
  227. package/dist/template/implementations/view_search_input.template.d.ts +2 -2
  228. package/dist/template/implementations/view_search_input.template.d.ts.map +1 -1
  229. package/dist/template/implementations/view_search_input.template.js +1 -1
  230. package/dist/template/index.d.ts +4 -2
  231. package/dist/template/index.d.ts.map +1 -1
  232. package/dist/template/index.js +5 -3
  233. package/dist/template/template-manager.d.ts +56 -0
  234. package/dist/template/template-manager.d.ts.map +1 -0
  235. package/dist/template/template-manager.js +125 -0
  236. package/dist/template/template-types.d.ts +16 -0
  237. package/dist/template/template-types.d.ts.map +1 -0
  238. package/dist/template/template-types.js +7 -0
  239. package/dist/template/template.d.ts +12 -2
  240. package/dist/template/template.d.ts.map +1 -1
  241. package/dist/template/template.js +19 -6
  242. package/dist/template/zod-converter.d.ts +40 -7
  243. package/dist/template/zod-converter.d.ts.map +1 -1
  244. package/dist/template/zod-converter.js +341 -58
  245. package/dist/testing/_relation-graph.d.ts +1 -1
  246. package/dist/testing/_relation-graph.d.ts.map +1 -1
  247. package/dist/testing/_relation-graph.js +12 -3
  248. package/dist/testing/fixture-manager.d.ts +42 -11
  249. package/dist/testing/fixture-manager.d.ts.map +1 -1
  250. package/dist/testing/fixture-manager.js +338 -236
  251. package/dist/types/types.d.ts +709 -104
  252. package/dist/types/types.d.ts.map +1 -1
  253. package/dist/types/types.js +309 -52
  254. package/dist/typings/knex.d.js +2 -2
  255. package/dist/utils/async-utils.d.ts.map +1 -1
  256. package/dist/utils/async-utils.js +3 -3
  257. package/dist/utils/console-util.js +1 -1
  258. package/dist/utils/controller.d.ts +1 -0
  259. package/dist/utils/controller.d.ts.map +1 -1
  260. package/dist/utils/controller.js +4 -1
  261. package/dist/utils/esm-utils.d.ts +0 -6
  262. package/dist/utils/esm-utils.d.ts.map +1 -1
  263. package/dist/utils/esm-utils.js +2 -9
  264. package/dist/utils/formatter.d.ts +3 -0
  265. package/dist/utils/formatter.d.ts.map +1 -0
  266. package/dist/utils/formatter.js +110 -0
  267. package/dist/utils/fs-utils.d.ts +1 -1
  268. package/dist/utils/fs-utils.d.ts.map +1 -1
  269. package/dist/utils/fs-utils.js +1 -1
  270. package/dist/utils/lodash-able.d.ts.map +1 -1
  271. package/dist/utils/lodash-able.js +1 -1
  272. package/dist/utils/object-utils.d.ts +44 -0
  273. package/dist/utils/object-utils.d.ts.map +1 -0
  274. package/dist/utils/object-utils.js +191 -0
  275. package/dist/utils/path-utils.d.ts +1 -1
  276. package/dist/utils/path-utils.d.ts.map +1 -1
  277. package/dist/utils/path-utils.js +3 -3
  278. package/dist/utils/process-utils.js +1 -1
  279. package/dist/utils/sql-parser.d.ts +5 -1
  280. package/dist/utils/sql-parser.d.ts.map +1 -1
  281. package/dist/utils/sql-parser.js +14 -3
  282. package/dist/utils/type-utils.d.ts +23 -0
  283. package/dist/utils/type-utils.d.ts.map +1 -0
  284. package/dist/utils/type-utils.js +45 -0
  285. package/dist/utils/utils.d.ts +7 -1
  286. package/dist/utils/utils.d.ts.map +1 -1
  287. package/dist/utils/utils.js +44 -5
  288. package/dist/utils/zod-error.d.ts +1 -1
  289. package/dist/utils/zod-error.d.ts.map +1 -1
  290. package/dist/utils/zod-error.js +1 -1
  291. package/package.json +54 -29
  292. package/src/ai/agents/agent.ts +87 -0
  293. package/src/ai/agents/index.ts +2 -0
  294. package/src/ai/agents/types.ts +47 -0
  295. package/src/ai/index.ts +1 -0
  296. package/src/ai/providers/rtzr/api.ts +37 -0
  297. package/src/ai/providers/rtzr/error.ts +34 -0
  298. package/src/ai/providers/rtzr/index.ts +4 -0
  299. package/src/ai/providers/rtzr/model.ts +201 -0
  300. package/src/ai/providers/rtzr/options.ts +49 -0
  301. package/src/ai/providers/rtzr/provider.ts +91 -0
  302. package/src/ai/providers/rtzr/utils.ts +127 -0
  303. package/src/api/base-frame.ts +4 -2
  304. package/src/api/caster.ts +17 -23
  305. package/src/api/code-converters.ts +176 -533
  306. package/src/api/config.ts +39 -56
  307. package/src/api/context.ts +7 -18
  308. package/src/api/decorators.ts +175 -46
  309. package/src/api/index.ts +2 -2
  310. package/src/api/sonamu.ts +133 -124
  311. package/src/api/validator.ts +83 -0
  312. package/src/bin/build-config.ts +7 -1
  313. package/src/bin/cli.ts +192 -110
  314. package/src/bin/loader-register.ts +38 -0
  315. package/src/database/_batch_update.ts +46 -31
  316. package/src/database/base-model.ts +390 -182
  317. package/src/database/base-model.types.ts +155 -0
  318. package/src/database/code-generator.ts +13 -32
  319. package/src/database/db.ts +36 -50
  320. package/src/database/puri-subset.test-d.ts +471 -0
  321. package/src/database/puri-subset.types.ts +195 -0
  322. package/src/database/puri-wrapper.ts +58 -67
  323. package/src/database/puri.ts +182 -126
  324. package/src/database/puri.types.ts +64 -31
  325. package/src/database/transaction-context.ts +1 -1
  326. package/src/database/upsert-builder.ts +262 -132
  327. package/src/entity/entity-manager.ts +36 -28
  328. package/src/entity/entity.ts +330 -249
  329. package/src/exceptions/error-handler.ts +3 -3
  330. package/src/exceptions/so-exceptions.ts +11 -11
  331. package/src/file-storage/driver.ts +5 -5
  332. package/src/file-storage/file-storage.ts +2 -2
  333. package/src/index.ts +18 -12
  334. package/src/migration/code-generation.ts +185 -172
  335. package/src/migration/migration-set.ts +80 -293
  336. package/src/migration/migrator.ts +182 -425
  337. package/src/migration/mysql-schema-reader.ts.txt +272 -0
  338. package/src/migration/postgresql-schema-reader.ts +310 -0
  339. package/src/migration/types.ts +6 -39
  340. package/src/naite/messaging-types.ts +51 -0
  341. package/src/naite/naite-reporter.ts +128 -0
  342. package/src/naite/naite.ts +378 -33
  343. package/src/shared/web.shared.ts.txt +20 -24
  344. package/src/stream/sse.ts +5 -5
  345. package/src/syncer/api-parser.ts +52 -69
  346. package/src/syncer/checksum.ts +25 -37
  347. package/src/syncer/code-generator.ts +58 -62
  348. package/src/syncer/entity-operations.ts +12 -15
  349. package/src/syncer/file-patterns.ts +2 -2
  350. package/src/syncer/index.ts +4 -4
  351. package/src/syncer/module-loader.ts +28 -25
  352. package/src/syncer/syncer.ts +155 -162
  353. package/src/template/entity-converter.ts +18 -27
  354. package/src/template/helpers.ts +8 -11
  355. package/src/template/implementations/entity.template.ts +6 -6
  356. package/src/template/implementations/generated.template.ts +99 -99
  357. package/src/template/implementations/generated_http.template.ts +21 -54
  358. package/src/template/implementations/generated_sso.template.ts +78 -65
  359. package/src/template/implementations/init_types.template.ts +4 -6
  360. package/src/template/implementations/model.template.ts +47 -38
  361. package/src/template/implementations/model_test.template.ts +3 -3
  362. package/src/template/implementations/service.template.ts +56 -80
  363. package/src/template/implementations/view_enums_buttonset.template.ts +2 -2
  364. package/src/template/implementations/view_enums_dropdown.template.ts +4 -4
  365. package/src/template/implementations/view_enums_select.template.ts +3 -3
  366. package/src/template/implementations/view_form.template.ts +34 -75
  367. package/src/template/implementations/view_id_all_select.template.ts +2 -2
  368. package/src/template/implementations/view_id_async_select.template.ts +9 -23
  369. package/src/template/implementations/view_list.template.ts +54 -95
  370. package/src/template/implementations/view_list_columns.template.ts +4 -10
  371. package/src/template/implementations/view_search_input.template.ts +2 -2
  372. package/src/template/index.ts +4 -2
  373. package/src/template/template-manager.ts +166 -0
  374. package/src/template/template-types.ts +16 -0
  375. package/src/template/template.ts +29 -10
  376. package/src/template/zod-converter.ts +407 -101
  377. package/src/testing/_relation-graph.ts +18 -11
  378. package/src/testing/fixture-manager.ts +468 -362
  379. package/src/types/types.ts +516 -248
  380. package/src/typings/knex.d.ts +7 -9
  381. package/src/utils/async-utils.ts +8 -12
  382. package/src/utils/console-util.ts +1 -1
  383. package/src/utils/controller.ts +3 -0
  384. package/src/utils/esm-utils.ts +8 -18
  385. package/src/utils/formatter.ts +109 -0
  386. package/src/utils/fs-utils.ts +1 -1
  387. package/src/utils/lodash-able.ts +1 -4
  388. package/src/utils/object-utils.ts +217 -0
  389. package/src/utils/path-utils.ts +3 -6
  390. package/src/utils/process-utils.ts +1 -1
  391. package/src/utils/sql-parser.ts +23 -5
  392. package/src/utils/type-utils.ts +83 -0
  393. package/src/utils/utils.ts +58 -9
  394. package/src/utils/zod-error.ts +3 -3
  395. package/dist/bin/cli-wrapper.d.ts +0 -3
  396. package/dist/bin/cli-wrapper.d.ts.map +0 -1
  397. package/dist/bin/cli-wrapper.js +0 -72
  398. package/dist/database/knex-plugins/knex-on-duplicate-update.d.ts +0 -2
  399. package/dist/database/knex-plugins/knex-on-duplicate-update.d.ts.map +0 -1
  400. package/dist/database/knex-plugins/knex-on-duplicate-update.js +0 -39
  401. package/dist/entity/entity-utils.d.ts +0 -61
  402. package/dist/entity/entity-utils.d.ts.map +0 -1
  403. package/dist/entity/entity-utils.js +0 -210
  404. package/src/bin/cli-wrapper.ts +0 -82
  405. package/src/database/knex-plugins/knex-on-duplicate-update.ts +0 -45
  406. package/src/entity/entity-utils.ts +0 -291
@@ -1,7 +1,7 @@
1
- import { Knex } from "knex";
1
+ import type { Knex } from "knex";
2
2
  type TableData = {
3
3
  references: Set<string>;
4
- rows: any[];
4
+ rows: Record<string, unknown>[];
5
5
  uniqueIndexes: {
6
6
  name?: string;
7
7
  columns: string[];
@@ -13,7 +13,7 @@ export type UBRef = {
13
13
  of: string;
14
14
  use?: string;
15
15
  };
16
- export declare function isRefField(field: any): field is UBRef;
16
+ export declare function isRefField(field: unknown): field is UBRef;
17
17
  export declare class UpsertBuilder {
18
18
  tables: Map<string, TableData>;
19
19
  constructor();
@@ -29,6 +29,12 @@ export declare class UpsertBuilder {
29
29
  chunkSize?: number;
30
30
  where?: string | string[];
31
31
  }): Promise<void>;
32
+ /**
33
+ * rows를 의존성 순서에 따라 레벨별로 그룹화
34
+ * - 자기 참조 없는 경우 : 모든 rows가 Level 0
35
+ * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화
36
+ */
37
+ private buildInsertLevels;
32
38
  }
33
39
  export {};
34
40
  //# sourceMappingURL=upsert-builder.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"upsert-builder.d.ts","sourceRoot":"","sources":["../../src/database/upsert-builder.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAK5B,KAAK,SAAS,GAAG;IACf,UAAU,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;IACxB,IAAI,EAAE,GAAG,EAAE,CAAC;IACZ,aAAa,EAAE;QAAE,IAAI,CAAC,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,EAAE,CAAA;KAAE,EAAE,CAAC;IACtD,UAAU,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CACjC,CAAC;AACF,MAAM,MAAM,KAAK,GAAG;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,EAAE,EAAE,MAAM,CAAC;IACX,GAAG,CAAC,EAAE,MAAM,CAAC;CACd,CAAC;AACF,wBAAgB,UAAU,CAAC,KAAK,EAAE,GAAG,GAAG,KAAK,IAAI,KAAK,CAOrD;AAED,qBAAa,aAAa;IACxB,MAAM,EAAE,GAAG,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;;IAK/B,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,SAAS;IAsBtC,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO;IAIpC,QAAQ,CAAC,CAAC,SAAS,MAAM,EACvB,SAAS,EAAE,MAAM,EACjB,GAAG,EAAE;SACF,GAAG,IAAI,CAAC,CAAC,CAAC,EACP,KAAK,GACL,MAAM,GACN,MAAM,GACN,OAAO,GACP,MAAM,GACN,IAAI,GACJ,MAAM,GACN,OAAO;KACZ,GACA,KAAK;IA0EF,MAAM,CACV,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,SAAS,CAAC,EAAE,MAAM,GACjB,OAAO,CAAC,MAAM,EAAE,CAAC;IAGd,UAAU,CACd,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,SAAS,CAAC,EAAE,MAAM,GACjB,OAAO,CAAC,MAAM,EAAE,CAAC;IAId,cAAc,CAClB,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,IAAI,EAAE,QAAQ,GAAG,QAAQ,EACzB,SAAS,CAAC,EAAE,MAAM,GACjB,OAAO,CAAC,MAAM,EAAE,CAAC;IAgHd,WAAW,CACf,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,OAAO,CAAC,EAAE;QACR,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,KAAK,CAAC,EAAE,MAAM,GAAG,MAAM,EAAE,CAAC;KAC3B,GACA,OAAO,CAAC,IAAI,CAAC;CA6BjB"}
1
+ {"version":3,"file":"upsert-builder.d.ts","sourceRoot":"","sources":["../../src/database/upsert-builder.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAOjC,KAAK,SAAS,GAAG;IACf,UAAU,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;IACxB,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,CAAC;IAChC,aAAa,EAAE;QAAE,IAAI,CAAC,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,EAAE,CAAA;KAAE,EAAE,CAAC;IACtD,UAAU,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CACjC,CAAC;AACF,MAAM,MAAM,KAAK,GAAG;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,EAAE,EAAE,MAAM,CAAC;IACX,GAAG,CAAC,EAAE,MAAM,CAAC;CACd,CAAC;AACF,wBAAgB,UAAU,CAAC,KAAK,EAAE,OAAO,GAAG,KAAK,IAAI,KAAK,CAOzD;AAED,qBAAa,aAAa;IACxB,MAAM,EAAE,GAAG,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;;IAK/B,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,SAAS;IAwBtC,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO;IAIpC,QAAQ,CAAC,CAAC,SAAS,MAAM,EACvB,SAAS,EAAE,MAAM,EACjB,GAAG,EAAE;SACF,GAAG,IAAI,CAAC,CAAC,CAAC,EAAE,KAAK,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,MAAM,GAAG,IAAI,GAAG,MAAM,GAAG,OAAO;KAClF,GACA,KAAK;IAqFF,MAAM,CAAC,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAG3E,UAAU,CAAC,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAI/E,cAAc,CAClB,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,IAAI,EAAE,QAAQ,GAAG,QAAQ,EACzB,SAAS,CAAC,EAAE,MAAM,GACjB,OAAO,CAAC,MAAM,EAAE,CAAC;IAqKd,WAAW,CACf,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,OAAO,CAAC,EAAE;QACR,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,KAAK,CAAC,EAAE,MAAM,GAAG,MAAM,EAAE,CAAC;KAC3B,GACA,OAAO,CAAC,IAAI,CAAC;IAyChB;;;;OAIG;IACH,OAAO,CAAC,iBAAiB;CA8D1B"}
@@ -1,10 +1,11 @@
1
1
  import { randomUUID } from "crypto";
2
- import * as _ from "lodash-es";
2
+ import { unique } from "radashi";
3
3
  import { EntityManager } from "../entity/entity-manager.js";
4
- import { nonNullable } from "../utils/utils.js";
4
+ import { Naite } from "../naite/naite.js";
5
+ import { assertDefined, chunk, nonNullable } from "../utils/utils.js";
5
6
  import { batchUpdate } from "./_batch_update.js";
6
7
  export function isRefField(field) {
7
- return field !== undefined && field !== null && field.of !== undefined && field.uuid !== undefined;
8
+ return field !== undefined && field !== null && field?.of !== undefined && field?.uuid !== undefined;
8
9
  }
9
10
  export class UpsertBuilder {
10
11
  tables;
@@ -13,22 +14,24 @@ export class UpsertBuilder {
13
14
  }
14
15
  getTable(tableName) {
15
16
  const table = this.tables.get(tableName);
16
- if (table === undefined) {
17
- const tableSpec = (()=>{
18
- try {
19
- return EntityManager.getTableSpec(tableName);
20
- } catch {
21
- return null;
22
- }
23
- })();
24
- this.tables.set(tableName, {
25
- references: new Set(),
26
- rows: [],
27
- uniqueIndexes: tableSpec?.uniqueIndexes ?? [],
28
- uniquesMap: new Map()
29
- });
17
+ if (table) {
18
+ return table;
30
19
  }
31
- return this.tables.get(tableName);
20
+ const tableSpec = (()=>{
21
+ try {
22
+ return EntityManager.getTableSpec(tableName);
23
+ } catch {
24
+ return null;
25
+ }
26
+ })();
27
+ const tableData = {
28
+ references: new Set(),
29
+ rows: [],
30
+ uniqueIndexes: tableSpec?.uniqueIndexes ?? [],
31
+ uniquesMap: new Map()
32
+ };
33
+ this.tables.set(tableName, tableData);
34
+ return tableData;
32
35
  }
33
36
  hasTable(tableName) {
34
37
  return this.tables.has(tableName);
@@ -52,17 +55,23 @@ export class UpsertBuilder {
52
55
  return uniqueKeyArray.join("---delimiter--");
53
56
  }).filter(nonNullable);
54
57
  // uuid 생성 로직
55
- const uuid = (()=>{
58
+ const { uuid, isReused } = (()=>{
56
59
  // 키를 순회하여 이미 존재하는 키가 있는지 확인
57
60
  if (uniqueKeys.length > 0) {
58
61
  for (const uniqueKey of uniqueKeys){
59
62
  if (table.uniquesMap.has(uniqueKey)) {
60
- return table.uniquesMap.get(uniqueKey); // 이미 has 체크를 했으므로 undefined 불가능
63
+ return {
64
+ uuid: assertDefined(table.uniquesMap.get(uniqueKey), "Unique key not found"),
65
+ isReused: true
66
+ };
61
67
  }
62
68
  }
63
69
  }
64
70
  // 찾을 수 없는 경우 생성
65
- return randomUUID();
71
+ return {
72
+ uuid: randomUUID(),
73
+ isReused: false
74
+ };
66
75
  })();
67
76
  // 모든 유니크키에 대해 유니크맵에 uuid 저장
68
77
  if (uniqueKeys.length > 0) {
@@ -72,28 +81,42 @@ export class UpsertBuilder {
72
81
  }
73
82
  // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가
74
83
  // 이 정보를 나중에 치환할 때 사용
75
- row = Object.keys(row).reduce((r, rowKey)=>{
76
- const rowValue = row[rowKey];
84
+ row = Object.fromEntries(Object.entries(row).map(([rowKey, rowValue])=>{
77
85
  if (isRefField(rowValue)) {
78
86
  rowValue.use ??= "id";
79
- table.references.add(rowValue.of + "." + rowValue.use);
80
- r[rowKey] = rowValue;
87
+ table.references.add(`${rowValue.of}.${rowValue.use}`);
88
+ return [
89
+ rowKey,
90
+ rowValue
91
+ ];
81
92
  } else if (typeof rowValue === "object" && !(rowValue instanceof Date)) {
82
93
  // object인 경우 JSON으로 변환
83
- r[rowKey] = rowValue === null ? null : JSON.stringify(rowValue);
94
+ return [
95
+ rowKey,
96
+ rowValue === null ? null : JSON.stringify(rowValue)
97
+ ];
84
98
  } else {
85
- r[rowKey] = rowValue;
99
+ return [
100
+ rowKey,
101
+ rowValue
102
+ ];
86
103
  }
87
- return r;
88
- }, {});
104
+ }));
89
105
  table.rows.push({
90
106
  uuid,
91
107
  ...row
92
108
  });
93
- return {
109
+ const result = {
94
110
  of: tableName,
95
111
  uuid: row.uuid ?? uuid
96
112
  };
113
+ Naite.t("puri:ub-register", {
114
+ tableName,
115
+ uuid: result.uuid,
116
+ isUuidReused: isReused,
117
+ row
118
+ });
119
+ return result;
97
120
  }
98
121
  async upsert(wdb, tableName, chunkSize) {
99
122
  return this.upsertOrInsert(wdb, tableName, "upsert", chunkSize);
@@ -116,7 +139,7 @@ export class UpsertBuilder {
116
139
  }
117
140
  // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출
118
141
  const { references, refTables } = Array.from(this.tables).reduce((r, [, table])=>{
119
- const reference = Array.from(table.references.values()).find((ref)=>ref.includes(tableName + "."));
142
+ const reference = Array.from(table.references.values()).find((ref)=>ref.includes(`${tableName}.`));
120
143
  if (reference) {
121
144
  r.references.push(reference);
122
145
  r.refTables.push(table);
@@ -126,90 +149,217 @@ export class UpsertBuilder {
126
149
  references: [],
127
150
  refTables: []
128
151
  });
129
- const extractFields = _.uniq(references).map((reference)=>reference.split(".")[1]);
130
- // 내부 참조 있는 경우 필터하여 분리
131
- const groups = _.groupBy(table.rows, (row)=>Object.entries(row).some(([, value])=>isRefField(value)) ? "selfRef" : "normal");
132
- const normalRows = groups.normal ?? [];
133
- const selfRefRows = groups.selfRef ?? [];
134
- const chunks = chunkSize ? _.chunk(normalRows, chunkSize) : [
135
- normalRows
136
- ];
152
+ const extractFields = unique(references).map((reference)=>reference.split(".")[1]).filter((field)=>field !== undefined);
153
+ // 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)
154
+ const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);
155
+ if (hasCircular) {
156
+ throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);
157
+ }
158
+ // upsert 모드일 때 유니크 인덱스가 없으면 에러
159
+ if (mode === "upsert" && table.uniqueIndexes.length === 0) {
160
+ throw new Error(`${tableName}에 unique index가 정의되지 않아 upsert를 할 수 없습니다.`);
161
+ }
137
162
  const uuidMap = new Map();
138
- for (const chunk of chunks){
139
- const q = wdb.insert(chunk).into(tableName);
140
- if (mode === "insert") {
141
- await q;
142
- } else if (mode === "upsert") {
143
- await q.onDuplicateUpdate.apply(q, Object.keys(normalRows[0]));
144
- }
145
- // upsert된 row들을 다시 조회하여 uuidMap에 저장
146
- const uuids = chunk.map((row)=>row.uuid);
147
- const upsertedRows = await wdb(tableName).select(_.uniq([
163
+ const allIds = [];
164
+ // 레벨별로 순차 처리
165
+ for (const levelRows of levels){
166
+ // 이전 레벨에서 얻은 ID로 자기 참조 해결
167
+ const resolvedRows = levelRows.map((row)=>{
168
+ const resolved = {
169
+ ...row
170
+ };
171
+ for (const [key, value] of Object.entries(row)){
172
+ if (isRefField(value) && value.of === tableName) {
173
+ const parent = uuidMap.get(value.uuid);
174
+ if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);
175
+ resolved[key] = parent[value.use ?? "id"];
176
+ Naite.t("puri:ub-ref-resolved", {
177
+ tableName,
178
+ field: key,
179
+ from: {
180
+ of: value.of,
181
+ uuid: value.uuid,
182
+ use: value.use ?? "id"
183
+ },
184
+ to: resolved[key]
185
+ });
186
+ }
187
+ }
188
+ return resolved;
189
+ });
190
+ // 현재 레벨 upsert
191
+ const levelChunks = chunkSize ? chunk(resolvedRows, chunkSize) : [
192
+ resolvedRows
193
+ ];
194
+ const selectFields = unique([
148
195
  "uuid",
149
196
  "id",
150
197
  ...extractFields
151
- ])).whereIn("uuid", uuids);
152
- upsertedRows.forEach((row)=>{
153
- uuidMap.set(row.uuid, row);
154
- });
198
+ ]);
199
+ for (const dataChunk of levelChunks){
200
+ if (dataChunk.length === 0) continue;
201
+ let resultRows;
202
+ if (mode === "insert") {
203
+ // INSERT 모드
204
+ await wdb.insert(dataChunk).into(tableName);
205
+ const uuids = dataChunk.map((r)=>r.uuid);
206
+ resultRows = await wdb(tableName).select(selectFields).whereIn("uuid", uuids);
207
+ } else {
208
+ // UPSERT 모드 (uniqueIndexes 이미 체크됨)
209
+ const conflictColumns = table.uniqueIndexes[0].columns;
210
+ const updateColumns = Object.keys(dataChunk[0]).filter((col)=>col !== "uuid" && !conflictColumns.includes(col));
211
+ // RETURNING으로 결과 받기
212
+ const query = wdb.insert(dataChunk).into(tableName).onConflict(conflictColumns);
213
+ // updateColumns가 비어있으면 ignore(), 아니면 merge()
214
+ if (updateColumns.length === 0) {
215
+ resultRows = await query.ignore().returning(selectFields);
216
+ } else {
217
+ resultRows = await query.merge(updateColumns).returning(selectFields);
218
+ }
219
+ }
220
+ // 양쪽 모드 공통 처리
221
+ for (const row of resultRows){
222
+ uuidMap.set(row.uuid, row);
223
+ allIds.push(row.id);
224
+ }
225
+ }
155
226
  }
156
227
  // 해당 테이블 참조를 실제 밸류로 변경
157
- refTables.map((table)=>{
228
+ for (const table of refTables){
158
229
  table.rows = table.rows.map((row)=>{
159
- Object.keys(row).map((key)=>{
230
+ for (const key of Object.keys(row)){
160
231
  const prop = row[key];
161
232
  if (isRefField(prop) && prop.of === tableName) {
162
233
  const parent = uuidMap.get(prop.uuid);
163
- if (parent === undefined) {
234
+ if (!parent) {
164
235
  console.error(prop);
165
236
  throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);
166
237
  }
167
- row[key] = parent[prop.use ?? "id"];
238
+ const resolvedValue = parent[prop.use ?? "id"];
239
+ row[key] = resolvedValue;
240
+ Naite.t("puri:ub-ref-resolved", {
241
+ tableName,
242
+ field: key,
243
+ from: {
244
+ of: prop.of,
245
+ uuid: prop.uuid,
246
+ use: prop.use ?? "id"
247
+ },
248
+ to: resolvedValue
249
+ });
168
250
  }
169
- });
251
+ }
170
252
  return row;
171
253
  });
172
- });
173
- const allIds = Array.from(uuidMap.values()).map((row)=>row.id);
174
- // 자기 참조가 있는 경우 재귀적으로 upsert
175
- if (selfRefRows.length > 0) {
176
- // 처리된 데이터를 제외하고 다시 upsert
177
- table.rows = selfRefRows;
178
- const selfRefIds = await this.upsert(wdb, tableName, chunkSize);
179
- allIds.push(...selfRefIds);
180
- } else {
181
- // 자기 참조가 없으면 해당 테이블의 데이터 초기화
182
- table.rows = [];
183
- table.references.clear();
184
- table.uniquesMap.clear();
185
254
  }
255
+ // 해당 테이블의 데이터 초기화
256
+ table.rows = [];
257
+ table.references.clear();
258
+ table.uniquesMap.clear();
259
+ Naite.t("puri:ub-upserted", {
260
+ tableName,
261
+ mode,
262
+ rowCount: allIds.length,
263
+ returnedIds: allIds
264
+ });
186
265
  return allIds;
187
266
  }
188
267
  async updateBatch(wdb, tableName, options) {
189
- options = _.defaults(options, {
190
- chunkSize: 500,
191
- where: "id"
192
- });
268
+ options = {
269
+ ...options,
270
+ chunkSize: options?.chunkSize ?? 500,
271
+ where: options?.where ?? "id"
272
+ };
193
273
  if (this.hasTable(tableName) === false) {
194
274
  return;
195
275
  }
196
276
  const table = this.tables.get(tableName);
197
- if (table.rows.length === 0) {
277
+ if (!table) {
278
+ throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);
279
+ } else if (table.rows.length === 0) {
198
280
  return;
199
281
  }
200
282
  const whereColumns = Array.isArray(options.where) ? options.where : [
201
283
  options.where ?? "id"
202
284
  ];
203
285
  const rows = table.rows.map((_row)=>{
204
- const { uuid, ...row } = _row;
286
+ const { uuid: _, ...row } = _row; // uuid 제외
205
287
  return row;
206
288
  });
207
289
  await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);
290
+ Naite.t("puri:ub-batch-updated", {
291
+ tableName,
292
+ rowCount: rows.length,
293
+ whereColumns
294
+ });
208
295
  // updateBatch 완료 후 처리된 데이터 제거
209
296
  table.rows = [];
210
297
  table.references.clear();
211
298
  table.uniquesMap.clear();
212
299
  }
300
+ // ============================================================================
301
+ // Private Helpers
302
+ // ============================================================================
303
+ /**
304
+ * rows를 의존성 순서에 따라 레벨별로 그룹화
305
+ * - 자기 참조 없는 경우 : 모든 rows가 Level 0
306
+ * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화
307
+ */ buildInsertLevels(rows, tableName) {
308
+ // 1. 자기 참조가 없으면 한 레벨로 처리
309
+ const hasSelfRef = rows.flatMap((row)=>Object.values(row)).some((value)=>isRefField(value) && value.of === tableName);
310
+ if (!hasSelfRef) return {
311
+ levels: [
312
+ rows
313
+ ],
314
+ hasCircular: false
315
+ };
316
+ // 2. uuid → row 매핑 (중복 uuid 방지)
317
+ const rowByUuid = new Map();
318
+ for (const row of rows){
319
+ const uuid = row.uuid;
320
+ if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);
321
+ rowByUuid.set(uuid, row);
322
+ }
323
+ let pending = Array.from(rowByUuid.values());
324
+ const levels = [];
325
+ const inserted = new Set();
326
+ // 3. 레벨별 분류
327
+ while(pending.length > 0){
328
+ const currentLevel = [];
329
+ const nextPending = [];
330
+ for (const row of pending){
331
+ // 이 row가 참조하는 자기 참조들
332
+ const selfRefs = Object.values(row).filter((value)=>isRefField(value) && value.of === tableName);
333
+ // 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함
334
+ const canInsert = selfRefs.every((ref)=>{
335
+ if (!rowByUuid.has(ref.uuid)) {
336
+ throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);
337
+ }
338
+ return inserted.has(ref.uuid);
339
+ });
340
+ if (canInsert) {
341
+ currentLevel.push(row);
342
+ } else {
343
+ nextPending.push(row);
344
+ }
345
+ }
346
+ // 순환 참조 감지
347
+ if (currentLevel.length === 0) return {
348
+ levels: [],
349
+ hasCircular: true
350
+ };
351
+ // 레벨 확정 + inserted 갱신
352
+ levels.push(currentLevel);
353
+ for (const row of currentLevel){
354
+ inserted.add(row.uuid);
355
+ }
356
+ pending = nextPending;
357
+ }
358
+ return {
359
+ levels,
360
+ hasCircular: false
361
+ };
362
+ }
213
363
  }
214
364
 
215
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/upsert-builder.ts"],"sourcesContent":["import { randomUUID } from \"crypto\";\nimport * as _ from \"lodash-es\";\nimport { Knex } from \"knex\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { nonNullable } from \"../utils/utils\";\nimport { RowWithId, batchUpdate } from \"./_batch_update\";\n\ntype TableData = {\n  references: Set<string>;\n  rows: any[];\n  uniqueIndexes: { name?: string; columns: string[] }[];\n  uniquesMap: Map<string, string>;\n};\nexport type UBRef = {\n  uuid: string;\n  of: string;\n  use?: string;\n};\nexport function isRefField(field: any): field is UBRef {\n  return (\n    field !== undefined &&\n    field !== null &&\n    field.of !== undefined &&\n    field.uuid !== undefined\n  );\n}\n\nexport class UpsertBuilder {\n  tables: Map<string, TableData>;\n  constructor() {\n    this.tables = new Map();\n  }\n\n  getTable(tableName: string): TableData {\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      const tableSpec = (() => {\n        try {\n          return EntityManager.getTableSpec(tableName);\n        } catch {\n          return null;\n        }\n      })();\n\n      this.tables.set(tableName, {\n        references: new Set(),\n        rows: [],\n        uniqueIndexes: tableSpec?.uniqueIndexes ?? [],\n        uniquesMap: new Map<string, string>(),\n      });\n    }\n\n    return this.tables.get(tableName)!;\n  }\n\n  hasTable(tableName: string): boolean {\n    return this.tables.has(tableName);\n  }\n\n  register<T extends string>(\n    tableName: string,\n    row: {\n      [key in T]?:\n        | UBRef\n        | string\n        | number\n        | boolean\n        | bigint\n        | null\n        | object\n        | unknown;\n    }\n  ): UBRef {\n    const table = this.getTable(tableName);\n\n    // 해당 테이블의 unique 인덱스를 순회하며 키 생성\n    const uniqueKeys = table.uniqueIndexes\n      .map((unqIndex) => {\n        const uniqueKeyArray = unqIndex.columns.map((unqCol) => {\n          const val = row[unqCol as keyof typeof row];\n          if (isRefField(val)) {\n            return val.uuid;\n          } else {\n            return row[unqCol as keyof typeof row] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입\n          }\n        });\n\n        // 값이 모두 null인 경우 키 생성 패스\n        if (uniqueKeyArray.length === 0) {\n          return null;\n        }\n        return uniqueKeyArray.join(\"---delimiter--\");\n      })\n      .filter(nonNullable);\n\n    // uuid 생성 로직\n    const uuid: string = (() => {\n      // 키를 순회하여 이미 존재하는 키가 있는지 확인\n      if (uniqueKeys.length > 0) {\n        for (const uniqueKey of uniqueKeys) {\n          if (table.uniquesMap.has(uniqueKey)) {\n            return table.uniquesMap.get(uniqueKey)!; // 이미 has 체크를 했으므로 undefined 불가능\n          }\n        }\n      }\n\n      // 찾을 수 없는 경우 생성\n      return randomUUID();\n    })();\n\n    // 모든 유니크키에 대해 유니크맵에 uuid 저장\n    if (uniqueKeys.length > 0) {\n      for (const uniqueKey of uniqueKeys) {\n        table.uniquesMap.set(uniqueKey, uuid);\n      }\n    }\n\n    // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가\n    // 이 정보를 나중에 치환할 때 사용\n    row = Object.keys(row).reduce((r, rowKey) => {\n      const rowValue = row[rowKey as keyof typeof row];\n\n      if (isRefField(rowValue)) {\n        rowValue.use ??= \"id\";\n        table.references.add(rowValue.of + \".\" + rowValue.use);\n        r[rowKey] = rowValue;\n      } else if (typeof rowValue === \"object\" && !(rowValue instanceof Date)) {\n        // object인 경우 JSON으로 변환\n        r[rowKey] = rowValue === null ? null : JSON.stringify(rowValue);\n      } else {\n        r[rowKey] = rowValue;\n      }\n      return r;\n    }, {} as any);\n\n    table.rows.push({\n      uuid,\n      ...row,\n    });\n\n    return {\n      of: tableName,\n      uuid: (row as { uuid?: string }).uuid ?? uuid,\n    };\n  }\n\n  async upsert(\n    wdb: Knex,\n    tableName: string,\n    chunkSize?: number\n  ): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"upsert\", chunkSize);\n  }\n  async insertOnly(\n    wdb: Knex,\n    tableName: string,\n    chunkSize?: number\n  ): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"insert\", chunkSize);\n  }\n\n  async upsertOrInsert(\n    wdb: Knex,\n    tableName: string,\n    mode: \"upsert\" | \"insert\",\n    chunkSize?: number\n  ): Promise<number[]> {\n    if (this.hasTable(tableName) === false) {\n      return [];\n    }\n\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);\n    } else if (table.rows.length === 0) {\n      throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);\n    }\n\n    if (\n      table.rows.some((row) =>\n        Object.entries(row).some(\n          ([, value]) => isRefField(value) && value.of !== tableName\n        )\n      )\n    ) {\n      throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);\n    }\n\n    // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출\n    const { references, refTables } = Array.from(this.tables).reduce(\n      (r, [, table]) => {\n        const reference = Array.from(table.references.values()).find((ref) =>\n          ref.includes(tableName + \".\")\n        );\n        if (reference) {\n          r.references.push(reference);\n          r.refTables.push(table);\n        }\n\n        return r;\n      },\n      {\n        references: [] as string[],\n        refTables: [] as TableData[],\n      }\n    );\n    const extractFields = _.uniq(references).map(\n      (reference) => reference.split(\".\")[1]\n    );\n\n    // 내부 참조 있는 경우 필터하여 분리\n    const groups = _.groupBy(table.rows, (row) =>\n      Object.entries(row).some(([, value]) => isRefField(value))\n        ? \"selfRef\"\n        : \"normal\"\n    );\n    const normalRows = groups.normal ?? [];\n    const selfRefRows = groups.selfRef ?? [];\n\n    const chunks = chunkSize ? _.chunk(normalRows, chunkSize) : [normalRows];\n    const uuidMap = new Map<string, any>();\n\n    for (const chunk of chunks) {\n      const q = wdb.insert(chunk).into(tableName);\n      if (mode === \"insert\") {\n        await q;\n      } else if (mode === \"upsert\") {\n        await q.onDuplicateUpdate.apply(q, Object.keys(normalRows[0]));\n      }\n\n      // upsert된 row들을 다시 조회하여 uuidMap에 저장\n      const uuids = chunk.map((row) => row.uuid);\n      const upsertedRows = await wdb(tableName)\n        .select(_.uniq([\"uuid\", \"id\", ...extractFields]))\n        .whereIn(\"uuid\", uuids);\n      upsertedRows.forEach((row: any) => {\n        uuidMap.set(row.uuid, row);\n      });\n    }\n\n    // 해당 테이블 참조를 실제 밸류로 변경\n    refTables.map((table) => {\n      table.rows = table.rows.map((row) => {\n        Object.keys(row).map((key) => {\n          const prop = row[key];\n          if (isRefField(prop) && prop.of === tableName) {\n            const parent = uuidMap.get(prop.uuid);\n            if (parent === undefined) {\n              console.error(prop);\n              throw new Error(\n                `존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`\n              );\n            }\n            row[key] = parent[prop.use ?? \"id\"];\n          }\n        });\n        return row;\n      });\n    });\n\n    const allIds = Array.from(uuidMap.values()).map((row) => row.id);\n\n    // 자기 참조가 있는 경우 재귀적으로 upsert\n    if (selfRefRows.length > 0) {\n      // 처리된 데이터를 제외하고 다시 upsert\n      table.rows = selfRefRows;\n      const selfRefIds = await this.upsert(wdb, tableName, chunkSize);\n      allIds.push(...selfRefIds);\n    } else {\n      // 자기 참조가 없으면 해당 테이블의 데이터 초기화\n      table.rows = [];\n      table.references.clear();\n      table.uniquesMap.clear();\n    }\n\n    return allIds;\n  }\n\n  async updateBatch(\n    wdb: Knex,\n    tableName: string,\n    options?: {\n      chunkSize?: number;\n      where?: string | string[];\n    }\n  ): Promise<void> {\n    options = _.defaults(options, {\n      chunkSize: 500,\n      where: \"id\",\n    });\n\n    if (this.hasTable(tableName) === false) {\n      return;\n    }\n    const table = this.tables.get(tableName)!;\n    if (table.rows.length === 0) {\n      return;\n    }\n\n    const whereColumns = Array.isArray(options.where)\n      ? options.where\n      : [options.where ?? \"id\"];\n    const rows = table.rows.map((_row) => {\n      const { uuid, ...row } = _row;\n      return row as RowWithId<string>;\n    });\n\n    await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);\n\n    // updateBatch 완료 후 처리된 데이터 제거\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n  }\n}\n"],"names":["randomUUID","_","EntityManager","nonNullable","batchUpdate","isRefField","field","undefined","of","uuid","UpsertBuilder","tables","Map","getTable","tableName","table","get","tableSpec","getTableSpec","set","references","Set","rows","uniqueIndexes","uniquesMap","hasTable","has","register","row","uniqueKeys","map","unqIndex","uniqueKeyArray","columns","unqCol","val","length","join","filter","uniqueKey","Object","keys","reduce","r","rowKey","rowValue","use","add","Date","JSON","stringify","push","upsert","wdb","chunkSize","upsertOrInsert","insertOnly","mode","Error","some","entries","value","refTables","Array","from","reference","values","find","ref","includes","extractFields","uniq","split","groups","groupBy","normalRows","normal","selfRefRows","selfRef","chunks","chunk","uuidMap","q","insert","into","onDuplicateUpdate","apply","uuids","upsertedRows","select","whereIn","forEach","key","prop","parent","console","error","allIds","id","selfRefIds","clear","updateBatch","options","defaults","where","whereColumns","isArray","_row"],"mappings":"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"}
365
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/upsert-builder.ts"],"sourcesContent":["import { randomUUID } from \"crypto\";\nimport type { Knex } from \"knex\";\nimport { unique } from \"radashi\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport { assertDefined, chunk, nonNullable } from \"../utils/utils\";\nimport { batchUpdate, type RowWithId } from \"./_batch_update\";\n\ntype TableData = {\n  references: Set<string>;\n  rows: Record<string, unknown>[];\n  uniqueIndexes: { name?: string; columns: string[] }[];\n  uniquesMap: Map<string, string>;\n};\nexport type UBRef = {\n  uuid: string;\n  of: string;\n  use?: string;\n};\nexport function isRefField(field: unknown): field is UBRef {\n  return (\n    field !== undefined &&\n    field !== null &&\n    (field as UBRef)?.of !== undefined &&\n    (field as UBRef)?.uuid !== undefined\n  );\n}\n\nexport class UpsertBuilder {\n  tables: Map<string, TableData>;\n  constructor() {\n    this.tables = new Map();\n  }\n\n  getTable(tableName: string): TableData {\n    const table = this.tables.get(tableName);\n    if (table) {\n      return table;\n    }\n\n    const tableSpec = (() => {\n      try {\n        return EntityManager.getTableSpec(tableName);\n      } catch {\n        return null;\n      }\n    })();\n\n    const tableData = {\n      references: new Set<string>(),\n      rows: [],\n      uniqueIndexes: tableSpec?.uniqueIndexes ?? [],\n      uniquesMap: new Map<string, string>(),\n    };\n    this.tables.set(tableName, tableData);\n    return tableData;\n  }\n\n  hasTable(tableName: string): boolean {\n    return this.tables.has(tableName);\n  }\n\n  register<T extends string>(\n    tableName: string,\n    row: {\n      [key in T]?: UBRef | string | number | boolean | bigint | null | object | unknown;\n    },\n  ): UBRef {\n    const table = this.getTable(tableName);\n\n    // 해당 테이블의 unique 인덱스를 순회하며 키 생성\n    const uniqueKeys = table.uniqueIndexes\n      .map((unqIndex) => {\n        const uniqueKeyArray = unqIndex.columns.map((unqCol) => {\n          const val = row[unqCol as keyof typeof row];\n          if (isRefField(val)) {\n            return val.uuid;\n          } else {\n            return row[unqCol as keyof typeof row] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입\n          }\n        });\n\n        // 값이 모두 null인 경우 키 생성 패스\n        if (uniqueKeyArray.length === 0) {\n          return null;\n        }\n        return uniqueKeyArray.join(\"---delimiter--\");\n      })\n      .filter(nonNullable);\n\n    // uuid 생성 로직\n    const { uuid, isReused } = (() => {\n      // 키를 순회하여 이미 존재하는 키가 있는지 확인\n      if (uniqueKeys.length > 0) {\n        for (const uniqueKey of uniqueKeys) {\n          if (table.uniquesMap.has(uniqueKey)) {\n            return {\n              uuid: assertDefined(table.uniquesMap.get(uniqueKey), \"Unique key not found\"),\n              isReused: true,\n            };\n          }\n        }\n      }\n\n      // 찾을 수 없는 경우 생성\n      return { uuid: randomUUID(), isReused: false };\n    })();\n\n    // 모든 유니크키에 대해 유니크맵에 uuid 저장\n    if (uniqueKeys.length > 0) {\n      for (const uniqueKey of uniqueKeys) {\n        table.uniquesMap.set(uniqueKey, uuid);\n      }\n    }\n\n    // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가\n    // 이 정보를 나중에 치환할 때 사용\n    row = Object.fromEntries(\n      Object.entries(row).map(([rowKey, rowValue]) => {\n        if (isRefField(rowValue)) {\n          rowValue.use ??= \"id\";\n          table.references.add(`${rowValue.of}.${rowValue.use}`);\n          return [rowKey, rowValue];\n        } else if (typeof rowValue === \"object\" && !(rowValue instanceof Date)) {\n          // object인 경우 JSON으로 변환\n          return [rowKey, rowValue === null ? null : JSON.stringify(rowValue)];\n        } else {\n          return [rowKey, rowValue];\n        }\n      }),\n    ) as { [key in T]?: unknown };\n\n    table.rows.push({\n      uuid,\n      ...row,\n    });\n\n    const result: UBRef = {\n      of: tableName,\n      uuid: (row as { uuid?: string }).uuid ?? uuid,\n    };\n\n    Naite.t(\"puri:ub-register\", {\n      tableName,\n      uuid: result.uuid,\n      isUuidReused: isReused,\n      row,\n    });\n\n    return result;\n  }\n\n  async upsert(wdb: Knex, tableName: string, chunkSize?: number): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"upsert\", chunkSize);\n  }\n  async insertOnly(wdb: Knex, tableName: string, chunkSize?: number): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"insert\", chunkSize);\n  }\n\n  async upsertOrInsert(\n    wdb: Knex,\n    tableName: string,\n    mode: \"upsert\" | \"insert\",\n    chunkSize?: number,\n  ): Promise<number[]> {\n    if (this.hasTable(tableName) === false) {\n      return [];\n    }\n\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);\n    } else if (table.rows.length === 0) {\n      throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);\n    }\n\n    if (\n      table.rows.some((row) =>\n        Object.entries(row).some(([, value]) => isRefField(value) && value.of !== tableName),\n      )\n    ) {\n      throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);\n    }\n\n    // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출\n    const { references, refTables } = Array.from(this.tables).reduce(\n      (r, [, table]) => {\n        const reference = Array.from(table.references.values()).find((ref) =>\n          ref.includes(`${tableName}.`),\n        );\n        if (reference) {\n          r.references.push(reference);\n          r.refTables.push(table);\n        }\n\n        return r;\n      },\n      {\n        references: [] as string[],\n        refTables: [] as TableData[],\n      },\n    );\n    const extractFields = unique(references)\n      .map((reference) => reference.split(\".\")[1])\n      .filter((field): field is string => field !== undefined);\n\n    // 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)\n    const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);\n\n    if (hasCircular) {\n      throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);\n    }\n\n    // upsert 모드일 때 유니크 인덱스가 없으면 에러\n    if (mode === \"upsert\" && table.uniqueIndexes.length === 0) {\n      throw new Error(`${tableName}에 unique index가 정의되지 않아 upsert를 할 수 없습니다.`);\n    }\n\n    const uuidMap = new Map<string, unknown>();\n    const allIds: number[] = [];\n\n    // 레벨별로 순차 처리\n    for (const levelRows of levels) {\n      // 이전 레벨에서 얻은 ID로 자기 참조 해결\n      const resolvedRows = levelRows.map((row) => {\n        const resolved = { ...row };\n        for (const [key, value] of Object.entries(row)) {\n          if (isRefField(value) && value.of === tableName) {\n            const parent = uuidMap.get(value.uuid);\n\n            if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);\n\n            resolved[key] = (parent as Record<string, unknown>)[value.use ?? \"id\"];\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: value.of, uuid: value.uuid, use: value.use ?? \"id\" },\n              to: resolved[key],\n            });\n          }\n        }\n        return resolved;\n      });\n\n      // 현재 레벨 upsert\n      const levelChunks = chunkSize ? chunk(resolvedRows, chunkSize) : [resolvedRows];\n      const selectFields = unique([\"uuid\", \"id\", ...extractFields]);\n\n      for (const dataChunk of levelChunks) {\n        if (dataChunk.length === 0) continue;\n\n        let resultRows: { uuid: string; id: number; [key: string]: unknown }[];\n\n        if (mode === \"insert\") {\n          // INSERT 모드\n          await wdb.insert(dataChunk).into(tableName);\n\n          const uuids = dataChunk.map((r) => r.uuid);\n          resultRows = await wdb(tableName)\n            .select(selectFields)\n            .whereIn(\"uuid\", uuids as readonly string[]);\n        } else {\n          // UPSERT 모드 (uniqueIndexes 이미 체크됨)\n          const conflictColumns = table.uniqueIndexes[0].columns;\n          const updateColumns = Object.keys(dataChunk[0]).filter(\n            (col) => col !== \"uuid\" && !conflictColumns.includes(col),\n          );\n\n          // RETURNING으로 결과 받기\n          const query = wdb.insert(dataChunk).into(tableName).onConflict(conflictColumns);\n\n          // updateColumns가 비어있으면 ignore(), 아니면 merge()\n          if (updateColumns.length === 0) {\n            resultRows = await query.ignore().returning(selectFields);\n          } else {\n            resultRows = await query.merge(updateColumns).returning(selectFields);\n          }\n        }\n\n        // 양쪽 모드 공통 처리\n        for (const row of resultRows) {\n          uuidMap.set(row.uuid, row);\n          allIds.push(row.id);\n        }\n      }\n    }\n\n    // 해당 테이블 참조를 실제 밸류로 변경\n    for (const table of refTables) {\n      table.rows = table.rows.map((row) => {\n        for (const key of Object.keys(row)) {\n          const prop = row[key];\n          if (isRefField(prop) && prop.of === tableName) {\n            const parent = uuidMap.get(prop.uuid);\n            if (!parent) {\n              console.error(prop);\n              throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);\n            }\n            const resolvedValue = (parent as Record<string, unknown>)[prop.use ?? \"id\"];\n            row[key] = resolvedValue;\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: prop.of, uuid: prop.uuid, use: prop.use ?? \"id\" },\n              to: resolvedValue,\n            });\n          }\n        }\n        return row;\n      });\n    }\n\n    // 해당 테이블의 데이터 초기화\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n\n    Naite.t(\"puri:ub-upserted\", {\n      tableName,\n      mode,\n      rowCount: allIds.length,\n      returnedIds: allIds,\n    });\n\n    return allIds;\n  }\n\n  async updateBatch(\n    wdb: Knex,\n    tableName: string,\n    options?: {\n      chunkSize?: number;\n      where?: string | string[];\n    },\n  ): Promise<void> {\n    options = {\n      ...options,\n      chunkSize: options?.chunkSize ?? 500,\n      where: options?.where ?? \"id\",\n    };\n\n    if (this.hasTable(tableName) === false) {\n      return;\n    }\n    const table = this.tables.get(tableName);\n    if (!table) {\n      throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);\n    } else if (table.rows.length === 0) {\n      return;\n    }\n\n    const whereColumns = Array.isArray(options.where) ? options.where : [options.where ?? \"id\"];\n    const rows = table.rows.map((_row) => {\n      const { uuid: _, ...row } = _row; // uuid 제외\n      return row as RowWithId<string>;\n    });\n\n    await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);\n\n    Naite.t(\"puri:ub-batch-updated\", {\n      tableName,\n      rowCount: rows.length,\n      whereColumns,\n    });\n\n    // updateBatch 완료 후 처리된 데이터 제거\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n  }\n\n  // ============================================================================\n  // Private Helpers\n  // ============================================================================\n\n  /**\n   * rows를 의존성 순서에 따라 레벨별로 그룹화\n   * - 자기 참조 없는 경우 : 모든 rows가 Level 0\n   * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화\n   */\n  private buildInsertLevels(\n    rows: Record<string, unknown>[],\n    tableName: string,\n  ): { levels: Record<string, unknown>[][]; hasCircular: boolean } {\n    // 1. 자기 참조가 없으면 한 레벨로 처리\n    const hasSelfRef = rows\n      .flatMap((row) => Object.values(row))\n      .some((value) => isRefField(value) && value.of === tableName);\n    if (!hasSelfRef) return { levels: [rows], hasCircular: false };\n\n    // 2. uuid → row 매핑 (중복 uuid 방지)\n    const rowByUuid = new Map<string, Record<string, unknown>>();\n    for (const row of rows) {\n      const uuid = row.uuid as string | undefined;\n      if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);\n      rowByUuid.set(uuid, row);\n    }\n\n    let pending = Array.from(rowByUuid.values());\n    const levels: Record<string, unknown>[][] = [];\n    const inserted = new Set<string>();\n\n    // 3. 레벨별 분류\n    while (pending.length > 0) {\n      const currentLevel: Record<string, unknown>[] = [];\n      const nextPending: Record<string, unknown>[] = [];\n\n      for (const row of pending) {\n        // 이 row가 참조하는 자기 참조들\n        const selfRefs = Object.values(row).filter(\n          (value) => isRefField(value) && value.of === tableName,\n        ) as UBRef[];\n\n        // 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함\n        const canInsert = selfRefs.every((ref) => {\n          if (!rowByUuid.has(ref.uuid)) {\n            throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);\n          }\n          return inserted.has(ref.uuid);\n        });\n\n        if (canInsert) {\n          currentLevel.push(row);\n        } else {\n          nextPending.push(row);\n        }\n      }\n\n      // 순환 참조 감지\n      if (currentLevel.length === 0) return { levels: [], hasCircular: true };\n\n      // 레벨 확정 + inserted 갱신\n      levels.push(currentLevel);\n      for (const row of currentLevel) {\n        inserted.add(row.uuid as string);\n      }\n\n      pending = nextPending;\n    }\n\n    return { levels, hasCircular: false };\n  }\n}\n"],"names":["randomUUID","unique","EntityManager","Naite","assertDefined","chunk","nonNullable","batchUpdate","isRefField","field","undefined","of","uuid","UpsertBuilder","tables","Map","getTable","tableName","table","get","tableSpec","getTableSpec","tableData","references","Set","rows","uniqueIndexes","uniquesMap","set","hasTable","has","register","row","uniqueKeys","map","unqIndex","uniqueKeyArray","columns","unqCol","val","length","join","filter","isReused","uniqueKey","Object","fromEntries","entries","rowKey","rowValue","use","add","Date","JSON","stringify","push","result","t","isUuidReused","upsert","wdb","chunkSize","upsertOrInsert","insertOnly","mode","Error","some","value","refTables","Array","from","reduce","r","reference","values","find","ref","includes","extractFields","split","levels","hasCircular","buildInsertLevels","uuidMap","allIds","levelRows","resolvedRows","resolved","key","parent","to","levelChunks","selectFields","dataChunk","resultRows","insert","into","uuids","select","whereIn","conflictColumns","updateColumns","keys","col","query","onConflict","ignore","returning","merge","id","prop","console","error","resolvedValue","clear","rowCount","returnedIds","updateBatch","options","where","whereColumns","isArray","_row","_","hasSelfRef","flatMap","rowByUuid","pending","inserted","currentLevel","nextPending","selfRefs","canInsert","every"],"mappings":"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"}