sonamu 0.8.11 → 0.8.13

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.
package/dist/ui/api.js CHANGED
@@ -19,7 +19,7 @@ import { FixtureManager } from "../testing/fixture-manager.js";
19
19
  import { BUILT_IN_TYPE_IDS, TemplateKey } from "../types/types.js";
20
20
  import { nonNullable } from "../utils/utils.js";
21
21
  import { setAiApi } from "./ai-api.js";
22
- import { editContent, getCddTree, readContent } from "./cdd-service.js";
22
+ import { editContent, getCddTree, openSourceFile, readContent } from "./cdd-service.js";
23
23
  export async function sonamuUIApiPlugin(fastify) {
24
24
  fastify.register(async (server)=>{
25
25
  // migrator
@@ -1104,6 +1104,13 @@ export async function sonamuUIApiPlugin(fastify) {
1104
1104
  const { filePath } = request.body;
1105
1105
  return editContent(filePath);
1106
1106
  });
1107
+ server.post("/api/cdd/openSource", async (request)=>{
1108
+ const { filePath } = request.body;
1109
+ openSourceFile(filePath);
1110
+ return {
1111
+ success: true
1112
+ };
1113
+ });
1107
1114
  // ui-web 빌드 파일 서빙
1108
1115
  const uiDistPath = path.resolve(import.meta.dirname, "../ui-web");
1109
1116
  // 정적 파일 서빙: 루트 폴더 전체 (assets, setting.svg 등)
@@ -1124,4 +1131,4 @@ export async function sonamuUIApiPlugin(fastify) {
1124
1131
  });
1125
1132
  }
1126
1133
 
1127
- //# 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\";\nimport { editContent, getCddTree, readContent } from \"./cdd-service\";\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 handler = () => {\n            clearTimeout(timeout);\n            resolve();\n          };\n\n          const timeout = setTimeout(() => {\n            Sonamu.syncer.eventEmitter.off(\"onHMRCompleted\", handler);\n            resolve();\n          }, 1500);\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        // 타겟 DB 설정 가져오기\n        const dbConfig = targetDb === \"fixture\" ? Sonamu.dbConfig.fixture : Sonamu.dbConfig.test;\n\n        // Knex 인스턴스 생성\n        const db = createKnexInstance(dbConfig);\n\n        try {\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          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        } finally {\n          await db.destroy();\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        // Fixture DB 설정 가져오기\n        const fixtureDbConfig = Sonamu.dbConfig.fixture;\n\n        // Knex 인스턴스 생성\n        const fixtureDb = createKnexInstance(fixtureDbConfig);\n\n        try {\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          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        } finally {\n          await fixtureDb.destroy();\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        // Source DB (production/development) - 읽기 전용\n        const sourceDb = DB.getDB(\"r\");\n\n        // Target DB (fixture)\n        const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n\n        try {\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          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        } finally {\n          // sourceDb는 Sonamu가 관리하므로 destroy하지 않음\n          await fixtureDb.destroy();\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        // Fixture DB 연결\n        const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n\n        try {\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          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        } finally {\n          await fixtureDb.destroy();\n        }\n      });\n\n      // CDD API\n      server.get(\"/api/cdd/tree\", async () => {\n        return getCddTree();\n      });\n\n      server.post<{ Body: { filePath: string } }>(\"/api/cdd/readContent\", async (request) => {\n        const { filePath } = request.body;\n        return readContent(filePath);\n      });\n\n      server.post<{ Body: { filePath: string } }>(\"/api/cdd/editContent\", async (request) => {\n        const { filePath } = request.body;\n        return editContent(filePath);\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","editContent","getCddTree","readContent","sonamuUIApiPlugin","fastify","register","server","migrator","waitForHMRCompleted","fn","waitPromise","Promise","resolve","handler","clearTimeout","timeout","setTimeout","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","filePath","uiDistPath","dirname","root","prefix","decorateReply","wildcard","headers","readFileSync","toString","projectName"],"mappings":"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"}
1134
+ //# 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\";\nimport { editContent, getCddTree, openSourceFile, readContent } from \"./cdd-service\";\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 handler = () => {\n            clearTimeout(timeout);\n            resolve();\n          };\n\n          const timeout = setTimeout(() => {\n            Sonamu.syncer.eventEmitter.off(\"onHMRCompleted\", handler);\n            resolve();\n          }, 1500);\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        // 타겟 DB 설정 가져오기\n        const dbConfig = targetDb === \"fixture\" ? Sonamu.dbConfig.fixture : Sonamu.dbConfig.test;\n\n        // Knex 인스턴스 생성\n        const db = createKnexInstance(dbConfig);\n\n        try {\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          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        } finally {\n          await db.destroy();\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        // Fixture DB 설정 가져오기\n        const fixtureDbConfig = Sonamu.dbConfig.fixture;\n\n        // Knex 인스턴스 생성\n        const fixtureDb = createKnexInstance(fixtureDbConfig);\n\n        try {\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          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        } finally {\n          await fixtureDb.destroy();\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        // Source DB (production/development) - 읽기 전용\n        const sourceDb = DB.getDB(\"r\");\n\n        // Target DB (fixture)\n        const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n\n        try {\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          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        } finally {\n          // sourceDb는 Sonamu가 관리하므로 destroy하지 않음\n          await fixtureDb.destroy();\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        // Fixture DB 연결\n        const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n\n        try {\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          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        } finally {\n          await fixtureDb.destroy();\n        }\n      });\n\n      // CDD API\n      server.get(\"/api/cdd/tree\", async () => {\n        return getCddTree();\n      });\n\n      server.post<{ Body: { filePath: string } }>(\"/api/cdd/readContent\", async (request) => {\n        const { filePath } = request.body;\n        return readContent(filePath);\n      });\n\n      server.post<{ Body: { filePath: string } }>(\"/api/cdd/editContent\", async (request) => {\n        const { filePath } = request.body;\n        return editContent(filePath);\n      });\n\n      server.post<{ Body: { filePath: string } }>(\"/api/cdd/openSource\", async (request) => {\n        const { filePath } = request.body;\n        openSourceFile(filePath);\n        return { success: true };\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","editContent","getCddTree","openSourceFile","readContent","sonamuUIApiPlugin","fastify","register","server","migrator","waitForHMRCompleted","fn","waitPromise","Promise","resolve","handler","clearTimeout","timeout","setTimeout","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","filePath","uiDistPath","dirname","root","prefix","decorateReply","wildcard","headers","readFileSync","toString","projectName"],"mappings":"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"}
@@ -21,4 +21,6 @@ export declare function editContent(filePath: string): Promise<{
21
21
  success: boolean;
22
22
  filePath: string;
23
23
  }>;
24
+ /** 소스 파일을 외부 에디터로 열기 (대기하지 않음) */
25
+ export declare function openSourceFile(filePath: string): void;
24
26
  //# sourceMappingURL=cdd-service.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"cdd-service.d.ts","sourceRoot":"","sources":["../../src/ui/cdd-service.ts"],"names":[],"mappings":"AAOA,MAAM,MAAM,WAAW,GAAG,UAAU,GAAG,MAAM,CAAC;AAE9C,MAAM,MAAM,WAAW,GAAG;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,yBAAyB;IACzB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,GAAG,WAAW,CAAC;IAC3B,mBAAmB;IACnB,QAAQ,CAAC,EAAE,WAAW,CAAC;IACvB,wBAAwB;IACxB,QAAQ,CAAC,EAAE,WAAW,EAAE,CAAC;CAC1B,CAAC;AAwDF,gCAAgC;AAChC,wBAAgB,UAAU,IAAI;IAAE,MAAM,EAAE,OAAO,CAAC;IAAC,IAAI,EAAE,WAAW,EAAE,CAAA;CAAE,CAOrE;AASD,mDAAmD;AACnD,wBAAgB,WAAW,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAarE;AAED,sCAAsC;AACtC,wBAAsB,WAAW,CAC/B,QAAQ,EAAE,MAAM,GACf,OAAO,CAAC;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAA;CAAE,CAAC,CA0CjD"}
1
+ {"version":3,"file":"cdd-service.d.ts","sourceRoot":"","sources":["../../src/ui/cdd-service.ts"],"names":[],"mappings":"AAOA,MAAM,MAAM,WAAW,GAAG,UAAU,GAAG,MAAM,CAAC;AAE9C,MAAM,MAAM,WAAW,GAAG;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,yBAAyB;IACzB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,GAAG,WAAW,CAAC;IAC3B,mBAAmB;IACnB,QAAQ,CAAC,EAAE,WAAW,CAAC;IACvB,wBAAwB;IACxB,QAAQ,CAAC,EAAE,WAAW,EAAE,CAAC;CAC1B,CAAC;AAwDF,gCAAgC;AAChC,wBAAgB,UAAU,IAAI;IAAE,MAAM,EAAE,OAAO,CAAC;IAAC,IAAI,EAAE,WAAW,EAAE,CAAA;CAAE,CAOrE;AASD,mDAAmD;AACnD,wBAAgB,WAAW,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAarE;AAED,sCAAsC;AACtC,wBAAsB,WAAW,CAC/B,QAAQ,EAAE,MAAM,GACf,OAAO,CAAC;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAA;CAAE,CAAC,CA0CjD;AA0DD,kCAAkC;AAClC,wBAAgB,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI,CAarD"}