sonamu 0.7.53 → 0.8.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 (271) hide show
  1. package/dist/api/config.d.ts +9 -1
  2. package/dist/api/config.d.ts.map +1 -1
  3. package/dist/api/config.js +1 -1
  4. package/dist/api/sonamu.d.ts +21 -1
  5. package/dist/api/sonamu.d.ts.map +1 -1
  6. package/dist/api/sonamu.js +159 -65
  7. package/dist/auth/plugins/entity-definitions/anonymous.d.ts +10 -0
  8. package/dist/auth/plugins/entity-definitions/anonymous.d.ts.map +1 -0
  9. package/dist/auth/plugins/entity-definitions/anonymous.js +23 -0
  10. package/dist/auth/plugins/entity-definitions/api-key.d.ts +9 -0
  11. package/dist/auth/plugins/entity-definitions/api-key.d.ts.map +1 -0
  12. package/dist/auth/plugins/entity-definitions/api-key.js +199 -0
  13. package/dist/auth/plugins/entity-definitions/index.d.ts +6 -0
  14. package/dist/auth/plugins/entity-definitions/index.d.ts.map +1 -1
  15. package/dist/auth/plugins/entity-definitions/index.js +20 -2
  16. package/dist/auth/plugins/entity-definitions/jwt.d.ts +9 -0
  17. package/dist/auth/plugins/entity-definitions/jwt.d.ts.map +1 -0
  18. package/dist/auth/plugins/entity-definitions/jwt.js +67 -0
  19. package/dist/auth/plugins/entity-definitions/organization.d.ts +9 -0
  20. package/dist/auth/plugins/entity-definitions/organization.d.ts.map +1 -0
  21. package/dist/auth/plugins/entity-definitions/organization.js +424 -0
  22. package/dist/auth/plugins/entity-definitions/passkey.d.ts +10 -0
  23. package/dist/auth/plugins/entity-definitions/passkey.d.ts.map +1 -0
  24. package/dist/auth/plugins/entity-definitions/passkey.js +129 -0
  25. package/dist/auth/plugins/entity-definitions/sso.d.ts +10 -0
  26. package/dist/auth/plugins/entity-definitions/sso.d.ts.map +1 -0
  27. package/dist/auth/plugins/entity-definitions/sso.js +110 -0
  28. package/dist/auth/plugins/entity-definitions/types.d.ts +1 -1
  29. package/dist/auth/plugins/entity-definitions/types.d.ts.map +1 -1
  30. package/dist/auth/plugins/entity-definitions/types.js +1 -1
  31. package/dist/auth/plugins/wrappers/admin.d.ts.map +1 -1
  32. package/dist/auth/plugins/wrappers/admin.js +2 -4
  33. package/dist/auth/plugins/wrappers/anonymous.d.ts +18 -0
  34. package/dist/auth/plugins/wrappers/anonymous.d.ts.map +1 -0
  35. package/dist/auth/plugins/wrappers/anonymous.js +26 -0
  36. package/dist/auth/plugins/wrappers/api-key.d.ts +18 -0
  37. package/dist/auth/plugins/wrappers/api-key.d.ts.map +1 -0
  38. package/dist/auth/plugins/wrappers/api-key.js +38 -0
  39. package/dist/auth/plugins/wrappers/index.d.ts +6 -0
  40. package/dist/auth/plugins/wrappers/index.d.ts.map +1 -1
  41. package/dist/auth/plugins/wrappers/index.js +7 -1
  42. package/dist/auth/plugins/wrappers/jwt.d.ts +18 -0
  43. package/dist/auth/plugins/wrappers/jwt.d.ts.map +1 -0
  44. package/dist/auth/plugins/wrappers/jwt.js +30 -0
  45. package/dist/auth/plugins/wrappers/organization.d.ts +18 -0
  46. package/dist/auth/plugins/wrappers/organization.d.ts.map +1 -0
  47. package/dist/auth/plugins/wrappers/organization.js +67 -0
  48. package/dist/auth/plugins/wrappers/passkey.d.ts +18 -0
  49. package/dist/auth/plugins/wrappers/passkey.d.ts.map +1 -0
  50. package/dist/auth/plugins/wrappers/passkey.js +32 -0
  51. package/dist/auth/plugins/wrappers/phone-number.d.ts.map +1 -1
  52. package/dist/auth/plugins/wrappers/phone-number.js +2 -4
  53. package/dist/auth/plugins/wrappers/sso.d.ts +853 -0
  54. package/dist/auth/plugins/wrappers/sso.d.ts.map +1 -0
  55. package/dist/auth/plugins/wrappers/sso.js +36 -0
  56. package/dist/auth/plugins/wrappers/two-factor.d.ts.map +1 -1
  57. package/dist/auth/plugins/wrappers/two-factor.js +2 -4
  58. package/dist/auth/plugins/wrappers/username.d.ts.map +1 -1
  59. package/dist/auth/plugins/wrappers/username.js +2 -4
  60. package/dist/bin/build-config.d.ts +2 -2
  61. package/dist/bin/build-config.js +6 -7
  62. package/dist/bin/cli.js +417 -32
  63. package/dist/bin/fixture.d.ts +27 -0
  64. package/dist/bin/fixture.d.ts.map +1 -0
  65. package/dist/bin/fixture.js +245 -0
  66. package/dist/cache/decorator.d.ts +4 -3
  67. package/dist/cache/decorator.d.ts.map +1 -1
  68. package/dist/cache/decorator.js +5 -4
  69. package/dist/cone/cone-generator.d.ts +33 -0
  70. package/dist/cone/cone-generator.d.ts.map +1 -0
  71. package/dist/cone/cone-generator.js +286 -0
  72. package/dist/database/_batch_update.d.ts.map +1 -1
  73. package/dist/database/_batch_update.js +16 -2
  74. package/dist/database/puri-subset.test-d.js +1 -1
  75. package/dist/database/puri-subset.types.d.ts +1 -1
  76. package/dist/database/puri-subset.types.d.ts.map +1 -1
  77. package/dist/database/puri-subset.types.js +1 -1
  78. package/dist/database/puri.d.ts +4 -0
  79. package/dist/database/puri.d.ts.map +1 -1
  80. package/dist/database/puri.js +20 -2
  81. package/dist/database/upsert-builder.d.ts.map +1 -1
  82. package/dist/database/upsert-builder.js +19 -3
  83. package/dist/dict/en.d.ts +15 -0
  84. package/dist/dict/en.d.ts.map +1 -1
  85. package/dist/dict/en.js +2 -1
  86. package/dist/dict/ko.d.ts +15 -0
  87. package/dist/dict/ko.d.ts.map +1 -1
  88. package/dist/dict/ko.js +2 -1
  89. package/dist/dict/rc-keys.d.ts +28 -0
  90. package/dist/dict/rc-keys.d.ts.map +1 -1
  91. package/dist/dict/rc-keys.js +31 -1
  92. package/dist/dict/sd.d.ts.map +1 -1
  93. package/dist/dict/sd.js +20 -4
  94. package/dist/entity/entity-manager.d.ts +298 -2
  95. package/dist/entity/entity-manager.d.ts.map +1 -1
  96. package/dist/entity/entity-manager.js +4 -1
  97. package/dist/entity/entity-template-cone.d.ts +14 -0
  98. package/dist/entity/entity-template-cone.d.ts.map +1 -0
  99. package/dist/entity/entity-template-cone.js +222 -0
  100. package/dist/entity/entity.d.ts +47 -2
  101. package/dist/entity/entity.d.ts.map +1 -1
  102. package/dist/entity/entity.js +161 -14
  103. package/dist/ssr/renderer.js +3 -3
  104. package/dist/syncer/api-parser.js +12 -1
  105. package/dist/syncer/checksum.d.ts +0 -14
  106. package/dist/syncer/checksum.d.ts.map +1 -1
  107. package/dist/syncer/checksum.js +1 -23
  108. package/dist/syncer/syncer-actions.d.ts.map +1 -1
  109. package/dist/syncer/syncer-actions.js +8 -2
  110. package/dist/syncer/syncer.d.ts +1 -1
  111. package/dist/syncer/syncer.d.ts.map +1 -1
  112. package/dist/syncer/syncer.js +17 -10
  113. package/dist/tasks/workflow-manager.d.ts +13 -1
  114. package/dist/tasks/workflow-manager.d.ts.map +1 -1
  115. package/dist/tasks/workflow-manager.js +18 -1
  116. package/dist/template/entity-converter.js +4 -4
  117. package/dist/template/helpers.d.ts +10 -0
  118. package/dist/template/helpers.d.ts.map +1 -1
  119. package/dist/template/helpers.js +48 -1
  120. package/dist/template/implementations/entry-server.template.d.ts +1 -1
  121. package/dist/template/implementations/entry-server.template.js +7 -2
  122. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  123. package/dist/template/implementations/generated.template.js +5 -1
  124. package/dist/template/implementations/generated_http.template.d.ts +1 -0
  125. package/dist/template/implementations/generated_http.template.d.ts.map +1 -1
  126. package/dist/template/implementations/generated_http.template.js +6 -2
  127. package/dist/template/implementations/generated_sso.template.d.ts.map +1 -1
  128. package/dist/template/implementations/generated_sso.template.js +29 -8
  129. package/dist/template/implementations/queries.template.d.ts.map +1 -1
  130. package/dist/template/implementations/queries.template.js +9 -1
  131. package/dist/template/implementations/sd.template.d.ts +1 -1
  132. package/dist/template/implementations/sd.template.d.ts.map +1 -1
  133. package/dist/template/implementations/sd.template.js +28 -4
  134. package/dist/template/implementations/services.template.d.ts.map +1 -1
  135. package/dist/template/implementations/services.template.js +12 -12
  136. package/dist/template/implementations/view_form.template.d.ts +11 -7
  137. package/dist/template/implementations/view_form.template.d.ts.map +1 -1
  138. package/dist/template/implementations/view_form.template.js +97 -87
  139. package/dist/template/implementations/view_list.template.d.ts +3 -3
  140. package/dist/template/implementations/view_list.template.d.ts.map +1 -1
  141. package/dist/template/implementations/view_list.template.js +115 -109
  142. package/dist/template/implementations/view_search_input.template.d.ts.map +1 -1
  143. package/dist/template/implementations/view_search_input.template.js +18 -14
  144. package/dist/template/zod-converter.d.ts.map +1 -1
  145. package/dist/template/zod-converter.js +95 -7
  146. package/dist/testing/_relation-graph.js +1 -1
  147. package/dist/testing/data-explorer.d.ts +61 -0
  148. package/dist/testing/data-explorer.d.ts.map +1 -0
  149. package/dist/testing/data-explorer.js +274 -0
  150. package/dist/testing/faker-mappings.d.ts +20 -0
  151. package/dist/testing/faker-mappings.d.ts.map +1 -0
  152. package/dist/testing/faker-mappings.js +421 -0
  153. package/dist/testing/fixture-generator.d.ts +161 -0
  154. package/dist/testing/fixture-generator.d.ts.map +1 -0
  155. package/dist/testing/fixture-generator.js +954 -0
  156. package/dist/testing/fixture-manager.d.ts +6 -1
  157. package/dist/testing/fixture-manager.d.ts.map +1 -1
  158. package/dist/testing/fixture-manager.js +72 -4
  159. package/dist/testing/index.d.ts +3 -0
  160. package/dist/testing/index.d.ts.map +1 -1
  161. package/dist/testing/index.js +4 -1
  162. package/dist/types/types.d.ts +1520 -26
  163. package/dist/types/types.d.ts.map +1 -1
  164. package/dist/types/types.js +136 -22
  165. package/dist/ui/ai-client.d.ts.map +1 -1
  166. package/dist/ui/ai-client.js +9 -4
  167. package/dist/ui/api.d.ts.map +1 -1
  168. package/dist/ui/api.js +303 -24
  169. package/dist/ui-web/assets/index-CsUr-_pV.js +254 -0
  170. package/dist/ui-web/assets/index-T42zzs1K.css +1 -0
  171. package/dist/ui-web/index.html +2 -2
  172. package/dist/utils/fs-utils.d.ts +2 -1
  173. package/dist/utils/fs-utils.d.ts.map +1 -1
  174. package/dist/utils/fs-utils.js +14 -3
  175. package/package.json +19 -11
  176. package/src/api/config.ts +12 -1
  177. package/src/api/sonamu.ts +179 -65
  178. package/src/auth/plugins/entity-definitions/anonymous.ts +17 -0
  179. package/src/auth/plugins/entity-definitions/api-key.ts +93 -0
  180. package/src/auth/plugins/entity-definitions/index.ts +18 -0
  181. package/src/auth/plugins/entity-definitions/jwt.ts +35 -0
  182. package/src/auth/plugins/entity-definitions/organization.ts +215 -0
  183. package/src/auth/plugins/entity-definitions/passkey.ts +64 -0
  184. package/src/auth/plugins/entity-definitions/sso.ts +62 -0
  185. package/src/auth/plugins/entity-definitions/types.ts +11 -1
  186. package/src/auth/plugins/wrappers/admin.ts +1 -3
  187. package/src/auth/plugins/wrappers/anonymous.ts +30 -0
  188. package/src/auth/plugins/wrappers/api-key.ts +42 -0
  189. package/src/auth/plugins/wrappers/index.ts +6 -0
  190. package/src/auth/plugins/wrappers/jwt.ts +34 -0
  191. package/src/auth/plugins/wrappers/organization.ts +73 -0
  192. package/src/auth/plugins/wrappers/passkey.ts +36 -0
  193. package/src/auth/plugins/wrappers/phone-number.ts +1 -3
  194. package/src/auth/plugins/wrappers/sso.ts +40 -0
  195. package/src/auth/plugins/wrappers/two-factor.ts +1 -3
  196. package/src/auth/plugins/wrappers/username.ts +1 -3
  197. package/src/bin/build-config.ts +6 -6
  198. package/src/bin/cli.ts +452 -31
  199. package/src/bin/fixture.ts +302 -0
  200. package/src/cache/decorator.ts +4 -3
  201. package/src/cone/cone-generator.ts +363 -0
  202. package/src/database/_batch_update.ts +11 -0
  203. package/src/database/puri-subset.test-d.ts +13 -13
  204. package/src/database/puri-subset.types.ts +1 -1
  205. package/src/database/puri.ts +43 -1
  206. package/src/database/upsert-builder.ts +16 -2
  207. package/src/dict/en.ts +1 -0
  208. package/src/dict/ko.ts +1 -0
  209. package/src/dict/rc-keys.ts +32 -0
  210. package/src/dict/sd.ts +23 -3
  211. package/src/entity/entity-manager.ts +4 -0
  212. package/src/entity/entity-template-cone.ts +298 -0
  213. package/src/entity/entity.ts +189 -13
  214. package/src/shared/app.shared.ts.txt +5 -0
  215. package/src/shared/web.shared.ts.txt +9 -5
  216. package/src/skills/project/README.md +21 -0
  217. package/src/skills/project/architecture.md +373 -0
  218. package/src/skills/project/business-logic.md +270 -0
  219. package/src/skills/project/requirements.md +160 -0
  220. package/src/skills/sonamu/SKILL.md +168 -3
  221. package/src/skills/sonamu/api.md +102 -0
  222. package/src/skills/sonamu/database.md +220 -1
  223. package/src/skills/sonamu/entity-relations.md +89 -1
  224. package/src/skills/sonamu/fixture-cli.md +501 -0
  225. package/src/skills/sonamu/frontend.md +214 -0
  226. package/src/skills/sonamu/i18n.md +95 -0
  227. package/src/skills/sonamu/model.md +153 -0
  228. package/src/skills/sonamu/project-init.md +178 -8
  229. package/src/skills/sonamu/scaffolding.md +112 -0
  230. package/src/skills/sonamu/subset.md +9 -3
  231. package/src/skills/sonamu/testing.md +287 -2
  232. package/src/skills/sonamu/workflow.md +70 -5
  233. package/src/ssr/renderer.ts +2 -2
  234. package/src/syncer/api-parser.ts +12 -0
  235. package/src/syncer/checksum.ts +0 -38
  236. package/src/syncer/syncer-actions.ts +7 -1
  237. package/src/syncer/syncer.ts +16 -5
  238. package/src/tasks/workflow-manager.ts +29 -8
  239. package/src/template/entity-converter.ts +3 -3
  240. package/src/template/helpers.ts +49 -0
  241. package/src/template/implementations/entry-server.template.ts +1 -1
  242. package/src/template/implementations/generated.template.ts +4 -0
  243. package/src/template/implementations/generated_http.template.ts +1 -0
  244. package/src/template/implementations/generated_sso.template.ts +40 -11
  245. package/src/template/implementations/queries.template.ts +8 -0
  246. package/src/template/implementations/sd.template.ts +22 -3
  247. package/src/template/implementations/services.template.ts +11 -10
  248. package/src/template/implementations/view_form.template.ts +111 -101
  249. package/src/template/implementations/view_list.template.ts +120 -119
  250. package/src/template/implementations/view_search_input.template.ts +17 -13
  251. package/src/template/zod-converter.ts +103 -6
  252. package/src/testing/_relation-graph.ts +1 -1
  253. package/src/testing/data-explorer.ts +427 -0
  254. package/src/testing/faker-mappings.ts +434 -0
  255. package/src/testing/fixture-generator.ts +1166 -0
  256. package/src/testing/fixture-manager.ts +91 -6
  257. package/src/testing/index.ts +3 -0
  258. package/src/types/types.ts +222 -26
  259. package/src/ui/ai-client.ts +9 -1
  260. package/src/ui/api.ts +429 -23
  261. package/src/utils/fs-utils.ts +14 -1
  262. package/dist/template/implementations/view_enums_select.template.d.ts +0 -17
  263. package/dist/template/implementations/view_enums_select.template.d.ts.map +0 -1
  264. package/dist/template/implementations/view_enums_select.template.js +0 -62
  265. package/dist/template/implementations/view_id_async_select.template.d.ts +0 -17
  266. package/dist/template/implementations/view_id_async_select.template.d.ts.map +0 -1
  267. package/dist/template/implementations/view_id_async_select.template.js +0 -125
  268. package/dist/ui-web/assets/index-Bd_2AkLb.css +0 -1
  269. package/dist/ui-web/assets/index-BpSbhQWo.js +0 -225
  270. package/src/template/implementations/view_enums_select.template.ts +0 -65
  271. package/src/template/implementations/view_id_async_select.template.ts +0 -139
package/dist/ui/api.js CHANGED
@@ -4,6 +4,8 @@ import inflection from "inflection";
4
4
  import path from "path";
5
5
  import { range } from "radashi";
6
6
  import { Sonamu } from "../api/sonamu.js";
7
+ import { DB } from "../database/db.js";
8
+ import { createKnexInstance } from "../database/knex.js";
7
9
  import { SD } from "../dict/sd.js";
8
10
  import { sonamuDictionary } from "../dict/sonamu-dictionary.js";
9
11
  import { EntityManager } from "../entity/entity-manager.js";
@@ -11,6 +13,8 @@ import { BadRequestException, isSoException, ServiceUnavailableException } from
11
13
  import { Migrator } from "../migration/migrator.js";
12
14
  import { SlackConfirm } from "../migration/slack-confirm.js";
13
15
  import { TemplateManager } from "../template/template-manager.js";
16
+ import { DataExplorer } from "../testing/data-explorer.js";
17
+ import { FixtureGenerator } from "../testing/fixture-generator.js";
14
18
  import { FixtureManager } from "../testing/fixture-manager.js";
15
19
  import { BUILT_IN_TYPE_IDS, TemplateKey } from "../types/types.js";
16
20
  import { nonNullable } from "../utils/utils.js";
@@ -550,6 +554,76 @@ export async function sonamuUIApiPlugin(fastify) {
550
554
  await entity.save();
551
555
  });
552
556
  });
557
+ server.post("/api/entity/updateCone", async (request)=>{
558
+ return await waitForHMRCompleted(async ()=>{
559
+ const { entityId, target, propName, enumId, subsetKey, cone } = request.body;
560
+ const entity = EntityManager.get(entityId);
561
+ if (target === "entity") {
562
+ entity.cone = cone;
563
+ } else if (target === "prop" && propName) {
564
+ const prop = entity.props.find((p)=>p.name === propName);
565
+ if (prop) {
566
+ prop.cone = cone;
567
+ }
568
+ } else if (target === "enum" && enumId) {
569
+ entity.enumCones[enumId] = cone;
570
+ } else if (target === "subset" && subsetKey) {
571
+ entity.subsetCones[subsetKey] = cone;
572
+ }
573
+ await entity.save();
574
+ return true;
575
+ });
576
+ });
577
+ server.post("/api/entity/generateCones", async (request, reply)=>{
578
+ return await waitForHMRCompleted(async ()=>{
579
+ const { entityId, preserveExisting, onlyEmpty, locale } = request.body;
580
+ try {
581
+ // Entity 존재 여부 확인
582
+ const entity = EntityManager.get(entityId);
583
+ // locale 기본값: Sonamu.config.i18n.defaultLocale 사용
584
+ const effectiveLocale = locale ?? Sonamu.config.i18n.defaultLocale;
585
+ // Cone 생성
586
+ const result = await entity.generateCones({
587
+ preserveExisting: preserveExisting ?? true,
588
+ onlyEmpty: onlyEmpty ?? false,
589
+ locale: effectiveLocale
590
+ });
591
+ return result;
592
+ } catch (error) {
593
+ const message = error instanceof Error ? error.message : String(error);
594
+ // Entity not found
595
+ if (message.includes("존재하지 않는 Entity")) {
596
+ reply.status(404);
597
+ return {
598
+ success: false,
599
+ error: `Entity not found: ${entityId}`
600
+ };
601
+ }
602
+ // API 키 없음
603
+ if (message.includes("ANTHROPIC_API_KEY not found")) {
604
+ reply.status(500);
605
+ return {
606
+ success: false,
607
+ error: "API key not configured"
608
+ };
609
+ }
610
+ // Rate limit
611
+ if (message.includes("Rate limit exceeded")) {
612
+ reply.status(429);
613
+ return {
614
+ success: false,
615
+ error: "Rate limit exceeded. Please try again later."
616
+ };
617
+ }
618
+ // 기타 에러
619
+ reply.status(500);
620
+ return {
621
+ success: false,
622
+ error: `Cone generation failed: ${message}`
623
+ };
624
+ }
625
+ });
626
+ });
553
627
  server.get("/api/entity/getTableColumns", async (request)=>{
554
628
  const { entityId } = request.query;
555
629
  const entity = EntityManager.get(entityId);
@@ -667,43 +741,26 @@ export async function sonamuUIApiPlugin(fastify) {
667
741
  return await migrator.generatePreparedCodes();
668
742
  });
669
743
  server.post("/api/scaffolding/getStatus", async (request)=>{
670
- const { templateGroupName, entityIds, templateKeys: _templateKeys, enumIds } = request.body;
744
+ const { entityIds, templateKeys: _templateKeys } = request.body;
671
745
  if ((entityIds ?? []).length === 0) {
672
746
  throw new BadRequestException(SD("sonamu.error.entityIdsRequired"));
673
747
  } else if ((_templateKeys ?? []).length === 0) {
674
748
  throw new BadRequestException(SD("sonamu.error.templateKeysRequired"));
675
- } else if (templateGroupName === "Enums" && (enumIds ?? []).length === 0) {
676
- throw new BadRequestException(SD("sonamu.error.enumIdsRequired"));
677
749
  }
678
750
  // sorting
679
751
  entityIds.sort((a, b)=>a < b ? -1 : a > b ? 1 : 0);
680
752
  const templateKeys = TemplateKey.options.filter((tk)=>_templateKeys.includes(tk));
681
753
  const combinations = entityIds.flatMap((entityId)=>{
682
- if (templateGroupName === "Enums") {
683
- const entityIds = [
754
+ return templateKeys.map((templateKey)=>[
684
755
  entityId,
685
- ...EntityManager.getChildrenIds(entityId)
686
- ];
687
- const allEnumIds = entityIds.flatMap((entityId)=>Object.keys(EntityManager.get(entityId).enumLabels));
688
- return templateKeys.flatMap((templateKey)=>allEnumIds.filter((enumId)=>enumIds.includes(enumId)).map((enumId)=>[
689
- entityId,
690
- templateKey,
691
- enumId
692
- ]));
693
- } else {
694
- return templateKeys.map((templateKey)=>[
695
- entityId,
696
- templateKey
697
- ]);
698
- }
756
+ templateKey
757
+ ]);
699
758
  });
700
- const statuses = await Promise.all(combinations.map(async ([entityId, templateKey, enumId])=>{
701
- const { subPath, fullPath, isExists } = await Sonamu.syncer.checkExistsGenCode(entityId, templateKey, enumId);
759
+ const statuses = await Promise.all(combinations.map(async ([entityId, templateKey])=>{
760
+ const { subPath, fullPath, isExists } = await Sonamu.syncer.checkExistsGenCode(entityId, templateKey);
702
761
  return {
703
762
  entityId,
704
- templateGroupName,
705
763
  templateKey,
706
- enumId,
707
764
  subPath,
708
765
  fullPath,
709
766
  isExists
@@ -811,6 +868,228 @@ export async function sonamuUIApiPlugin(fastify) {
811
868
  server.post("/api/i18n/checkUsage", async (request)=>{
812
869
  return sonamuDictionary.checkUsage(request.body.keys);
813
870
  });
871
+ // Tasks API
872
+ server.get("/api/tasks/status", async ()=>{
873
+ try {
874
+ Sonamu.workflows;
875
+ return {
876
+ active: true
877
+ };
878
+ } catch {
879
+ return {
880
+ active: false
881
+ };
882
+ }
883
+ });
884
+ server.get("/api/tasks/workflowDefinitions", async ()=>{
885
+ const definitions = Sonamu.workflows.workflowDefinitions;
886
+ return {
887
+ definitions
888
+ };
889
+ });
890
+ server.get("/api/tasks/workflowRuns", async (request)=>{
891
+ const backend = Sonamu.workflows.backend;
892
+ const { limit, after, before, order, status, workflowName, createdAfter, createdBefore } = request.query;
893
+ return backend.listWorkflowRuns({
894
+ limit: limit ? Number.parseInt(limit, 10) : undefined,
895
+ after,
896
+ before,
897
+ order,
898
+ status: status ? status.split(",") : undefined,
899
+ workflowName: workflowName || undefined,
900
+ createdAfter: createdAfter ? new Date(createdAfter) : undefined,
901
+ createdBefore: createdBefore ? new Date(createdBefore) : undefined
902
+ });
903
+ });
904
+ server.get("/api/tasks/workflowRuns/:id", async (request)=>{
905
+ const backend = Sonamu.workflows.backend;
906
+ const workflowRun = await backend.getWorkflowRun({
907
+ workflowRunId: request.params.id
908
+ });
909
+ if (!workflowRun) {
910
+ throw new Error(`Workflow run not found: ${request.params.id}`);
911
+ }
912
+ return workflowRun;
913
+ });
914
+ server.post("/api/tasks/workflowRuns/:id/cancel", async (request)=>{
915
+ const backend = Sonamu.workflows.backend;
916
+ return backend.cancelWorkflowRun({
917
+ workflowRunId: request.params.id
918
+ });
919
+ });
920
+ server.post("/api/tasks/workflowRuns/:id/pause", async (request)=>{
921
+ const backend = Sonamu.workflows.backend;
922
+ return backend.pauseWorkflowRun({
923
+ workflowRunId: request.params.id
924
+ });
925
+ });
926
+ server.post("/api/tasks/workflowRuns/:id/resume", async (request)=>{
927
+ const backend = Sonamu.workflows.backend;
928
+ return backend.resumeWorkflowRun({
929
+ workflowRunId: request.params.id
930
+ });
931
+ });
932
+ server.get("/api/tasks/workflowRuns/:id/steps", async (request)=>{
933
+ const backend = Sonamu.workflows.backend;
934
+ const { limit, after, before } = request.query;
935
+ return backend.listStepAttempts({
936
+ workflowRunId: request.params.id,
937
+ limit: limit ? Number.parseInt(limit, 10) : undefined,
938
+ after,
939
+ before
940
+ });
941
+ });
942
+ /**
943
+ * Health Check API
944
+ * MCP 도구가 Sonamu 서버를 자동 감지하기 위한 엔드포인트
945
+ */ server.get("/api/sonamu/health", async (request)=>{
946
+ const address = request.server.server.address();
947
+ const port = address && typeof address === "object" ? address.port : 0;
948
+ return {
949
+ ok: true,
950
+ project: process.cwd().split("/").pop() || "unknown",
951
+ port,
952
+ timestamp: new Date().toISOString()
953
+ };
954
+ });
955
+ /**
956
+ * Fixture 생성 API
957
+ */ server.post("/api/sonamu/fixture/generate", async (request, reply)=>{
958
+ const { entity, count = 1, overrides, targetDb = "fixture" } = request.body;
959
+ try {
960
+ // 타겟 DB 설정 가져오기
961
+ const dbConfig = targetDb === "fixture" ? Sonamu.dbConfig.fixture : Sonamu.dbConfig.test;
962
+ // Knex 인스턴스 생성
963
+ const db = createKnexInstance(dbConfig);
964
+ // FixtureGenerator 생성
965
+ const generator = new FixtureGenerator(db, db, targetDb, EntityManager);
966
+ // 단일 Entity 배치 생성
967
+ const fixtures = await generator.generateBatch([
968
+ {
969
+ entity,
970
+ count,
971
+ overrides: overrides ?? {}
972
+ }
973
+ ]);
974
+ // Knex 연결 종료
975
+ await db.destroy();
976
+ return {
977
+ success: true,
978
+ entity,
979
+ count: fixtures.length,
980
+ fixtures,
981
+ targetDb
982
+ };
983
+ } catch (error) {
984
+ reply.status(400);
985
+ return {
986
+ success: false,
987
+ error: error instanceof Error ? error.message : String(error)
988
+ };
989
+ }
990
+ });
991
+ /**
992
+ * Fixture 데이터 탐색 API
993
+ */ server.post("/api/sonamu/fixture/explore", async (request, reply)=>{
994
+ const { entity, strategy, limit = 10, where } = request.body;
995
+ try {
996
+ // Fixture DB 설정 가져오기
997
+ const fixtureDbConfig = Sonamu.dbConfig.fixture;
998
+ // Knex 인스턴스 생성
999
+ const fixtureDb = createKnexInstance(fixtureDbConfig);
1000
+ // DataExplorer 생성
1001
+ const explorer = new DataExplorer(fixtureDb, EntityManager);
1002
+ const data = await explorer.explore(entity, {
1003
+ strategy,
1004
+ limit,
1005
+ where
1006
+ });
1007
+ // Knex 연결 종료
1008
+ await fixtureDb.destroy();
1009
+ return {
1010
+ success: true,
1011
+ entity,
1012
+ strategy,
1013
+ count: data.length,
1014
+ data
1015
+ };
1016
+ } catch (error) {
1017
+ reply.status(400);
1018
+ return {
1019
+ success: false,
1020
+ error: error instanceof Error ? error.message : String(error)
1021
+ };
1022
+ }
1023
+ });
1024
+ /**
1025
+ * Fixture 데이터 가져오기 (fetch) API
1026
+ * production/development DB에서 실제 데이터를 fixture DB로 import
1027
+ */ server.post("/api/sonamu/fixture/fetch", async (request, reply)=>{
1028
+ const { entity, strategy = "recent", limit = 10, includeRelations = true, maxDepth = 2 } = request.body;
1029
+ try {
1030
+ // Source DB (production/development) - 읽기 전용
1031
+ const sourceDb = DB.getDB("r");
1032
+ // Target DB (fixture)
1033
+ const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);
1034
+ // FixtureGenerator 생성
1035
+ const generator = new FixtureGenerator(sourceDb, fixtureDb, "fixture", EntityManager);
1036
+ // production 데이터를 fixture DB로 import
1037
+ const results = await generator.importFromSource(entity, {
1038
+ strategy,
1039
+ limit,
1040
+ includeRelations,
1041
+ maxDepth
1042
+ });
1043
+ // Knex 연결 종료 (sourceDb는 Sonamu가 관리하므로 destroy하지 않음)
1044
+ await fixtureDb.destroy();
1045
+ return {
1046
+ success: true,
1047
+ entity,
1048
+ strategy,
1049
+ count: results.length,
1050
+ imported: results
1051
+ };
1052
+ } catch (error) {
1053
+ reply.status(400);
1054
+ return {
1055
+ success: false,
1056
+ error: error instanceof Error ? error.message : String(error)
1057
+ };
1058
+ }
1059
+ });
1060
+ /**
1061
+ * Fixture 데이터 삭제 (clean) API
1062
+ * FK 순서를 고려하여 안전하게 삭제
1063
+ */ server.post("/api/sonamu/fixture/clean", async (request, reply)=>{
1064
+ const { entities } = request.body;
1065
+ try {
1066
+ // Fixture DB 연결
1067
+ const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);
1068
+ // 삭제할 Entity 목록 결정
1069
+ const targetEntities = entities && entities.length > 0 ? entities : EntityManager.getAllIds();
1070
+ // Entity ID를 테이블명으로 변환 (snake_case 복수형)
1071
+ const tableNames = targetEntities.map((entityId)=>{
1072
+ const entity = EntityManager.get(entityId);
1073
+ return entity.table;
1074
+ });
1075
+ // PostgreSQL: TRUNCATE CASCADE로 FK 순서 무관하게 안전하게 삭제
1076
+ // CASCADE 옵션으로 의존성 있는 데이터도 함께 삭제
1077
+ await fixtureDb.raw(`TRUNCATE TABLE ${tableNames.map((t)=>`"${t}"`).join(", ")} RESTART IDENTITY CASCADE`);
1078
+ // Knex 연결 종료
1079
+ await fixtureDb.destroy();
1080
+ return {
1081
+ success: true,
1082
+ cleaned: tableNames,
1083
+ count: tableNames.length
1084
+ };
1085
+ } catch (error) {
1086
+ reply.status(400);
1087
+ return {
1088
+ success: false,
1089
+ error: error instanceof Error ? error.message : String(error)
1090
+ };
1091
+ }
1092
+ });
814
1093
  // ui-web 빌드 파일 서빙
815
1094
  const uiDistPath = path.resolve(import.meta.dirname, "../ui-web");
816
1095
  // 정적 파일 서빙: 루트 폴더 전체 (assets, setting.svg 등)
@@ -831,4 +1110,4 @@ export async function sonamuUIApiPlugin(fastify) {
831
1110
  });
832
1111
  }
833
1112
 
834
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/ui/api.ts"],"sourcesContent":["import { execSync } from \"child_process\";\nimport type { FastifyInstance } from \"fastify\";\nimport fs from \"fs\";\nimport inflection from \"inflection\";\nimport path from \"path\";\nimport { range } from \"radashi\";\nimport { Sonamu } from \"../api/sonamu\";\nimport type { SonamuDBConfig } from \"../database/db\";\nimport { SD } from \"../dict/sd\";\nimport { sonamuDictionary } from \"../dict/sonamu-dictionary\";\nimport type { Entity } from \"../entity/entity\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport {\n  BadRequestException,\n  isSoException,\n  ServiceUnavailableException,\n} from \"../exceptions/so-exceptions\";\nimport { type MigrationResult, Migrator } from \"../migration/migrator\";\nimport { SlackConfirm, type SlackConfirmPendingResult } from \"../migration/slack-confirm\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport { type DuplicateCheckOptions, FixtureManager } from \"../testing/fixture-manager\";\nimport {\n  BUILT_IN_TYPE_IDS,\n  type EntityIndex,\n  type EntityProp,\n  type EntitySubsetRow,\n  type FixtureRecord,\n  type FixtureSearchOptions,\n  type FlattenSubsetRow,\n  type PathAndCode,\n  TemplateKey,\n} from \"../types/types\";\nimport { nonNullable } from \"../utils/utils\";\nimport { setAiApi } from \"./ai-api\";\n\nexport async function sonamuUIApiPlugin(fastify: FastifyInstance) {\n  fastify.register(\n    async (server) => {\n      // migrator\n      const migrator = new Migrator();\n\n      // waitForHMRCompleted\n      async function waitForHMRCompleted<T>(fn: () => Promise<T>): Promise<T> {\n        const waitPromise = new Promise<void>((resolve) => {\n          const timeout = setTimeout(() => {\n            resolve();\n          }, 1500);\n\n          const handler = () => {\n            clearTimeout(timeout);\n            Sonamu.syncer.eventEmitter.off(\"onHMRCompleted\", handler);\n            resolve();\n          };\n\n          Sonamu.syncer.eventEmitter.once(\"onHMRCompleted\", handler);\n        });\n\n        const result = await fn();\n        await waitPromise;\n        return result;\n      }\n\n      await setAiApi(server);\n\n      server.get(\"/api/sonamu/config\", async () => {\n        return Sonamu.config;\n      });\n\n      server.get<{\n        Querystring: {\n          entityId?: string;\n          preset?: \"types\" | \"entity.json\" | \"generated\";\n          absPath?: string;\n        };\n      }>(\"/api/tools/openVscode\", async (request) => {\n        const { entityId, preset, absPath } = request.query;\n\n        const targetPath = (() => {\n          if (entityId && preset) {\n            const entity = EntityManager.get(entityId);\n            const { names } = entity;\n\n            const { apiRootPath } = Sonamu;\n            const filename = (() => {\n              switch (preset) {\n                case \"types\":\n                  return `${names.fs}.types.ts`;\n                case \"entity.json\":\n                  return `${names.fs}.entity.json`;\n                case \"generated\":\n                  return `${names.fs}.generated.ts`;\n              }\n            })();\n            return `${apiRootPath}/src/application/${entity.names.parentFs}/${filename}`;\n          } else {\n            if (!absPath) {\n              throw new BadRequestException(SD(\"sonamu.error.presetOrAbsPathRequired\"));\n            }\n            return absPath;\n          }\n        })();\n        execSync(`code ${targetPath}`);\n      });\n\n      server.get<{\n        Querystring: {\n          origin: string;\n          entityId?: string;\n        };\n      }>(\"/api/tools/getSuggestion\", async (request) => {\n        const { origin, entityId } = request.query;\n\n        // 치환 용어집\n        const glossary = new Map<string, string>([\n          [\"status\", \"상태\"],\n          [\"type\", \"타입\"],\n          [\"image\", \"이미지\"],\n          [\"images\", \"이미지리스트\"],\n          [\"url\", \"URL\"],\n          [\"id\", \"ID\"],\n          [\"name\", `{EntityID}명`],\n          [\"title\", \"{EntityID}명\"],\n          [\"parent\", \"상위{EntityID}\"],\n          [\"desc\", \"설명\"],\n          [\"at\", \"일시\"],\n          [\"created\", \"등록\"],\n          [\"updated\", \"수정\"],\n          [\"deleted\", \"삭제\"],\n          [\"by\", \"유저\"],\n          [\"date\", \"일자\"],\n          [\"time\", \"시간\"],\n          [\"ko\", \"(한글)\"],\n          [\"en\", \"(영문)\"],\n          [\"krw\", \"(원)\"],\n          [\"usd\", \"(USD)\"],\n          [\"color\", \"컬러\"],\n          [\"code\", \"코드\"],\n          [\"x\", \"X좌표\"],\n          [\"y\", \"Y좌표\"],\n          [\"current\", \"현재\"],\n          [\"stock\", \"재고\"],\n          [\"total\", \"총\"],\n          [\"admin\", \"관리자\"],\n          [\"group\", \"그룹\"],\n          [\"item\", \"아이템\"],\n          [\"cnt\", \"수량\"],\n          [\"price\", \"가격\"],\n          [\"preset\", \"프리셋\"],\n          [\"acct\", \"계좌\"],\n          [\"tel\", \"전화번호\"],\n          [\"no\", \"번호\"],\n          [\"body\", \"내용\"],\n          [\"content\", \"내용\"],\n          [\"orderno\", \"정렬순서\"],\n          [\"priority\", \"우선순위\"],\n          [\"text\", \"텍스트\"],\n          [\"key\", \"키\"],\n          [\"sum\", \"합산\"],\n          [\"expected\", \"예상\"],\n          [\"actual\", \"실제\"],\n        ]);\n        // 전체 엔티티 순회하며, 엔티티 타이틀과 프롭 설명을 치환 용어집에 추가\n        for (const entityId of EntityManager.getAllIds()) {\n          const entity = EntityManager.get(entityId);\n          if ((entity.title ?? \"\") !== \"\") {\n            glossary.set(inflection.underscore(entity.id), entity.title);\n            glossary.set(\n              inflection.underscore(inflection.pluralize(entity.id)),\n              `${entity.title}리스트`,\n            );\n          }\n\n          entity.props.forEach((prop) => {\n            if (glossary.has(prop.name)) {\n              return;\n            }\n            if (prop.desc) {\n              glossary.set(prop.name, prop.desc.replace(entity.title ?? \"\", \"{EntityID}\"));\n            }\n          });\n        }\n\n        const suggested = (() => {\n          // 단어 분리, 가능한 조합 생성\n          const words = origin.split(\"_\");\n          const combinations = [...range(words.length, 0, -1)].flatMap((len) => {\n            return [\n              ...range(0, words.length - len + 1, (idx) => {\n                return {\n                  len,\n                  w: words.slice(idx, idx + len).join(\"_\"),\n                };\n              }),\n            ];\n          });\n\n          // 조합을 순회하며, 치환 용어집에 있는 단어가 포함된 경우, 치환 용어로 치환\n          const REPLACED_PREFIX = \"#REPLACED//\"; // 치환된 단어를 join 이후에도 식별하기 위해 prefix 추가\n          let remainArr: string[] = [...words];\n          for (const comb of combinations) {\n            const remainStr = remainArr.join(\"_\");\n            if (remainStr.includes(comb.w) && glossary.has(comb.w)) {\n              remainArr = remainStr\n                .replace(comb.w, REPLACED_PREFIX + glossary.get(comb.w))\n                .split(\"_\");\n            }\n          }\n\n          return remainArr\n            .map((r) => {\n              if (r.startsWith(REPLACED_PREFIX)) {\n                return r.replace(REPLACED_PREFIX, \"\");\n              } else {\n                return r.toUpperCase();\n              }\n            })\n            .join(\"\")\n            .replace(/{EntityID}/g, entityId ? EntityManager.get(entityId).title : \"\");\n        })();\n\n        return { suggested };\n      });\n\n      server.get(\"/api/entity/findMany\", async () => {\n        const entityIds = EntityManager.getAllIds();\n\n        function flattenSubsetRows(subsetRows: EntitySubsetRow[]): FlattenSubsetRow[] {\n          return subsetRows.flatMap((subsetRow) => {\n            const { children, ...sRow } = subsetRow;\n            return [sRow, ...flattenSubsetRows(children)];\n          });\n        }\n\n        const entities = await Promise.all(\n          entityIds.map((entityId) => {\n            const entity = EntityManager.get(entityId);\n            const subsetRows = entity.getSubsetRows();\n\n            return {\n              ...entity,\n              flattenSubsetRows: flattenSubsetRows(subsetRows),\n            };\n          }),\n        );\n\n        entities.sort((a, b) => {\n          const aId = a.parentId ?? a.id;\n          const bId = b.parentId ?? b.id;\n          if (aId < bId) return -1;\n          if (aId > bId) return 1;\n          if (aId === bId) {\n            if (a.parentId === undefined) return -1;\n            if (b.parentId === undefined) return 1;\n            return 0;\n          }\n          return 0;\n        });\n        return { entities };\n      });\n\n      server.get<{\n        Querystring: {\n          filter?: \"enums\" | \"types\";\n          reload?: \"1\";\n        };\n      }>(\"/api/entity/typeIds\", async (request): Promise<{ typeIds: string[] }> => {\n        const { filter, reload } = request.query;\n\n        if (reload === \"1\") {\n          await Sonamu.syncer.autoloadTypes();\n        }\n\n        const typeIds = (() => {\n          // 프로젝트에서 정의한 타입들\n          const projectTypeIds = Object.entries(Sonamu.syncer.types)\n            .filter(([_typeId, zodType]) => (zodType._zod.def.type as string) !== \"enum\")\n            .map(([typeId, _zodType]) => typeId);\n\n          // 내장 타입들 (sonamu 코어에서 제공)\n          const builtInTypeIds = [...BUILT_IN_TYPE_IDS];\n\n          // 모든 타입 병합\n          const allTypeIds = [...builtInTypeIds, ...projectTypeIds];\n\n          if (filter === \"types\") {\n            return allTypeIds;\n          }\n\n          const enumIds = EntityManager.getAllIds().flatMap((entityId) => {\n            const entity = EntityManager.get(entityId);\n            return Object.keys(entity.enumLabels);\n          });\n\n          if (filter === \"enums\") {\n            return enumIds;\n          } else {\n            return [...allTypeIds, ...enumIds];\n          }\n        })();\n\n        return {\n          typeIds,\n        };\n      });\n\n      server.post<{\n        Body: {\n          form: {\n            id: string;\n            title: string;\n            table: string;\n            parentId?: string;\n          };\n        };\n      }>(\"/api/entity/create\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { form } = request.body;\n          await Sonamu.syncer.createEntity({ ...form, entityId: form.id });\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n        };\n      }>(\"/api/entity/del\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId } = request.body;\n          return await Sonamu.syncer.delEntity(entityId);\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newValues: {\n            title: string;\n            table: string;\n            parentId?: string;\n          };\n        };\n      }>(\"/api/entity/modifyEntityBase\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, newValues } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.title = newValues.title;\n          entity.table = newValues.table;\n          entity.parentId = newValues.parentId;\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          subsetKey: string;\n          fields: string[];\n          fieldsInternal?: string[];\n        };\n      }>(\"/api/entity/modifySubset\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, subsetKey, fields, fieldsInternal } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.subsets[subsetKey] = fields;\n          if (fieldsInternal !== undefined) {\n            if (fieldsInternal.length > 0) {\n              entity.subsetsInternal[subsetKey] = fieldsInternal;\n            } else {\n              delete entity.subsetsInternal[subsetKey];\n            }\n          }\n          await entity.save();\n\n          return { updated: fields, updatedInternal: fieldsInternal };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          subsetKey: string;\n        };\n      }>(\"/api/entity/delSubset\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, subsetKey } = request.body;\n          const entity = EntityManager.get(entityId);\n          delete entity.subsets[subsetKey];\n          delete entity.subsetsInternal[subsetKey];\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newProp: EntityProp;\n          at?: number;\n        };\n      }>(\"/api/entity/createProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, newProp } = request.body;\n          const entity = EntityManager.get(entityId);\n          await entity.createProp(newProp, at);\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newProp: EntityProp;\n          at: number;\n        };\n      }>(\"/api/entity/modifyProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, newProp } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.modifyProp(newProp, at);\n\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          at: number;\n        };\n      }>(\"/api/entity/delProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.delProp(at);\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          at: number;\n          to: number;\n        };\n      }>(\"/api/entity/moveProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, to } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.moveProp(at, to);\n\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          indexes: EntityIndex[];\n        };\n      }>(\"/api/entity/modifyIndexes\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, indexes } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.indexes = indexes;\n          await entity.save();\n\n          return { updated: indexes };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumLabels: Entity[\"enumLabels\"];\n        };\n      }>(\"/api/entity/modifyEnumLabels\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumLabels } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.enumLabels = enumLabels;\n          await entity.save();\n\n          return { updated: enumLabels };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newEnumId: string;\n        };\n      }>(\"/api/entity/createEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, newEnumId } = request.body;\n          const entity = EntityManager.get(entityId);\n\n          if (entity.enumLabels[newEnumId]) {\n            throw new Error(`이미 존재하는 enumId입니다: ${newEnumId}`);\n          }\n\n          entity.enumLabels[newEnumId] = {\n            ...(newEnumId.endsWith(\"Status\")\n              ? {\n                  active: \"노출\",\n                  hidden: \"숨김\",\n                }\n              : {\n                  \"\": \"\",\n                }),\n          };\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumId: {\n            before: string;\n            after: string;\n          };\n        };\n      }>(\"/api/entity/modifyEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumId } = request.body;\n          const entityIds = EntityManager.getAllIds();\n          const isExists = entityIds.some((entityId) => {\n            const entity = EntityManager.get(entityId);\n            return Object.keys(entity.enumLabels).includes(enumId.after);\n          });\n          if (isExists) {\n            throw new Error(`이미 존재하는 EnumId입니다: ${enumId.after}`);\n          }\n\n          const entity = EntityManager.get(entityId);\n          entity.enumLabels[enumId.after] = entity.enumLabels[enumId.before];\n          delete entity.enumLabels[enumId.before];\n\n          await entity.save();\n\n          for (const entityId of entityIds) {\n            const entity = EntityManager.get(entityId);\n            for (const prop of entity.props) {\n              if (prop.type === \"enum\" && prop.id === enumId.before) {\n                prop.id = enumId.after;\n              }\n            }\n            await entity.save();\n          }\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumId: string;\n        };\n      }>(\"/api/entity/deleteEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumId } = request.body;\n\n          const entityIds = EntityManager.getAllIds();\n          const isReferenced = entityIds\n            .flatMap((entityId) => EntityManager.get(entityId).props)\n            .some((prop) => prop.type === \"enum\" && prop.id === enumId);\n          if (isReferenced) {\n            throw new Error(`${enumId}를 참조하는 프로퍼티가 존재합니다.`);\n          }\n\n          const entity = EntityManager.get(entityId);\n          delete entity.enumLabels[enumId];\n          await entity.save();\n        });\n      });\n\n      server.get<{\n        Querystring: {\n          entityId: string;\n        };\n      }>(\"/api/entity/getTableColumns\", async (request) => {\n        const { entityId } = request.query;\n        const entity = EntityManager.get(entityId);\n        const columns = entity.getTableColumns();\n        return { columns };\n      });\n\n      server.get(\"/api/migrations/status\", async () => {\n        const status = await migrator.getStatus();\n\n        return { status };\n      });\n\n      server.post<{\n        Body: {\n          action: \"apply\" | \"rollback\" | \"shadow\";\n          targets: (keyof SonamuDBConfig)[];\n          force?: boolean;\n          forceReason?: string;\n          requestor?: string;\n        };\n      }>(\n        \"/api/migrations/runAction\",\n        async (request): Promise<MigrationResult | SlackConfirmPendingResult> => {\n          const { action, targets, force, forceReason, requestor } = request.body;\n\n          if (action === \"shadow\") {\n            return migrator.runShadowTest();\n          }\n\n          // Slack 승인 체크 (apply 시에만)\n          if (action === \"apply\") {\n            const slackConfirm = new SlackConfirm();\n            const requiresApproval = targets.some((t) => slackConfirm.isTargetRequiresApproval(t));\n\n            // 로컬 DB인 경우 승인 스킵\n            const localHosts = [\"localhost\", \"127.0.0.1\", \"0.0.0.0\", \"::1\"];\n            const isLocalTarget = targets.every((target) => {\n              const targetConfig = Sonamu.dbConfig[target];\n              const host = (targetConfig?.connection as { host?: string })?.host ?? \"localhost\";\n              return localHosts.includes(host.toLowerCase());\n            });\n\n            if (requiresApproval && slackConfirm.isConfigured() && !isLocalTarget) {\n              const { conns } = await migrator.getStatus();\n\n              // 모든 타겟 DB에서 pending인 마이그레이션의 합집합을 구합니다.\n              const pendingMigrations = [\n                ...new Set(\n                  conns\n                    .filter((conn) => targets.includes(conn.connKey as keyof SonamuDBConfig))\n                    .flatMap((conn) => conn.pending),\n                ),\n              ];\n\n              if (pendingMigrations.length > 0) {\n                // 기존 승인 요청 확인\n                const existing = await slackConfirm.getExistingRequest(pendingMigrations);\n\n                if (existing) {\n                  // 기존 요청이 있으면 승인 상태 확인\n                  const { approved, rejected } = await slackConfirm.checkApproval(\n                    existing.channel,\n                    existing.ts,\n                  );\n\n                  if (approved) {\n                    // 승인됨 → 실행\n                    const result = await migrator.runAction(action, targets);\n                    if (result.length > 0) {\n                      await slackConfirm.logExecution(\n                        existing.channel,\n                        existing.ts,\n                        result,\n                        requestor,\n                      );\n                    }\n                    return result;\n                  } else if (rejected) {\n                    throw new BadRequestException(SD(\"sonamu.error.migrationRejected\"));\n                  } else if (force) {\n                    // Force 진행\n                    await slackConfirm.forceApproval(\n                      existing.channel,\n                      existing.ts,\n                      forceReason ?? \"사유 없음\",\n                      requestor,\n                    );\n                    const result = await migrator.runAction(action, targets);\n                    if (result.length > 0) {\n                      await slackConfirm.logExecution(\n                        existing.channel,\n                        existing.ts,\n                        result,\n                        requestor,\n                      );\n                    }\n                    return result;\n                  } else {\n                    // 대기중\n                    return {\n                      type: \"pending\",\n                      channel: existing.channel,\n                      ts: existing.ts,\n                    };\n                  }\n                } else {\n                  // 새 승인 요청 발송\n                  const { channel, ts } = await slackConfirm.postApprovalRequest(\n                    pendingMigrations,\n                    targets,\n                    requestor,\n                  );\n                  await slackConfirm.saveRequest(pendingMigrations, channel, ts);\n\n                  return {\n                    type: \"pending\",\n                    channel,\n                    ts,\n                  };\n                }\n              }\n            }\n          }\n\n          return migrator.runAction(action, targets);\n        },\n      );\n\n      server.post<{\n        Body: {\n          channel: string;\n          ts: string;\n        };\n      }>(\"/api/migrations/checkApproval\", async (request) => {\n        const { channel, ts } = request.body;\n        const slackConfirm = new SlackConfirm();\n\n        if (!slackConfirm.isConfigured()) {\n          return { approved: true, rejected: false };\n        }\n\n        return slackConfirm.checkApproval(channel, ts);\n      });\n\n      server.post<{\n        Body: {\n          channel: string;\n          ts: string;\n          reason: string;\n          requestor?: string;\n        };\n      }>(\"/api/migrations/forceApproval\", async (request) => {\n        const { channel, ts, reason, requestor } = request.body;\n        const slackConfirm = new SlackConfirm();\n\n        if (!slackConfirm.isConfigured()) {\n          throw new BadRequestException(SD(\"sonamu.error.slackConfirmNotConfigured\"));\n        }\n\n        await slackConfirm.forceApproval(channel, ts, reason, requestor);\n        return { success: true };\n      });\n\n      server.post<{\n        Body: {\n          codeNames: string[];\n        };\n      }>(\"/api/migrations/delCodes\", async (request) => {\n        const { codeNames } = request.body;\n        return await migrator.delCodes(codeNames);\n      });\n\n      server.post(\"/api/migrations/generatePreparedCodes\", async (_requestt) => {\n        return await migrator.generatePreparedCodes();\n      });\n\n      server.post<{\n        Body: {\n          templateGroupName: \"Entity\" | \"Enums\";\n          entityIds: string[];\n          templateKeys: string[];\n          enumIds: string[];\n        };\n      }>(\"/api/scaffolding/getStatus\", async (request) => {\n        const { templateGroupName, entityIds, templateKeys: _templateKeys, enumIds } = request.body;\n        if ((entityIds ?? []).length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.entityIdsRequired\"));\n        } else if ((_templateKeys ?? []).length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.templateKeysRequired\"));\n        } else if (templateGroupName === \"Enums\" && (enumIds ?? []).length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.enumIdsRequired\"));\n        }\n\n        // sorting\n        entityIds.sort((a, b) => (a < b ? -1 : a > b ? 1 : 0));\n        const templateKeys = TemplateKey.options.filter((tk) => _templateKeys.includes(tk));\n\n        const combinations = entityIds.flatMap((entityId) => {\n          if (templateGroupName === \"Enums\") {\n            const entityIds = [entityId, ...EntityManager.getChildrenIds(entityId)];\n            const allEnumIds = entityIds.flatMap((entityId) =>\n              Object.keys(EntityManager.get(entityId).enumLabels),\n            );\n            return templateKeys.flatMap((templateKey) =>\n              allEnumIds\n                .filter((enumId) => enumIds.includes(enumId))\n                .map((enumId) => [entityId, templateKey, enumId]),\n            );\n          } else {\n            return templateKeys.map((templateKey) => [entityId, templateKey]);\n          }\n        });\n\n        const statuses = await Promise.all(\n          combinations.map(async ([entityId, templateKey, enumId]) => {\n            const { subPath, fullPath, isExists } = await Sonamu.syncer.checkExistsGenCode(\n              entityId,\n              templateKey as TemplateKey,\n              enumId,\n            );\n            return {\n              entityId,\n              templateGroupName,\n              templateKey,\n              enumId,\n              subPath,\n              fullPath,\n              isExists,\n            };\n          }),\n        );\n        return { statuses };\n      });\n\n      server.post<{\n        Body: {\n          options: {\n            entityId: string;\n            templateKey: string;\n            enumId?: string;\n            overwrite?: boolean;\n          }[];\n        };\n      }>(\"/api/scaffolding/generate\", async (request) => {\n        const { options } = request.body;\n        if (options.length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.optionsRequired\"));\n        }\n\n        // 1. 모든 템플릿에서 필요한 dict 키를 수집\n        const keys = options.flatMap(({ templateKey }) => {\n          const template = TemplateManager.get(templateKey);\n          return template.getRequiredDictKeys() ?? [];\n        });\n\n        // 2. target별로 ensureDictKeys 호출 (순차 처리)\n        await sonamuDictionary.ensureDictKeys([...new Set(keys)]);\n\n        // 3. 템플릿 생성 (병렬 처리)\n        const result = await Promise.all(\n          options.map(async ({ entityId, templateKey, enumId, overwrite }) => {\n            try {\n              return await Sonamu.syncer.generateTemplate(\n                templateKey as TemplateKey,\n                {\n                  entityId,\n                  enumId,\n                } as {\n                  entityId: string;\n                  enumId?: string;\n                },\n                {\n                  overwrite,\n                },\n              );\n            } catch (e) {\n              if (isSoException(e) && e.statusCode === 541) {\n                return null;\n              } else {\n                console.error(e);\n                throw e;\n              }\n            }\n          }),\n        );\n\n        if (result.filter(nonNullable).length === 0) {\n          throw new ServiceUnavailableException(SD(\"sonamu.error.allFilesGenerated\"));\n        }\n        return result;\n      });\n\n      server.post<{\n        Body: {\n          option: {\n            entityId: string;\n            templateKey: string;\n            enumId?: string;\n          };\n        };\n      }>(\"/api/scaffolding/preview\", async (request): Promise<{ pathAndCodes: PathAndCode[] }> => {\n        const { option } = request.body;\n\n        try {\n          const { templateKey, ...templateOptions } = option;\n          const pathAndCodes = await Sonamu.syncer.renderTemplate(\n            templateKey as TemplateKey,\n            templateOptions,\n          );\n\n          return { pathAndCodes };\n        } catch (e) {\n          console.error(e);\n          throw e;\n        }\n      });\n\n      server.post(\"/api/fixture\", async (request) => {\n        const { sourceDB, targetDB, search, duplicateCheck } = request.body as {\n          sourceDB: keyof SonamuDBConfig;\n          targetDB: keyof SonamuDBConfig;\n          search: FixtureSearchOptions;\n          duplicateCheck?: DuplicateCheckOptions;\n        };\n\n        return FixtureManager.getFixtures(sourceDB, targetDB, search, duplicateCheck);\n      });\n\n      server.post(\"/api/fixture/import\", async (request) => {\n        const { db, fixtures } = request.body as {\n          db: keyof SonamuDBConfig;\n          fixtures: FixtureRecord[];\n        };\n\n        return FixtureManager.insertFixtures(db, fixtures);\n      });\n\n      server.post(\"/api/fixture/addFixtureLoader\", async (request) => {\n        const { code } = request.body as { code: string };\n\n        return FixtureManager.addFixtureLoader(code);\n      });\n\n      server.get(\"/api/i18n/dictionary\", async () => {\n        return sonamuDictionary.getDictionary();\n      });\n\n      server.get(\"/api/i18n/export\", async (_request, reply) => {\n        const { filename, buffer } = await sonamuDictionary.exportToExcel();\n        reply\n          .header(\n            \"Content-Type\",\n            \"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet\",\n          )\n          .header(\"Content-Disposition\", `attachment; filename=\"${filename}\"`)\n          .send(buffer);\n      });\n\n      server.post(\"/api/i18n/import\", async (request) => {\n        const data = await request.file();\n        if (!data) {\n          throw new BadRequestException(SD(\"sonamu.error.fileNotUploaded\"));\n        }\n        const buffer = await data.toBuffer();\n        return sonamuDictionary.importFromExcel(buffer);\n      });\n\n      server.post<{\n        Body: {\n          oldKey: string;\n          newKey: string;\n          source: \"entity\" | \"project\" | \"sonamu\";\n          values: Record<string, string>;\n        };\n      }>(\"/api/i18n/update\", async (request) => {\n        await sonamuDictionary.updateEntry(request.body);\n        return { success: true };\n      });\n\n      server.post<{\n        Body: {\n          key: string;\n          values: Record<string, string>;\n        };\n      }>(\"/api/i18n/create\", async (request) => {\n        await sonamuDictionary.createEntry(request.body);\n        return { success: true };\n      });\n\n      server.post<{\n        Body: {\n          key: string;\n        };\n      }>(\"/api/i18n/delete\", async (request) => {\n        await sonamuDictionary.deleteEntry(request.body.key);\n        return { success: true };\n      });\n\n      server.post<{ Body: { keys: string[] } }>(\"/api/i18n/checkUsage\", async (request) => {\n        return sonamuDictionary.checkUsage(request.body.keys);\n      });\n\n      // ui-web 빌드 파일 서빙\n      const uiDistPath = path.resolve(import.meta.dirname, \"../ui-web\");\n\n      // 정적 파일 서빙: 루트 폴더 전체 (assets, setting.svg 등)\n      server.register(await import(\"@fastify/static\"), {\n        root: uiDistPath,\n        prefix: \"/\",\n        decorateReply: false,\n        wildcard: false,\n      });\n\n      // SPA fallback - 정적 파일이 없는 모든 경로는 index.html로\n      server.get(\"*\", async (_request, reply) => {\n        reply.headers({ \"Content-type\": \"text/html\" }).send(\n          fs\n            .readFileSync(path.resolve(uiDistPath, \"index.html\"))\n            .toString()\n            .replace(\"{{projectName}}\", Sonamu.config.projectName ?? \"UnknownSonamuProject\"),\n        );\n      });\n    },\n    { prefix: \"/sonamu-ui\" },\n  );\n}\n"],"names":["execSync","fs","inflection","path","range","Sonamu","SD","sonamuDictionary","EntityManager","BadRequestException","isSoException","ServiceUnavailableException","Migrator","SlackConfirm","TemplateManager","FixtureManager","BUILT_IN_TYPE_IDS","TemplateKey","nonNullable","setAiApi","sonamuUIApiPlugin","fastify","register","server","migrator","waitForHMRCompleted","fn","waitPromise","Promise","resolve","timeout","setTimeout","handler","clearTimeout","syncer","eventEmitter","off","once","result","get","config","request","entityId","preset","absPath","query","targetPath","entity","names","apiRootPath","filename","parentFs","origin","glossary","Map","getAllIds","title","set","underscore","id","pluralize","props","forEach","prop","has","name","desc","replace","suggested","words","split","combinations","length","flatMap","len","idx","w","slice","join","REPLACED_PREFIX","remainArr","comb","remainStr","includes","map","r","startsWith","toUpperCase","entityIds","flattenSubsetRows","subsetRows","subsetRow","children","sRow","entities","all","getSubsetRows","sort","a","b","aId","parentId","bId","undefined","filter","reload","autoloadTypes","typeIds","projectTypeIds","Object","entries","types","_typeId","zodType","_zod","def","type","typeId","_zodType","builtInTypeIds","allTypeIds","enumIds","keys","enumLabels","post","form","body","createEntity","delEntity","newValues","table","save","subsetKey","fields","fieldsInternal","subsets","subsetsInternal","updated","updatedInternal","at","newProp","createProp","modifyProp","delProp","to","moveProp","indexes","newEnumId","Error","endsWith","active","hidden","enumId","isExists","some","after","before","isReferenced","columns","getTableColumns","status","getStatus","action","targets","force","forceReason","requestor","runShadowTest","slackConfirm","requiresApproval","t","isTargetRequiresApproval","localHosts","isLocalTarget","every","target","targetConfig","dbConfig","host","connection","toLowerCase","isConfigured","conns","pendingMigrations","Set","conn","connKey","pending","existing","getExistingRequest","approved","rejected","checkApproval","channel","ts","runAction","logExecution","forceApproval","postApprovalRequest","saveRequest","reason","success","codeNames","delCodes","_requestt","generatePreparedCodes","templateGroupName","templateKeys","_templateKeys","options","tk","getChildrenIds","allEnumIds","templateKey","statuses","subPath","fullPath","checkExistsGenCode","template","getRequiredDictKeys","ensureDictKeys","overwrite","generateTemplate","e","statusCode","console","error","option","templateOptions","pathAndCodes","renderTemplate","sourceDB","targetDB","search","duplicateCheck","getFixtures","db","fixtures","insertFixtures","code","addFixtureLoader","getDictionary","_request","reply","buffer","exportToExcel","header","send","data","file","toBuffer","importFromExcel","updateEntry","createEntry","deleteEntry","key","checkUsage","uiDistPath","dirname","root","prefix","decorateReply","wildcard","headers","readFileSync","toString","projectName"],"mappings":"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"}
1113
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/ui/api.ts"],"sourcesContent":["import { execSync } from \"child_process\";\nimport type { FastifyInstance } from \"fastify\";\nimport fs from \"fs\";\nimport inflection from \"inflection\";\nimport type { AddressInfo } from \"net\";\nimport path from \"path\";\nimport { range } from \"radashi\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { DB, type SonamuDBConfig } from \"../database/db\";\nimport { createKnexInstance } from \"../database/knex\";\nimport { SD } from \"../dict/sd\";\nimport { sonamuDictionary } from \"../dict/sonamu-dictionary\";\nimport type { Entity } from \"../entity/entity\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport {\n  BadRequestException,\n  isSoException,\n  ServiceUnavailableException,\n} from \"../exceptions/so-exceptions\";\nimport { type MigrationResult, Migrator } from \"../migration/migrator\";\nimport { SlackConfirm, type SlackConfirmPendingResult } from \"../migration/slack-confirm\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport { DataExplorer } from \"../testing/data-explorer\";\nimport { FixtureGenerator } from \"../testing/fixture-generator\";\nimport { type DuplicateCheckOptions, FixtureManager } from \"../testing/fixture-manager\";\nimport {\n  BUILT_IN_TYPE_IDS,\n  type Cone,\n  type EntityIndex,\n  type EntityProp,\n  type EntitySubsetRow,\n  type FixtureRecord,\n  type FixtureSearchOptions,\n  type FlattenSubsetRow,\n  type PathAndCode,\n  TemplateKey,\n} from \"../types/types\";\nimport { nonNullable } from \"../utils/utils\";\nimport { setAiApi } from \"./ai-api\";\n\nexport async function sonamuUIApiPlugin(fastify: FastifyInstance) {\n  fastify.register(\n    async (server) => {\n      // migrator\n      const migrator = new Migrator();\n\n      // waitForHMRCompleted\n      async function waitForHMRCompleted<T>(fn: () => Promise<T>): Promise<T> {\n        const waitPromise = new Promise<void>((resolve) => {\n          const timeout = setTimeout(() => {\n            resolve();\n          }, 1500);\n\n          const handler = () => {\n            clearTimeout(timeout);\n            Sonamu.syncer.eventEmitter.off(\"onHMRCompleted\", handler);\n            resolve();\n          };\n\n          Sonamu.syncer.eventEmitter.once(\"onHMRCompleted\", handler);\n        });\n\n        const result = await fn();\n        await waitPromise;\n        return result;\n      }\n\n      await setAiApi(server);\n\n      server.get(\"/api/sonamu/config\", async () => {\n        return Sonamu.config;\n      });\n\n      server.get<{\n        Querystring: {\n          entityId?: string;\n          preset?: \"types\" | \"entity.json\" | \"generated\";\n          absPath?: string;\n        };\n      }>(\"/api/tools/openVscode\", async (request) => {\n        const { entityId, preset, absPath } = request.query;\n\n        const targetPath = (() => {\n          if (entityId && preset) {\n            const entity = EntityManager.get(entityId);\n            const { names } = entity;\n\n            const { apiRootPath } = Sonamu;\n            const filename = (() => {\n              switch (preset) {\n                case \"types\":\n                  return `${names.fs}.types.ts`;\n                case \"entity.json\":\n                  return `${names.fs}.entity.json`;\n                case \"generated\":\n                  return `${names.fs}.generated.ts`;\n              }\n            })();\n            return `${apiRootPath}/src/application/${entity.names.parentFs}/${filename}`;\n          } else {\n            if (!absPath) {\n              throw new BadRequestException(SD(\"sonamu.error.presetOrAbsPathRequired\"));\n            }\n            return absPath;\n          }\n        })();\n        execSync(`code ${targetPath}`);\n      });\n\n      server.get<{\n        Querystring: {\n          origin: string;\n          entityId?: string;\n        };\n      }>(\"/api/tools/getSuggestion\", async (request) => {\n        const { origin, entityId } = request.query;\n\n        // 치환 용어집\n        const glossary = new Map<string, string>([\n          [\"status\", \"상태\"],\n          [\"type\", \"타입\"],\n          [\"image\", \"이미지\"],\n          [\"images\", \"이미지리스트\"],\n          [\"url\", \"URL\"],\n          [\"id\", \"ID\"],\n          [\"name\", `{EntityID}명`],\n          [\"title\", \"{EntityID}명\"],\n          [\"parent\", \"상위{EntityID}\"],\n          [\"desc\", \"설명\"],\n          [\"at\", \"일시\"],\n          [\"created\", \"등록\"],\n          [\"updated\", \"수정\"],\n          [\"deleted\", \"삭제\"],\n          [\"by\", \"유저\"],\n          [\"date\", \"일자\"],\n          [\"time\", \"시간\"],\n          [\"ko\", \"(한글)\"],\n          [\"en\", \"(영문)\"],\n          [\"krw\", \"(원)\"],\n          [\"usd\", \"(USD)\"],\n          [\"color\", \"컬러\"],\n          [\"code\", \"코드\"],\n          [\"x\", \"X좌표\"],\n          [\"y\", \"Y좌표\"],\n          [\"current\", \"현재\"],\n          [\"stock\", \"재고\"],\n          [\"total\", \"총\"],\n          [\"admin\", \"관리자\"],\n          [\"group\", \"그룹\"],\n          [\"item\", \"아이템\"],\n          [\"cnt\", \"수량\"],\n          [\"price\", \"가격\"],\n          [\"preset\", \"프리셋\"],\n          [\"acct\", \"계좌\"],\n          [\"tel\", \"전화번호\"],\n          [\"no\", \"번호\"],\n          [\"body\", \"내용\"],\n          [\"content\", \"내용\"],\n          [\"orderno\", \"정렬순서\"],\n          [\"priority\", \"우선순위\"],\n          [\"text\", \"텍스트\"],\n          [\"key\", \"키\"],\n          [\"sum\", \"합산\"],\n          [\"expected\", \"예상\"],\n          [\"actual\", \"실제\"],\n        ]);\n        // 전체 엔티티 순회하며, 엔티티 타이틀과 프롭 설명을 치환 용어집에 추가\n        for (const entityId of EntityManager.getAllIds()) {\n          const entity = EntityManager.get(entityId);\n          if ((entity.title ?? \"\") !== \"\") {\n            glossary.set(inflection.underscore(entity.id), entity.title);\n            glossary.set(\n              inflection.underscore(inflection.pluralize(entity.id)),\n              `${entity.title}리스트`,\n            );\n          }\n\n          entity.props.forEach((prop) => {\n            if (glossary.has(prop.name)) {\n              return;\n            }\n            if (prop.desc) {\n              glossary.set(prop.name, prop.desc.replace(entity.title ?? \"\", \"{EntityID}\"));\n            }\n          });\n        }\n\n        const suggested = (() => {\n          // 단어 분리, 가능한 조합 생성\n          const words = origin.split(\"_\");\n          const combinations = [...range(words.length, 0, -1)].flatMap((len) => {\n            return [\n              ...range(0, words.length - len + 1, (idx) => {\n                return {\n                  len,\n                  w: words.slice(idx, idx + len).join(\"_\"),\n                };\n              }),\n            ];\n          });\n\n          // 조합을 순회하며, 치환 용어집에 있는 단어가 포함된 경우, 치환 용어로 치환\n          const REPLACED_PREFIX = \"#REPLACED//\"; // 치환된 단어를 join 이후에도 식별하기 위해 prefix 추가\n          let remainArr: string[] = [...words];\n          for (const comb of combinations) {\n            const remainStr = remainArr.join(\"_\");\n            if (remainStr.includes(comb.w) && glossary.has(comb.w)) {\n              remainArr = remainStr\n                .replace(comb.w, REPLACED_PREFIX + glossary.get(comb.w))\n                .split(\"_\");\n            }\n          }\n\n          return remainArr\n            .map((r) => {\n              if (r.startsWith(REPLACED_PREFIX)) {\n                return r.replace(REPLACED_PREFIX, \"\");\n              } else {\n                return r.toUpperCase();\n              }\n            })\n            .join(\"\")\n            .replace(/{EntityID}/g, entityId ? EntityManager.get(entityId).title : \"\");\n        })();\n\n        return { suggested };\n      });\n\n      server.get(\"/api/entity/findMany\", async () => {\n        const entityIds = EntityManager.getAllIds();\n\n        function flattenSubsetRows(subsetRows: EntitySubsetRow[]): FlattenSubsetRow[] {\n          return subsetRows.flatMap((subsetRow) => {\n            const { children, ...sRow } = subsetRow;\n            return [sRow, ...flattenSubsetRows(children)];\n          });\n        }\n\n        const entities = await Promise.all(\n          entityIds.map((entityId) => {\n            const entity = EntityManager.get(entityId);\n            const subsetRows = entity.getSubsetRows();\n\n            return {\n              ...entity,\n              flattenSubsetRows: flattenSubsetRows(subsetRows),\n            };\n          }),\n        );\n\n        entities.sort((a, b) => {\n          const aId = a.parentId ?? a.id;\n          const bId = b.parentId ?? b.id;\n          if (aId < bId) return -1;\n          if (aId > bId) return 1;\n          if (aId === bId) {\n            if (a.parentId === undefined) return -1;\n            if (b.parentId === undefined) return 1;\n            return 0;\n          }\n          return 0;\n        });\n        return { entities };\n      });\n\n      server.get<{\n        Querystring: {\n          filter?: \"enums\" | \"types\";\n          reload?: \"1\";\n        };\n      }>(\"/api/entity/typeIds\", async (request): Promise<{ typeIds: string[] }> => {\n        const { filter, reload } = request.query;\n\n        if (reload === \"1\") {\n          await Sonamu.syncer.autoloadTypes();\n        }\n\n        const typeIds = (() => {\n          // 프로젝트에서 정의한 타입들\n          const projectTypeIds = Object.entries(Sonamu.syncer.types)\n            .filter(([_typeId, zodType]) => (zodType._zod.def.type as string) !== \"enum\")\n            .map(([typeId, _zodType]) => typeId);\n\n          // 내장 타입들 (sonamu 코어에서 제공)\n          const builtInTypeIds = [...BUILT_IN_TYPE_IDS];\n\n          // 모든 타입 병합\n          const allTypeIds = [...builtInTypeIds, ...projectTypeIds];\n\n          if (filter === \"types\") {\n            return allTypeIds;\n          }\n\n          const enumIds = EntityManager.getAllIds().flatMap((entityId) => {\n            const entity = EntityManager.get(entityId);\n            return Object.keys(entity.enumLabels);\n          });\n\n          if (filter === \"enums\") {\n            return enumIds;\n          } else {\n            return [...allTypeIds, ...enumIds];\n          }\n        })();\n\n        return {\n          typeIds,\n        };\n      });\n\n      server.post<{\n        Body: {\n          form: {\n            id: string;\n            title: string;\n            table: string;\n            parentId?: string;\n          };\n        };\n      }>(\"/api/entity/create\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { form } = request.body;\n          await Sonamu.syncer.createEntity({ ...form, entityId: form.id });\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n        };\n      }>(\"/api/entity/del\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId } = request.body;\n          return await Sonamu.syncer.delEntity(entityId);\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newValues: {\n            title: string;\n            table: string;\n            parentId?: string;\n          };\n        };\n      }>(\"/api/entity/modifyEntityBase\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, newValues } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.title = newValues.title;\n          entity.table = newValues.table;\n          entity.parentId = newValues.parentId;\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          subsetKey: string;\n          fields: string[];\n          fieldsInternal?: string[];\n        };\n      }>(\"/api/entity/modifySubset\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, subsetKey, fields, fieldsInternal } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.subsets[subsetKey] = fields;\n          if (fieldsInternal !== undefined) {\n            if (fieldsInternal.length > 0) {\n              entity.subsetsInternal[subsetKey] = fieldsInternal;\n            } else {\n              delete entity.subsetsInternal[subsetKey];\n            }\n          }\n          await entity.save();\n\n          return { updated: fields, updatedInternal: fieldsInternal };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          subsetKey: string;\n        };\n      }>(\"/api/entity/delSubset\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, subsetKey } = request.body;\n          const entity = EntityManager.get(entityId);\n          delete entity.subsets[subsetKey];\n          delete entity.subsetsInternal[subsetKey];\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newProp: EntityProp;\n          at?: number;\n        };\n      }>(\"/api/entity/createProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, newProp } = request.body;\n          const entity = EntityManager.get(entityId);\n          await entity.createProp(newProp, at);\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newProp: EntityProp;\n          at: number;\n        };\n      }>(\"/api/entity/modifyProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, newProp } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.modifyProp(newProp, at);\n\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          at: number;\n        };\n      }>(\"/api/entity/delProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.delProp(at);\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          at: number;\n          to: number;\n        };\n      }>(\"/api/entity/moveProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, to } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.moveProp(at, to);\n\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          indexes: EntityIndex[];\n        };\n      }>(\"/api/entity/modifyIndexes\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, indexes } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.indexes = indexes;\n          await entity.save();\n\n          return { updated: indexes };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumLabels: Entity[\"enumLabels\"];\n        };\n      }>(\"/api/entity/modifyEnumLabels\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumLabels } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.enumLabels = enumLabels;\n          await entity.save();\n\n          return { updated: enumLabels };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newEnumId: string;\n        };\n      }>(\"/api/entity/createEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, newEnumId } = request.body;\n          const entity = EntityManager.get(entityId);\n\n          if (entity.enumLabels[newEnumId]) {\n            throw new Error(`이미 존재하는 enumId입니다: ${newEnumId}`);\n          }\n\n          entity.enumLabels[newEnumId] = {\n            ...(newEnumId.endsWith(\"Status\")\n              ? {\n                  active: \"노출\",\n                  hidden: \"숨김\",\n                }\n              : {\n                  \"\": \"\",\n                }),\n          };\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumId: {\n            before: string;\n            after: string;\n          };\n        };\n      }>(\"/api/entity/modifyEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumId } = request.body;\n          const entityIds = EntityManager.getAllIds();\n          const isExists = entityIds.some((entityId) => {\n            const entity = EntityManager.get(entityId);\n            return Object.keys(entity.enumLabels).includes(enumId.after);\n          });\n          if (isExists) {\n            throw new Error(`이미 존재하는 EnumId입니다: ${enumId.after}`);\n          }\n\n          const entity = EntityManager.get(entityId);\n          entity.enumLabels[enumId.after] = entity.enumLabels[enumId.before];\n          delete entity.enumLabels[enumId.before];\n\n          await entity.save();\n\n          for (const entityId of entityIds) {\n            const entity = EntityManager.get(entityId);\n            for (const prop of entity.props) {\n              if (prop.type === \"enum\" && prop.id === enumId.before) {\n                prop.id = enumId.after;\n              }\n            }\n            await entity.save();\n          }\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumId: string;\n        };\n      }>(\"/api/entity/deleteEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumId } = request.body;\n\n          const entityIds = EntityManager.getAllIds();\n          const isReferenced = entityIds\n            .flatMap((entityId) => EntityManager.get(entityId).props)\n            .some((prop) => prop.type === \"enum\" && prop.id === enumId);\n          if (isReferenced) {\n            throw new Error(`${enumId}를 참조하는 프로퍼티가 존재합니다.`);\n          }\n\n          const entity = EntityManager.get(entityId);\n          delete entity.enumLabels[enumId];\n          await entity.save();\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          target: \"entity\" | \"prop\" | \"enum\" | \"subset\";\n          propName?: string;\n          enumId?: string;\n          subsetKey?: string;\n          cone: Cone;\n        };\n      }>(\"/api/entity/updateCone\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, target, propName, enumId, subsetKey, cone } = request.body;\n          const entity = EntityManager.get(entityId);\n\n          if (target === \"entity\") {\n            entity.cone = cone;\n          } else if (target === \"prop\" && propName) {\n            const prop = entity.props.find((p) => p.name === propName);\n            if (prop) {\n              (prop as { cone?: Cone }).cone = cone;\n            }\n          } else if (target === \"enum\" && enumId) {\n            entity.enumCones[enumId] = cone;\n          } else if (target === \"subset\" && subsetKey) {\n            entity.subsetCones[subsetKey] = cone;\n          }\n\n          await entity.save();\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          preserveExisting?: boolean;\n          onlyEmpty?: boolean;\n          locale?: \"ko\" | \"en\" | \"ja\";\n        };\n      }>(\"/api/entity/generateCones\", async (request, reply) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, preserveExisting, onlyEmpty, locale } = request.body;\n\n          try {\n            // Entity 존재 여부 확인\n            const entity = EntityManager.get(entityId);\n\n            // locale 기본값: Sonamu.config.i18n.defaultLocale 사용\n            const effectiveLocale =\n              locale ?? (Sonamu.config.i18n.defaultLocale as \"ko\" | \"en\" | \"ja\");\n\n            // Cone 생성\n            const result = await entity.generateCones({\n              preserveExisting: preserveExisting ?? true,\n              onlyEmpty: onlyEmpty ?? false,\n              locale: effectiveLocale,\n            });\n\n            return result;\n          } catch (error: unknown) {\n            const message = error instanceof Error ? error.message : String(error);\n\n            // Entity not found\n            if (message.includes(\"존재하지 않는 Entity\")) {\n              reply.status(404);\n              return {\n                success: false,\n                error: `Entity not found: ${entityId}`,\n              };\n            }\n\n            // API 키 없음\n            if (message.includes(\"ANTHROPIC_API_KEY not found\")) {\n              reply.status(500);\n              return {\n                success: false,\n                error: \"API key not configured\",\n              };\n            }\n\n            // Rate limit\n            if (message.includes(\"Rate limit exceeded\")) {\n              reply.status(429);\n              return {\n                success: false,\n                error: \"Rate limit exceeded. Please try again later.\",\n              };\n            }\n\n            // 기타 에러\n            reply.status(500);\n            return {\n              success: false,\n              error: `Cone generation failed: ${message}`,\n            };\n          }\n        });\n      });\n\n      server.get<{\n        Querystring: {\n          entityId: string;\n        };\n      }>(\"/api/entity/getTableColumns\", async (request) => {\n        const { entityId } = request.query;\n        const entity = EntityManager.get(entityId);\n        const columns = entity.getTableColumns();\n        return { columns };\n      });\n\n      server.get(\"/api/migrations/status\", async () => {\n        const status = await migrator.getStatus();\n\n        return { status };\n      });\n\n      server.post<{\n        Body: {\n          action: \"apply\" | \"rollback\" | \"shadow\";\n          targets: (keyof SonamuDBConfig)[];\n          force?: boolean;\n          forceReason?: string;\n          requestor?: string;\n        };\n      }>(\n        \"/api/migrations/runAction\",\n        async (request): Promise<MigrationResult | SlackConfirmPendingResult> => {\n          const { action, targets, force, forceReason, requestor } = request.body;\n\n          if (action === \"shadow\") {\n            return migrator.runShadowTest();\n          }\n\n          // Slack 승인 체크 (apply 시에만)\n          if (action === \"apply\") {\n            const slackConfirm = new SlackConfirm();\n            const requiresApproval = targets.some((t) => slackConfirm.isTargetRequiresApproval(t));\n\n            // 로컬 DB인 경우 승인 스킵\n            const localHosts = [\"localhost\", \"127.0.0.1\", \"0.0.0.0\", \"::1\"];\n            const isLocalTarget = targets.every((target) => {\n              const targetConfig = Sonamu.dbConfig[target];\n              const host = (targetConfig?.connection as { host?: string })?.host ?? \"localhost\";\n              return localHosts.includes(host.toLowerCase());\n            });\n\n            if (requiresApproval && slackConfirm.isConfigured() && !isLocalTarget) {\n              const { conns } = await migrator.getStatus();\n\n              // 모든 타겟 DB에서 pending인 마이그레이션의 합집합을 구합니다.\n              const pendingMigrations = [\n                ...new Set(\n                  conns\n                    .filter((conn) => targets.includes(conn.connKey as keyof SonamuDBConfig))\n                    .flatMap((conn) => conn.pending),\n                ),\n              ];\n\n              if (pendingMigrations.length > 0) {\n                // 기존 승인 요청 확인\n                const existing = await slackConfirm.getExistingRequest(pendingMigrations);\n\n                if (existing) {\n                  // 기존 요청이 있으면 승인 상태 확인\n                  const { approved, rejected } = await slackConfirm.checkApproval(\n                    existing.channel,\n                    existing.ts,\n                  );\n\n                  if (approved) {\n                    // 승인됨 → 실행\n                    const result = await migrator.runAction(action, targets);\n                    if (result.length > 0) {\n                      await slackConfirm.logExecution(\n                        existing.channel,\n                        existing.ts,\n                        result,\n                        requestor,\n                      );\n                    }\n                    return result;\n                  } else if (rejected) {\n                    throw new BadRequestException(SD(\"sonamu.error.migrationRejected\"));\n                  } else if (force) {\n                    // Force 진행\n                    await slackConfirm.forceApproval(\n                      existing.channel,\n                      existing.ts,\n                      forceReason ?? \"사유 없음\",\n                      requestor,\n                    );\n                    const result = await migrator.runAction(action, targets);\n                    if (result.length > 0) {\n                      await slackConfirm.logExecution(\n                        existing.channel,\n                        existing.ts,\n                        result,\n                        requestor,\n                      );\n                    }\n                    return result;\n                  } else {\n                    // 대기중\n                    return {\n                      type: \"pending\",\n                      channel: existing.channel,\n                      ts: existing.ts,\n                    };\n                  }\n                } else {\n                  // 새 승인 요청 발송\n                  const { channel, ts } = await slackConfirm.postApprovalRequest(\n                    pendingMigrations,\n                    targets,\n                    requestor,\n                  );\n                  await slackConfirm.saveRequest(pendingMigrations, channel, ts);\n\n                  return {\n                    type: \"pending\",\n                    channel,\n                    ts,\n                  };\n                }\n              }\n            }\n          }\n\n          return migrator.runAction(action, targets);\n        },\n      );\n\n      server.post<{\n        Body: {\n          channel: string;\n          ts: string;\n        };\n      }>(\"/api/migrations/checkApproval\", async (request) => {\n        const { channel, ts } = request.body;\n        const slackConfirm = new SlackConfirm();\n\n        if (!slackConfirm.isConfigured()) {\n          return { approved: true, rejected: false };\n        }\n\n        return slackConfirm.checkApproval(channel, ts);\n      });\n\n      server.post<{\n        Body: {\n          channel: string;\n          ts: string;\n          reason: string;\n          requestor?: string;\n        };\n      }>(\"/api/migrations/forceApproval\", async (request) => {\n        const { channel, ts, reason, requestor } = request.body;\n        const slackConfirm = new SlackConfirm();\n\n        if (!slackConfirm.isConfigured()) {\n          throw new BadRequestException(SD(\"sonamu.error.slackConfirmNotConfigured\"));\n        }\n\n        await slackConfirm.forceApproval(channel, ts, reason, requestor);\n        return { success: true };\n      });\n\n      server.post<{\n        Body: {\n          codeNames: string[];\n        };\n      }>(\"/api/migrations/delCodes\", async (request) => {\n        const { codeNames } = request.body;\n        return await migrator.delCodes(codeNames);\n      });\n\n      server.post(\"/api/migrations/generatePreparedCodes\", async (_requestt) => {\n        return await migrator.generatePreparedCodes();\n      });\n\n      server.post<{\n        Body: {\n          entityIds: string[];\n          templateKeys: string[];\n        };\n      }>(\"/api/scaffolding/getStatus\", async (request) => {\n        const { entityIds, templateKeys: _templateKeys } = request.body;\n        if ((entityIds ?? []).length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.entityIdsRequired\"));\n        } else if ((_templateKeys ?? []).length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.templateKeysRequired\"));\n        }\n\n        // sorting\n        entityIds.sort((a, b) => (a < b ? -1 : a > b ? 1 : 0));\n        const templateKeys = TemplateKey.options.filter((tk) => _templateKeys.includes(tk));\n\n        const combinations = entityIds.flatMap((entityId) => {\n          return templateKeys.map((templateKey) => [entityId, templateKey]);\n        });\n\n        const statuses = await Promise.all(\n          combinations.map(async ([entityId, templateKey]) => {\n            const { subPath, fullPath, isExists } = await Sonamu.syncer.checkExistsGenCode(\n              entityId,\n              templateKey as TemplateKey,\n            );\n            return {\n              entityId,\n              templateKey,\n              subPath,\n              fullPath,\n              isExists,\n            };\n          }),\n        );\n        return { statuses };\n      });\n\n      server.post<{\n        Body: {\n          options: {\n            entityId: string;\n            templateKey: string;\n            enumId?: string;\n            overwrite?: boolean;\n          }[];\n        };\n      }>(\"/api/scaffolding/generate\", async (request) => {\n        const { options } = request.body;\n        if (options.length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.optionsRequired\"));\n        }\n\n        // 1. 모든 템플릿에서 필요한 dict 키를 수집\n        const keys = options.flatMap(({ templateKey }) => {\n          const template = TemplateManager.get(templateKey);\n          return template.getRequiredDictKeys() ?? [];\n        });\n\n        // 2. target별로 ensureDictKeys 호출 (순차 처리)\n        await sonamuDictionary.ensureDictKeys([...new Set(keys)]);\n\n        // 3. 템플릿 생성 (병렬 처리)\n        const result = await Promise.all(\n          options.map(async ({ entityId, templateKey, enumId, overwrite }) => {\n            try {\n              return await Sonamu.syncer.generateTemplate(\n                templateKey as TemplateKey,\n                {\n                  entityId,\n                  enumId,\n                } as {\n                  entityId: string;\n                  enumId?: string;\n                },\n                {\n                  overwrite,\n                },\n              );\n            } catch (e) {\n              if (isSoException(e) && e.statusCode === 541) {\n                return null;\n              } else {\n                console.error(e);\n                throw e;\n              }\n            }\n          }),\n        );\n\n        if (result.filter(nonNullable).length === 0) {\n          throw new ServiceUnavailableException(SD(\"sonamu.error.allFilesGenerated\"));\n        }\n        return result;\n      });\n\n      server.post<{\n        Body: {\n          option: {\n            entityId: string;\n            templateKey: string;\n            enumId?: string;\n          };\n        };\n      }>(\"/api/scaffolding/preview\", async (request): Promise<{ pathAndCodes: PathAndCode[] }> => {\n        const { option } = request.body;\n\n        try {\n          const { templateKey, ...templateOptions } = option;\n          const pathAndCodes = await Sonamu.syncer.renderTemplate(\n            templateKey as TemplateKey,\n            templateOptions,\n          );\n\n          return { pathAndCodes };\n        } catch (e) {\n          console.error(e);\n          throw e;\n        }\n      });\n\n      server.post(\"/api/fixture\", async (request) => {\n        const { sourceDB, targetDB, search, duplicateCheck } = request.body as {\n          sourceDB: keyof SonamuDBConfig;\n          targetDB: keyof SonamuDBConfig;\n          search: FixtureSearchOptions;\n          duplicateCheck?: DuplicateCheckOptions;\n        };\n\n        return FixtureManager.getFixtures(sourceDB, targetDB, search, duplicateCheck);\n      });\n\n      server.post(\"/api/fixture/import\", async (request) => {\n        const { db, fixtures } = request.body as {\n          db: keyof SonamuDBConfig;\n          fixtures: FixtureRecord[];\n        };\n\n        return FixtureManager.insertFixtures(db, fixtures);\n      });\n\n      server.post(\"/api/fixture/addFixtureLoader\", async (request) => {\n        const { code } = request.body as { code: string };\n\n        return FixtureManager.addFixtureLoader(code);\n      });\n\n      server.get(\"/api/i18n/dictionary\", async () => {\n        return sonamuDictionary.getDictionary();\n      });\n\n      server.get(\"/api/i18n/export\", async (_request, reply) => {\n        const { filename, buffer } = await sonamuDictionary.exportToExcel();\n        reply\n          .header(\n            \"Content-Type\",\n            \"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet\",\n          )\n          .header(\"Content-Disposition\", `attachment; filename=\"${filename}\"`)\n          .send(buffer);\n      });\n\n      server.post(\"/api/i18n/import\", async (request) => {\n        const data = await request.file();\n        if (!data) {\n          throw new BadRequestException(SD(\"sonamu.error.fileNotUploaded\"));\n        }\n        const buffer = await data.toBuffer();\n        return sonamuDictionary.importFromExcel(buffer);\n      });\n\n      server.post<{\n        Body: {\n          oldKey: string;\n          newKey: string;\n          source: \"entity\" | \"project\" | \"sonamu\";\n          values: Record<string, string>;\n        };\n      }>(\"/api/i18n/update\", async (request) => {\n        await sonamuDictionary.updateEntry(request.body);\n        return { success: true };\n      });\n\n      server.post<{\n        Body: {\n          key: string;\n          values: Record<string, string>;\n        };\n      }>(\"/api/i18n/create\", async (request) => {\n        await sonamuDictionary.createEntry(request.body);\n        return { success: true };\n      });\n\n      server.post<{\n        Body: {\n          key: string;\n        };\n      }>(\"/api/i18n/delete\", async (request) => {\n        await sonamuDictionary.deleteEntry(request.body.key);\n        return { success: true };\n      });\n\n      server.post<{ Body: { keys: string[] } }>(\"/api/i18n/checkUsage\", async (request) => {\n        return sonamuDictionary.checkUsage(request.body.keys);\n      });\n\n      // Tasks API\n      server.get(\"/api/tasks/status\", async () => {\n        try {\n          Sonamu.workflows;\n          return { active: true };\n        } catch {\n          return { active: false };\n        }\n      });\n\n      server.get(\"/api/tasks/workflowDefinitions\", async () => {\n        const definitions = Sonamu.workflows.workflowDefinitions;\n        return { definitions };\n      });\n\n      server.get<{\n        Querystring: {\n          limit?: string;\n          after?: string;\n          before?: string;\n          order?: \"asc\" | \"desc\";\n          status?: string;\n          workflowName?: string;\n          createdAfter?: string;\n          createdBefore?: string;\n        };\n      }>(\"/api/tasks/workflowRuns\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        const { limit, after, before, order, status, workflowName, createdAfter, createdBefore } =\n          request.query;\n        return backend.listWorkflowRuns({\n          limit: limit ? Number.parseInt(limit, 10) : undefined,\n          after,\n          before,\n          order,\n          status: status ? status.split(\",\") : undefined,\n          workflowName: workflowName || undefined,\n          createdAfter: createdAfter ? new Date(createdAfter) : undefined,\n          createdBefore: createdBefore ? new Date(createdBefore) : undefined,\n        });\n      });\n\n      server.get<{\n        Params: { id: string };\n      }>(\"/api/tasks/workflowRuns/:id\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        const workflowRun = await backend.getWorkflowRun({\n          workflowRunId: request.params.id,\n        });\n        if (!workflowRun) {\n          throw new Error(`Workflow run not found: ${request.params.id}`);\n        }\n        return workflowRun;\n      });\n\n      server.post<{\n        Params: { id: string };\n      }>(\"/api/tasks/workflowRuns/:id/cancel\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        return backend.cancelWorkflowRun({\n          workflowRunId: request.params.id,\n        });\n      });\n\n      server.post<{\n        Params: { id: string };\n      }>(\"/api/tasks/workflowRuns/:id/pause\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        return backend.pauseWorkflowRun({\n          workflowRunId: request.params.id,\n        });\n      });\n\n      server.post<{\n        Params: { id: string };\n      }>(\"/api/tasks/workflowRuns/:id/resume\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        return backend.resumeWorkflowRun({\n          workflowRunId: request.params.id,\n        });\n      });\n\n      server.get<{\n        Params: { id: string };\n        Querystring: {\n          limit?: string;\n          after?: string;\n          before?: string;\n        };\n      }>(\"/api/tasks/workflowRuns/:id/steps\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        const { limit, after, before } = request.query;\n        return backend.listStepAttempts({\n          workflowRunId: request.params.id,\n          limit: limit ? Number.parseInt(limit, 10) : undefined,\n          after,\n          before,\n        });\n      });\n\n      /**\n       * Health Check API\n       * MCP 도구가 Sonamu 서버를 자동 감지하기 위한 엔드포인트\n       */\n      server.get(\"/api/sonamu/health\", async (request) => {\n        const address = request.server.server.address();\n        const port = address && typeof address === \"object\" ? (address as AddressInfo).port : 0;\n\n        return {\n          ok: true,\n          project: process.cwd().split(\"/\").pop() || \"unknown\",\n          port,\n          timestamp: new Date().toISOString(),\n        };\n      });\n\n      /**\n       * Fixture 생성 API\n       */\n      server.post<{\n        Body: {\n          entity: string;\n          count?: number;\n          overrides?: Record<string, unknown>;\n          targetDb?: \"fixture\" | \"test\";\n        };\n      }>(\"/api/sonamu/fixture/generate\", async (request, reply) => {\n        const { entity, count = 1, overrides, targetDb = \"fixture\" } = request.body;\n\n        try {\n          // 타겟 DB 설정 가져오기\n          const dbConfig = targetDb === \"fixture\" ? Sonamu.dbConfig.fixture : Sonamu.dbConfig.test;\n\n          // Knex 인스턴스 생성\n          const db = createKnexInstance(dbConfig);\n\n          // FixtureGenerator 생성\n          const generator = new FixtureGenerator(db, db, targetDb, EntityManager);\n\n          // 단일 Entity 배치 생성\n          const fixtures = await generator.generateBatch([\n            {\n              entity,\n              count,\n              overrides: overrides ?? {},\n            },\n          ]);\n\n          // Knex 연결 종료\n          await db.destroy();\n\n          return {\n            success: true,\n            entity,\n            count: fixtures.length,\n            fixtures,\n            targetDb,\n          };\n        } catch (error) {\n          reply.status(400);\n          return {\n            success: false,\n            error: error instanceof Error ? error.message : String(error),\n          };\n        }\n      });\n\n      /**\n       * Fixture 데이터 탐색 API\n       */\n      server.post<{\n        Body: {\n          entity: string;\n          strategy: \"sample\" | \"recent\" | \"random\" | \"query\";\n          limit?: number;\n          where?: Record<string, unknown>;\n        };\n      }>(\"/api/sonamu/fixture/explore\", async (request, reply) => {\n        const { entity, strategy, limit = 10, where } = request.body;\n\n        try {\n          // Fixture DB 설정 가져오기\n          const fixtureDbConfig = Sonamu.dbConfig.fixture;\n\n          // Knex 인스턴스 생성\n          const fixtureDb = createKnexInstance(fixtureDbConfig);\n\n          // DataExplorer 생성\n          const explorer = new DataExplorer(fixtureDb, EntityManager);\n\n          const data = await explorer.explore(entity, {\n            strategy,\n            limit,\n            where,\n          });\n\n          // Knex 연결 종료\n          await fixtureDb.destroy();\n\n          return {\n            success: true,\n            entity,\n            strategy,\n            count: data.length,\n            data,\n          };\n        } catch (error) {\n          reply.status(400);\n          return {\n            success: false,\n            error: error instanceof Error ? error.message : String(error),\n          };\n        }\n      });\n\n      /**\n       * Fixture 데이터 가져오기 (fetch) API\n       * production/development DB에서 실제 데이터를 fixture DB로 import\n       */\n      server.post<{\n        Body: {\n          entity: string;\n          strategy?: \"sample\" | \"recent\" | \"random\" | \"query\";\n          limit?: number;\n          includeRelations?: boolean;\n          maxDepth?: number;\n        };\n      }>(\"/api/sonamu/fixture/fetch\", async (request, reply) => {\n        const {\n          entity,\n          strategy = \"recent\",\n          limit = 10,\n          includeRelations = true,\n          maxDepth = 2,\n        } = request.body;\n\n        try {\n          // Source DB (production/development) - 읽기 전용\n          const sourceDb = DB.getDB(\"r\");\n\n          // Target DB (fixture)\n          const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n\n          // FixtureGenerator 생성\n          const generator = new FixtureGenerator(sourceDb, fixtureDb, \"fixture\", EntityManager);\n\n          // production 데이터를 fixture DB로 import\n          const results = await generator.importFromSource(entity, {\n            strategy,\n            limit,\n            includeRelations,\n            maxDepth,\n          });\n\n          // Knex 연결 종료 (sourceDb는 Sonamu가 관리하므로 destroy하지 않음)\n          await fixtureDb.destroy();\n\n          return {\n            success: true,\n            entity,\n            strategy,\n            count: results.length,\n            imported: results,\n          };\n        } catch (error) {\n          reply.status(400);\n          return {\n            success: false,\n            error: error instanceof Error ? error.message : String(error),\n          };\n        }\n      });\n\n      /**\n       * Fixture 데이터 삭제 (clean) API\n       * FK 순서를 고려하여 안전하게 삭제\n       */\n      server.post<{\n        Body: {\n          entities?: string[];\n        };\n      }>(\"/api/sonamu/fixture/clean\", async (request, reply) => {\n        const { entities } = request.body;\n\n        try {\n          // Fixture DB 연결\n          const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n\n          // 삭제할 Entity 목록 결정\n          const targetEntities =\n            entities && entities.length > 0 ? entities : EntityManager.getAllIds();\n\n          // Entity ID를 테이블명으로 변환 (snake_case 복수형)\n          const tableNames = targetEntities.map((entityId) => {\n            const entity = EntityManager.get(entityId);\n            return entity.table;\n          });\n\n          // PostgreSQL: TRUNCATE CASCADE로 FK 순서 무관하게 안전하게 삭제\n          // CASCADE 옵션으로 의존성 있는 데이터도 함께 삭제\n          await fixtureDb.raw(\n            `TRUNCATE TABLE ${tableNames.map((t) => `\"${t}\"`).join(\", \")} RESTART IDENTITY CASCADE`,\n          );\n\n          // Knex 연결 종료\n          await fixtureDb.destroy();\n\n          return {\n            success: true,\n            cleaned: tableNames,\n            count: tableNames.length,\n          };\n        } catch (error) {\n          reply.status(400);\n          return {\n            success: false,\n            error: error instanceof Error ? error.message : String(error),\n          };\n        }\n      });\n\n      // ui-web 빌드 파일 서빙\n      const uiDistPath = path.resolve(import.meta.dirname, \"../ui-web\");\n\n      // 정적 파일 서빙: 루트 폴더 전체 (assets, setting.svg 등)\n      server.register(await import(\"@fastify/static\"), {\n        root: uiDistPath,\n        prefix: \"/\",\n        decorateReply: false,\n        wildcard: false,\n      });\n\n      // SPA fallback - 정적 파일이 없는 모든 경로는 index.html로\n      server.get(\"*\", async (_request, reply) => {\n        reply.headers({ \"Content-type\": \"text/html\" }).send(\n          fs\n            .readFileSync(path.resolve(uiDistPath, \"index.html\"))\n            .toString()\n            .replace(\"{{projectName}}\", Sonamu.config.projectName ?? \"UnknownSonamuProject\"),\n        );\n      });\n    },\n    { prefix: \"/sonamu-ui\" },\n  );\n}\n"],"names":["execSync","fs","inflection","path","range","Sonamu","DB","createKnexInstance","SD","sonamuDictionary","EntityManager","BadRequestException","isSoException","ServiceUnavailableException","Migrator","SlackConfirm","TemplateManager","DataExplorer","FixtureGenerator","FixtureManager","BUILT_IN_TYPE_IDS","TemplateKey","nonNullable","setAiApi","sonamuUIApiPlugin","fastify","register","server","migrator","waitForHMRCompleted","fn","waitPromise","Promise","resolve","timeout","setTimeout","handler","clearTimeout","syncer","eventEmitter","off","once","result","get","config","request","entityId","preset","absPath","query","targetPath","entity","names","apiRootPath","filename","parentFs","origin","glossary","Map","getAllIds","title","set","underscore","id","pluralize","props","forEach","prop","has","name","desc","replace","suggested","words","split","combinations","length","flatMap","len","idx","w","slice","join","REPLACED_PREFIX","remainArr","comb","remainStr","includes","map","r","startsWith","toUpperCase","entityIds","flattenSubsetRows","subsetRows","subsetRow","children","sRow","entities","all","getSubsetRows","sort","a","b","aId","parentId","bId","undefined","filter","reload","autoloadTypes","typeIds","projectTypeIds","Object","entries","types","_typeId","zodType","_zod","def","type","typeId","_zodType","builtInTypeIds","allTypeIds","enumIds","keys","enumLabels","post","form","body","createEntity","delEntity","newValues","table","save","subsetKey","fields","fieldsInternal","subsets","subsetsInternal","updated","updatedInternal","at","newProp","createProp","modifyProp","delProp","to","moveProp","indexes","newEnumId","Error","endsWith","active","hidden","enumId","isExists","some","after","before","isReferenced","target","propName","cone","find","p","enumCones","subsetCones","reply","preserveExisting","onlyEmpty","locale","effectiveLocale","i18n","defaultLocale","generateCones","error","message","String","status","success","columns","getTableColumns","getStatus","action","targets","force","forceReason","requestor","runShadowTest","slackConfirm","requiresApproval","t","isTargetRequiresApproval","localHosts","isLocalTarget","every","targetConfig","dbConfig","host","connection","toLowerCase","isConfigured","conns","pendingMigrations","Set","conn","connKey","pending","existing","getExistingRequest","approved","rejected","checkApproval","channel","ts","runAction","logExecution","forceApproval","postApprovalRequest","saveRequest","reason","codeNames","delCodes","_requestt","generatePreparedCodes","templateKeys","_templateKeys","options","tk","templateKey","statuses","subPath","fullPath","checkExistsGenCode","template","getRequiredDictKeys","ensureDictKeys","overwrite","generateTemplate","e","statusCode","console","option","templateOptions","pathAndCodes","renderTemplate","sourceDB","targetDB","search","duplicateCheck","getFixtures","db","fixtures","insertFixtures","code","addFixtureLoader","getDictionary","_request","buffer","exportToExcel","header","send","data","file","toBuffer","importFromExcel","updateEntry","createEntry","deleteEntry","key","checkUsage","workflows","definitions","workflowDefinitions","backend","limit","order","workflowName","createdAfter","createdBefore","listWorkflowRuns","Number","parseInt","Date","workflowRun","getWorkflowRun","workflowRunId","params","cancelWorkflowRun","pauseWorkflowRun","resumeWorkflowRun","listStepAttempts","address","port","ok","project","process","cwd","pop","timestamp","toISOString","count","overrides","targetDb","fixture","test","generator","generateBatch","destroy","strategy","where","fixtureDbConfig","fixtureDb","explorer","explore","includeRelations","maxDepth","sourceDb","getDB","results","importFromSource","imported","targetEntities","tableNames","raw","cleaned","uiDistPath","dirname","root","prefix","decorateReply","wildcard","headers","readFileSync","toString","projectName"],"mappings":"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"}