sonamu 0.8.21 → 0.8.22

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
@@ -871,19 +871,24 @@ export async function sonamuUIApiPlugin(fastify) {
871
871
  });
872
872
  // Tasks API
873
873
  server.get("/api/tasks/status", async ()=>{
874
- return {
875
- active: Sonamu.workflows !== null
876
- };
874
+ try {
875
+ Sonamu.workflows;
876
+ return {
877
+ active: true
878
+ };
879
+ } catch {
880
+ return {
881
+ active: false
882
+ };
883
+ }
877
884
  });
878
885
  server.get("/api/tasks/workflowDefinitions", async ()=>{
879
- if (!Sonamu.workflows) throw new Error("Workflows not initialized (database not configured)");
880
886
  const definitions = Sonamu.workflows.workflowDefinitions;
881
887
  return {
882
888
  definitions
883
889
  };
884
890
  });
885
891
  server.get("/api/tasks/workflowRuns", async (request)=>{
886
- if (!Sonamu.workflows) throw new Error("Workflows not initialized (database not configured)");
887
892
  const backend = Sonamu.workflows.backend;
888
893
  const { limit, after, before, order, status, workflowName, createdAfter, createdBefore } = request.query;
889
894
  return backend.listWorkflowRuns({
@@ -898,7 +903,6 @@ export async function sonamuUIApiPlugin(fastify) {
898
903
  });
899
904
  });
900
905
  server.get("/api/tasks/workflowRuns/:id", async (request)=>{
901
- if (!Sonamu.workflows) throw new Error("Workflows not initialized (database not configured)");
902
906
  const backend = Sonamu.workflows.backend;
903
907
  const workflowRun = await backend.getWorkflowRun({
904
908
  workflowRunId: request.params.id
@@ -909,28 +913,24 @@ export async function sonamuUIApiPlugin(fastify) {
909
913
  return workflowRun;
910
914
  });
911
915
  server.post("/api/tasks/workflowRuns/:id/cancel", async (request)=>{
912
- if (!Sonamu.workflows) throw new Error("Workflows not initialized (database not configured)");
913
916
  const backend = Sonamu.workflows.backend;
914
917
  return backend.cancelWorkflowRun({
915
918
  workflowRunId: request.params.id
916
919
  });
917
920
  });
918
921
  server.post("/api/tasks/workflowRuns/:id/pause", async (request)=>{
919
- if (!Sonamu.workflows) throw new Error("Workflows not initialized (database not configured)");
920
922
  const backend = Sonamu.workflows.backend;
921
923
  return backend.pauseWorkflowRun({
922
924
  workflowRunId: request.params.id
923
925
  });
924
926
  });
925
927
  server.post("/api/tasks/workflowRuns/:id/resume", async (request)=>{
926
- if (!Sonamu.workflows) throw new Error("Workflows not initialized (database not configured)");
927
928
  const backend = Sonamu.workflows.backend;
928
929
  return backend.resumeWorkflowRun({
929
930
  workflowRunId: request.params.id
930
931
  });
931
932
  });
932
933
  server.get("/api/tasks/workflowRuns/:id/steps", async (request)=>{
933
- if (!Sonamu.workflows) throw new Error("Workflows not initialized (database not configured)");
934
934
  const backend = Sonamu.workflows.backend;
935
935
  const { limit, after, before } = request.query;
936
936
  return backend.listStepAttempts({
@@ -1146,4 +1146,4 @@ export async function sonamuUIApiPlugin(fastify) {
1146
1146
  });
1147
1147
  }
1148
1148
 
1149
- //# 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 {\n  editContent,\n  editSchema,\n  getCddTree,\n  getDashboard,\n  listSchemas,\n  openSourceFile,\n  readContent,\n  readSchema,\n} 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        return { active: Sonamu.workflows !== null };\n      });\n\n      server.get(\"/api/tasks/workflowDefinitions\", async () => {\n        if (!Sonamu.workflows)\n          throw new Error(\"Workflows not initialized (database not configured)\");\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        if (!Sonamu.workflows)\n          throw new Error(\"Workflows not initialized (database not configured)\");\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        if (!Sonamu.workflows)\n          throw new Error(\"Workflows not initialized (database not configured)\");\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        if (!Sonamu.workflows)\n          throw new Error(\"Workflows not initialized (database not configured)\");\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        if (!Sonamu.workflows)\n          throw new Error(\"Workflows not initialized (database not configured)\");\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        if (!Sonamu.workflows)\n          throw new Error(\"Workflows not initialized (database not configured)\");\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        if (!Sonamu.workflows)\n          throw new Error(\"Workflows not initialized (database not configured)\");\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/dashboard\", async () => {\n        return getDashboard();\n      });\n\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      // CDD Schema API\n      server.get(\"/api/cdd/schemas\", async () => {\n        return listSchemas();\n      });\n\n      server.post<{ Body: { schemaKey: string } }>(\"/api/cdd/readSchema\", async (request) => {\n        const { schemaKey } = request.body;\n        return readSchema(schemaKey);\n      });\n\n      server.post<{ Body: { schemaKey: string } }>(\"/api/cdd/editSchema\", async (request) => {\n        const { schemaKey } = request.body;\n        return editSchema(schemaKey);\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","editSchema","getCddTree","getDashboard","listSchemas","openSourceFile","readContent","readSchema","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","schemaKey","uiDistPath","dirname","root","prefix","decorateReply","wildcard","headers","readFileSync","toString","projectName"],"mappings":"AAAA,SAASA,QAAQ,QAAQ,gBAAgB;AAEzC,OAAOC,QAAQ,KAAK;AACpB,OAAOC,gBAAgB,aAAa;AAEpC,OAAOC,UAAU,OAAO;AACxB,SAASC,KAAK,QAAQ,UAAU;AAChC,SAASC,MAAM,QAAQ,mBAAgB;AACvC,SAASC,EAAE,QAA6B,oBAAiB;AACzD,SAASC,kBAAkB,QAAQ,sBAAmB;AACtD,SAASC,EAAE,QAAQ,gBAAa;AAChC,SAASC,gBAAgB,QAAQ,+BAA4B;AAE7D,SAASC,aAAa,QAAQ,8BAA2B;AACzD,SACEC,mBAAmB,EACnBC,aAAa,EACbC,2BAA2B,QACtB,iCAA8B;AACrC,SAA+BC,QAAQ,QAAQ,2BAAwB;AACvE,SAASC,YAAY,QAAwC,gCAA6B;AAC1F,SAASC,eAAe,QAAQ,kCAA+B;AAC/D,SAASC,YAAY,QAAQ,8BAA2B;AACxD,SAASC,gBAAgB,QAAQ,kCAA+B;AAChE,SAAqCC,cAAc,QAAQ,gCAA6B;AACxF,SACEC,iBAAiB,EASjBC,WAAW,QACN,oBAAiB;AACxB,SAASC,WAAW,QAAQ,oBAAiB;AAC7C,SAASC,QAAQ,QAAQ,cAAW;AACpC,SACEC,WAAW,EACXC,UAAU,EACVC,UAAU,EACVC,YAAY,EACZC,WAAW,EACXC,cAAc,EACdC,WAAW,EACXC,UAAU,QACL,mBAAgB;AAEvB,OAAO,eAAeC,kBAAkBC,OAAwB;IAC9DA,QAAQC,QAAQ,CACd,OAAOC;QACL,WAAW;QACX,MAAMC,WAAW,IAAItB;QAErB,sBAAsB;QACtB,eAAeuB,oBAAuBC,EAAoB;YACxD,MAAMC,cAAc,IAAIC,QAAc,CAACC;gBACrC,MAAMC,UAAU;oBACdC,aAAaC;oBACbH;gBACF;gBAEA,MAAMG,UAAUC,WAAW;oBACzBxC,OAAOyC,MAAM,CAACC,YAAY,CAACC,GAAG,CAAC,kBAAkBN;oBACjDD;gBACF,GAAG;gBAEHpC,OAAOyC,MAAM,CAACC,YAAY,CAACE,IAAI,CAAC,kBAAkBP;YACpD;YAEA,MAAMQ,SAAS,MAAMZ;YACrB,MAAMC;YACN,OAAOW;QACT;QAEA,MAAM3B,SAASY;QAEfA,OAAOgB,GAAG,CAAC,sBAAsB;YAC/B,OAAO9C,OAAO+C,MAAM;QACtB;QAEAjB,OAAOgB,GAAG,CAMP,yBAAyB,OAAOE;YACjC,MAAM,EAAEC,QAAQ,EAAEC,MAAM,EAAEC,OAAO,EAAE,GAAGH,QAAQI,KAAK;YAEnD,MAAMC,aAAa,AAAC,CAAA;gBAClB,IAAIJ,YAAYC,QAAQ;oBACtB,MAAMI,SAASjD,cAAcyC,GAAG,CAACG;oBACjC,MAAM,EAAEM,KAAK,EAAE,GAAGD;oBAElB,MAAM,EAAEE,WAAW,EAAE,GAAGxD;oBACxB,MAAMyD,WAAW,AAAC,CAAA;wBAChB,OAAQP;4BACN,KAAK;gCACH,OAAO,GAAGK,MAAM3D,EAAE,CAAC,SAAS,CAAC;4BAC/B,KAAK;gCACH,OAAO,GAAG2D,MAAM3D,EAAE,CAAC,YAAY,CAAC;4BAClC,KAAK;gCACH,OAAO,GAAG2D,MAAM3D,EAAE,CAAC,aAAa,CAAC;wBACrC;oBACF,CAAA;oBACA,OAAO,GAAG4D,YAAY,iBAAiB,EAAEF,OAAOC,KAAK,CAACG,QAAQ,CAAC,CAAC,EAAED,UAAU;gBAC9E,OAAO;oBACL,IAAI,CAACN,SAAS;wBACZ,MAAM,IAAI7C,oBAAoBH,GAAG;oBACnC;oBACA,OAAOgD;gBACT;YACF,CAAA;YACAxD,SAAS,CAAC,KAAK,EAAE0D,YAAY;QAC/B;QAEAvB,OAAOgB,GAAG,CAKP,4BAA4B,OAAOE;YACpC,MAAM,EAAEW,MAAM,EAAEV,QAAQ,EAAE,GAAGD,QAAQI,KAAK;YAE1C,SAAS;YACT,MAAMQ,WAAW,IAAIC,IAAoB;gBACvC;oBAAC;oBAAU;iBAAK;gBAChB;oBAAC;oBAAQ;iBAAK;gBACd;oBAAC;oBAAS;iBAAM;gBAChB;oBAAC;oBAAU;iBAAS;gBACpB;oBAAC;oBAAO;iBAAM;gBACd;oBAAC;oBAAM;iBAAK;gBACZ;oBAAC;oBAAQ,CAAC,WAAW,CAAC;iBAAC;gBACvB;oBAAC;oBAAS;iBAAc;gBACxB;oBAAC;oBAAU;iBAAe;gBAC1B;oBAAC;oBAAQ;iBAAK;gBACd;oBAAC;oBAAM;iBAAK;gBACZ;oBAAC;oBAAW;iBAAK;gBACjB;oBAAC;oBAAW;iBAAK;gBACjB;oBAAC;oBAAW;iBAAK;gBACjB;oBAAC;oBAAM;iBAAK;gBACZ;oBAAC;oBAAQ;iBAAK;gBACd;oBAAC;oBAAQ;iBAAK;gBACd;oBAAC;oBAAM;iBAAO;gBACd;oBAAC;oBAAM;iBAAO;gBACd;oBAAC;oBAAO;iBAAM;gBACd;oBAAC;oBAAO;iBAAQ;gBAChB;oBAAC;oBAAS;iBAAK;gBACf;oBAAC;oBAAQ;iBAAK;gBACd;oBAAC;oBAAK;iBAAM;gBACZ;oBAAC;oBAAK;iBAAM;gBACZ;oBAAC;oBAAW;iBAAK;gBACjB;oBAAC;oBAAS;iBAAK;gBACf;oBAAC;oBAAS;iBAAI;gBACd;oBAAC;oBAAS;iBAAM;gBAChB;oBAAC;oBAAS;iBAAK;gBACf;oBAAC;oBAAQ;iBAAM;gBACf;oBAAC;oBAAO;iBAAK;gBACb;oBAAC;oBAAS;iBAAK;gBACf;oBAAC;oBAAU;iBAAM;gBACjB;oBAAC;oBAAQ;iBAAK;gBACd;oBAAC;oBAAO;iBAAO;gBACf;oBAAC;oBAAM;iBAAK;gBACZ;oBAAC;oBAAQ;iBAAK;gBACd;oBAAC;oBAAW;iBAAK;gBACjB;oBAAC;oBAAW;iBAAO;gBACnB;oBAAC;oBAAY;iBAAO;gBACpB;oBAAC;oBAAQ;iBAAM;gBACf;oBAAC;oBAAO;iBAAI;gBACZ;oBAAC;oBAAO;iBAAK;gBACb;oBAAC;oBAAY;iBAAK;gBAClB;oBAAC;oBAAU;iBAAK;aACjB;YACD,0CAA0C;YAC1C,KAAK,MAAMZ,YAAY5C,cAAcyD,SAAS,GAAI;gBAChD,MAAMR,SAASjD,cAAcyC,GAAG,CAACG;gBACjC,IAAI,AAACK,CAAAA,OAAOS,KAAK,IAAI,EAAC,MAAO,IAAI;oBAC/BH,SAASI,GAAG,CAACnE,WAAWoE,UAAU,CAACX,OAAOY,EAAE,GAAGZ,OAAOS,KAAK;oBAC3DH,SAASI,GAAG,CACVnE,WAAWoE,UAAU,CAACpE,WAAWsE,SAAS,CAACb,OAAOY,EAAE,IACpD,GAAGZ,OAAOS,KAAK,CAAC,GAAG,CAAC;gBAExB;gBAEAT,OAAOc,KAAK,CAACC,OAAO,CAAC,CAACC;oBACpB,IAAIV,SAASW,GAAG,CAACD,KAAKE,IAAI,GAAG;wBAC3B;oBACF;oBACA,IAAIF,KAAKG,IAAI,EAAE;wBACbb,SAASI,GAAG,CAACM,KAAKE,IAAI,EAAEF,KAAKG,IAAI,CAACC,OAAO,CAACpB,OAAOS,KAAK,IAAI,IAAI;oBAChE;gBACF;YACF;YAEA,MAAMY,YAAY,AAAC,CAAA;gBACjB,mBAAmB;gBACnB,MAAMC,QAAQjB,OAAOkB,KAAK,CAAC;gBAC3B,MAAMC,eAAe;uBAAI/E,MAAM6E,MAAMG,MAAM,EAAE,GAAG,CAAC;iBAAG,CAACC,OAAO,CAAC,CAACC;oBAC5D,OAAO;2BACFlF,MAAM,GAAG6E,MAAMG,MAAM,GAAGE,MAAM,GAAG,CAACC;4BACnC,OAAO;gCACLD;gCACAE,GAAGP,MAAMQ,KAAK,CAACF,KAAKA,MAAMD,KAAKI,IAAI,CAAC;4BACtC;wBACF;qBACD;gBACH;gBAEA,6CAA6C;gBAC7C,MAAMC,kBAAkB,eAAe,sCAAsC;gBAC7E,IAAIC,YAAsB;uBAAIX;iBAAM;gBACpC,KAAK,MAAMY,QAAQV,aAAc;oBAC/B,MAAMW,YAAYF,UAAUF,IAAI,CAAC;oBACjC,IAAII,UAAUC,QAAQ,CAACF,KAAKL,CAAC,KAAKvB,SAASW,GAAG,CAACiB,KAAKL,CAAC,GAAG;wBACtDI,YAAYE,UACTf,OAAO,CAACc,KAAKL,CAAC,EAAEG,kBAAkB1B,SAASd,GAAG,CAAC0C,KAAKL,CAAC,GACrDN,KAAK,CAAC;oBACX;gBACF;gBAEA,OAAOU,UACJI,GAAG,CAAC,CAACC;oBACJ,IAAIA,EAAEC,UAAU,CAACP,kBAAkB;wBACjC,OAAOM,EAAElB,OAAO,CAACY,iBAAiB;oBACpC,OAAO;wBACL,OAAOM,EAAEE,WAAW;oBACtB;gBACF,GACCT,IAAI,CAAC,IACLX,OAAO,CAAC,eAAezB,WAAW5C,cAAcyC,GAAG,CAACG,UAAUc,KAAK,GAAG;YAC3E,CAAA;YAEA,OAAO;gBAAEY;YAAU;QACrB;QAEA7C,OAAOgB,GAAG,CAAC,wBAAwB;YACjC,MAAMiD,YAAY1F,cAAcyD,SAAS;YAEzC,SAASkC,kBAAkBC,UAA6B;gBACtD,OAAOA,WAAWjB,OAAO,CAAC,CAACkB;oBACzB,MAAM,EAAEC,QAAQ,EAAE,GAAGC,MAAM,GAAGF;oBAC9B,OAAO;wBAACE;2BAASJ,kBAAkBG;qBAAU;gBAC/C;YACF;YAEA,MAAME,WAAW,MAAMlE,QAAQmE,GAAG,CAChCP,UAAUJ,GAAG,CAAC,CAAC1C;gBACb,MAAMK,SAASjD,cAAcyC,GAAG,CAACG;gBACjC,MAAMgD,aAAa3C,OAAOiD,aAAa;gBAEvC,OAAO;oBACL,GAAGjD,MAAM;oBACT0C,mBAAmBA,kBAAkBC;gBACvC;YACF;YAGFI,SAASG,IAAI,CAAC,CAACC,GAAGC;gBAChB,MAAMC,MAAMF,EAAEG,QAAQ,IAAIH,EAAEvC,EAAE;gBAC9B,MAAM2C,MAAMH,EAAEE,QAAQ,IAAIF,EAAExC,EAAE;gBAC9B,IAAIyC,MAAME,KAAK,OAAO,CAAC;gBACvB,IAAIF,MAAME,KAAK,OAAO;gBACtB,IAAIF,QAAQE,KAAK;oBACf,IAAIJ,EAAEG,QAAQ,KAAKE,WAAW,OAAO,CAAC;oBACtC,IAAIJ,EAAEE,QAAQ,KAAKE,WAAW,OAAO;oBACrC,OAAO;gBACT;gBACA,OAAO;YACT;YACA,OAAO;gBAAET;YAAS;QACpB;QAEAvE,OAAOgB,GAAG,CAKP,uBAAuB,OAAOE;YAC/B,MAAM,EAAE+D,MAAM,EAAEC,MAAM,EAAE,GAAGhE,QAAQI,KAAK;YAExC,IAAI4D,WAAW,KAAK;gBAClB,MAAMhH,OAAOyC,MAAM,CAACwE,aAAa;YACnC;YAEA,MAAMC,UAAU,AAAC,CAAA;gBACf,iBAAiB;gBACjB,MAAMC,iBAAiBC,OAAOC,OAAO,CAACrH,OAAOyC,MAAM,CAAC6E,KAAK,EACtDP,MAAM,CAAC,CAAC,CAACQ,SAASC,QAAQ,GAAK,AAACA,QAAQC,IAAI,CAACC,GAAG,CAACC,IAAI,KAAgB,QACrEhC,GAAG,CAAC,CAAC,CAACiC,QAAQC,SAAS,GAAKD;gBAE/B,0BAA0B;gBAC1B,MAAME,iBAAiB;uBAAI/G;iBAAkB;gBAE7C,WAAW;gBACX,MAAMgH,aAAa;uBAAID;uBAAmBX;iBAAe;gBAEzD,IAAIJ,WAAW,SAAS;oBACtB,OAAOgB;gBACT;gBAEA,MAAMC,UAAU3H,cAAcyD,SAAS,GAAGkB,OAAO,CAAC,CAAC/B;oBACjD,MAAMK,SAASjD,cAAcyC,GAAG,CAACG;oBACjC,OAAOmE,OAAOa,IAAI,CAAC3E,OAAO4E,UAAU;gBACtC;gBAEA,IAAInB,WAAW,SAAS;oBACtB,OAAOiB;gBACT,OAAO;oBACL,OAAO;2BAAID;2BAAeC;qBAAQ;gBACpC;YACF,CAAA;YAEA,OAAO;gBACLd;YACF;QACF;QAEApF,OAAOqG,IAAI,CASR,sBAAsB,OAAOnF;YAC9B,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEoG,IAAI,EAAE,GAAGpF,QAAQqF,IAAI;gBAC7B,MAAMrI,OAAOyC,MAAM,CAAC6F,YAAY,CAAC;oBAAE,GAAGF,IAAI;oBAAEnF,UAAUmF,KAAKlE,EAAE;gBAAC;gBAE9D,OAAO;YACT;QACF;QAEApC,OAAOqG,IAAI,CAIR,mBAAmB,OAAOnF;YAC3B,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAE,GAAGD,QAAQqF,IAAI;gBACjC,OAAO,MAAMrI,OAAOyC,MAAM,CAAC8F,SAAS,CAACtF;YACvC;QACF;QAEAnB,OAAOqG,IAAI,CASR,gCAAgC,OAAOnF;YACxC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAEuF,SAAS,EAAE,GAAGxF,QAAQqF,IAAI;gBAC5C,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBACjCK,OAAOS,KAAK,GAAGyE,UAAUzE,KAAK;gBAC9BT,OAAOmF,KAAK,GAAGD,UAAUC,KAAK;gBAC9BnF,OAAOsD,QAAQ,GAAG4B,UAAU5B,QAAQ;gBACpC,MAAMtD,OAAOoF,IAAI;gBAEjB,OAAO;YACT;QACF;QAEA5G,OAAOqG,IAAI,CAOR,4BAA4B,OAAOnF;YACpC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAE0F,SAAS,EAAEC,MAAM,EAAEC,cAAc,EAAE,GAAG7F,QAAQqF,IAAI;gBACpE,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBACjCK,OAAOwF,OAAO,CAACH,UAAU,GAAGC;gBAC5B,IAAIC,mBAAmB/B,WAAW;oBAChC,IAAI+B,eAAe9D,MAAM,GAAG,GAAG;wBAC7BzB,OAAOyF,eAAe,CAACJ,UAAU,GAAGE;oBACtC,OAAO;wBACL,OAAOvF,OAAOyF,eAAe,CAACJ,UAAU;oBAC1C;gBACF;gBACA,MAAMrF,OAAOoF,IAAI;gBAEjB,OAAO;oBAAEM,SAASJ;oBAAQK,iBAAiBJ;gBAAe;YAC5D;QACF;QAEA/G,OAAOqG,IAAI,CAKR,yBAAyB,OAAOnF;YACjC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAE0F,SAAS,EAAE,GAAG3F,QAAQqF,IAAI;gBAC5C,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBACjC,OAAOK,OAAOwF,OAAO,CAACH,UAAU;gBAChC,OAAOrF,OAAOyF,eAAe,CAACJ,UAAU;gBACxC,MAAMrF,OAAOoF,IAAI;gBAEjB,OAAO;YACT;QACF;QAEA5G,OAAOqG,IAAI,CAMR,0BAA0B,OAAOnF;YAClC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAEiG,EAAE,EAAEC,OAAO,EAAE,GAAGnG,QAAQqF,IAAI;gBAC9C,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBACjC,MAAMK,OAAO8F,UAAU,CAACD,SAASD;gBACjC,OAAO;YACT;QACF;QAEApH,OAAOqG,IAAI,CAMR,0BAA0B,OAAOnF;YAClC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAEiG,EAAE,EAAEC,OAAO,EAAE,GAAGnG,QAAQqF,IAAI;gBAE9C,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBACjCK,OAAO+F,UAAU,CAACF,SAASD;gBAE3B,OAAO;YACT;QACF;QAEApH,OAAOqG,IAAI,CAKR,uBAAuB,OAAOnF;YAC/B,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAEiG,EAAE,EAAE,GAAGlG,QAAQqF,IAAI;gBAErC,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBACjCK,OAAOgG,OAAO,CAACJ;gBACf,OAAO;YACT;QACF;QAEApH,OAAOqG,IAAI,CAMR,wBAAwB,OAAOnF;YAChC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAEiG,EAAE,EAAEK,EAAE,EAAE,GAAGvG,QAAQqF,IAAI;gBAEzC,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBACjCK,OAAOkG,QAAQ,CAACN,IAAIK;gBAEpB,OAAO;YACT;QACF;QAEAzH,OAAOqG,IAAI,CAKR,6BAA6B,OAAOnF;YACrC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAEwG,OAAO,EAAE,GAAGzG,QAAQqF,IAAI;gBAC1C,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBACjCK,OAAOmG,OAAO,GAAGA;gBACjB,MAAMnG,OAAOoF,IAAI;gBAEjB,OAAO;oBAAEM,SAASS;gBAAQ;YAC5B;QACF;QAEA3H,OAAOqG,IAAI,CAKR,gCAAgC,OAAOnF;YACxC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAEiF,UAAU,EAAE,GAAGlF,QAAQqF,IAAI;gBAC7C,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBACjCK,OAAO4E,UAAU,GAAGA;gBACpB,MAAM5E,OAAOoF,IAAI;gBAEjB,OAAO;oBAAEM,SAASd;gBAAW;YAC/B;QACF;QAEApG,OAAOqG,IAAI,CAKR,4BAA4B,OAAOnF;YACpC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAEyG,SAAS,EAAE,GAAG1G,QAAQqF,IAAI;gBAC5C,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBAEjC,IAAIK,OAAO4E,UAAU,CAACwB,UAAU,EAAE;oBAChC,MAAM,IAAIC,MAAM,CAAC,mBAAmB,EAAED,WAAW;gBACnD;gBAEApG,OAAO4E,UAAU,CAACwB,UAAU,GAAG;oBAC7B,GAAIA,UAAUE,QAAQ,CAAC,YACnB;wBACEC,QAAQ;wBACRC,QAAQ;oBACV,IACA;wBACE,IAAI;oBACN,CAAC;gBACP;gBACA,MAAMxG,OAAOoF,IAAI;gBAEjB,OAAO;YACT;QACF;QAEA5G,OAAOqG,IAAI,CAQR,4BAA4B,OAAOnF;YACpC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAE8G,MAAM,EAAE,GAAG/G,QAAQqF,IAAI;gBACzC,MAAMtC,YAAY1F,cAAcyD,SAAS;gBACzC,MAAMkG,WAAWjE,UAAUkE,IAAI,CAAC,CAAChH;oBAC/B,MAAMK,SAASjD,cAAcyC,GAAG,CAACG;oBACjC,OAAOmE,OAAOa,IAAI,CAAC3E,OAAO4E,UAAU,EAAExC,QAAQ,CAACqE,OAAOG,KAAK;gBAC7D;gBACA,IAAIF,UAAU;oBACZ,MAAM,IAAIL,MAAM,CAAC,mBAAmB,EAAEI,OAAOG,KAAK,EAAE;gBACtD;gBAEA,MAAM5G,SAASjD,cAAcyC,GAAG,CAACG;gBACjCK,OAAO4E,UAAU,CAAC6B,OAAOG,KAAK,CAAC,GAAG5G,OAAO4E,UAAU,CAAC6B,OAAOI,MAAM,CAAC;gBAClE,OAAO7G,OAAO4E,UAAU,CAAC6B,OAAOI,MAAM,CAAC;gBAEvC,MAAM7G,OAAOoF,IAAI;gBAEjB,KAAK,MAAMzF,YAAY8C,UAAW;oBAChC,MAAMzC,SAASjD,cAAcyC,GAAG,CAACG;oBACjC,KAAK,MAAMqB,QAAQhB,OAAOc,KAAK,CAAE;wBAC/B,IAAIE,KAAKqD,IAAI,KAAK,UAAUrD,KAAKJ,EAAE,KAAK6F,OAAOI,MAAM,EAAE;4BACrD7F,KAAKJ,EAAE,GAAG6F,OAAOG,KAAK;wBACxB;oBACF;oBACA,MAAM5G,OAAOoF,IAAI;gBACnB;YACF;QACF;QAEA5G,OAAOqG,IAAI,CAKR,4BAA4B,OAAOnF;YACpC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAE8G,MAAM,EAAE,GAAG/G,QAAQqF,IAAI;gBAEzC,MAAMtC,YAAY1F,cAAcyD,SAAS;gBACzC,MAAMsG,eAAerE,UAClBf,OAAO,CAAC,CAAC/B,WAAa5C,cAAcyC,GAAG,CAACG,UAAUmB,KAAK,EACvD6F,IAAI,CAAC,CAAC3F,OAASA,KAAKqD,IAAI,KAAK,UAAUrD,KAAKJ,EAAE,KAAK6F;gBACtD,IAAIK,cAAc;oBAChB,MAAM,IAAIT,MAAM,GAAGI,OAAO,mBAAmB,CAAC;gBAChD;gBAEA,MAAMzG,SAASjD,cAAcyC,GAAG,CAACG;gBACjC,OAAOK,OAAO4E,UAAU,CAAC6B,OAAO;gBAChC,MAAMzG,OAAOoF,IAAI;YACnB;QACF;QAEA5G,OAAOqG,IAAI,CASR,0BAA0B,OAAOnF;YAClC,OAAO,MAAMhB,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAEoH,MAAM,EAAEC,QAAQ,EAAEP,MAAM,EAAEpB,SAAS,EAAE4B,IAAI,EAAE,GAAGvH,QAAQqF,IAAI;gBAC5E,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;gBAEjC,IAAIoH,WAAW,UAAU;oBACvB/G,OAAOiH,IAAI,GAAGA;gBAChB,OAAO,IAAIF,WAAW,UAAUC,UAAU;oBACxC,MAAMhG,OAAOhB,OAAOc,KAAK,CAACoG,IAAI,CAAC,CAACC,IAAMA,EAAEjG,IAAI,KAAK8F;oBACjD,IAAIhG,MAAM;wBACPA,KAAyBiG,IAAI,GAAGA;oBACnC;gBACF,OAAO,IAAIF,WAAW,UAAUN,QAAQ;oBACtCzG,OAAOoH,SAAS,CAACX,OAAO,GAAGQ;gBAC7B,OAAO,IAAIF,WAAW,YAAY1B,WAAW;oBAC3CrF,OAAOqH,WAAW,CAAChC,UAAU,GAAG4B;gBAClC;gBAEA,MAAMjH,OAAOoF,IAAI;gBACjB,OAAO;YACT;QACF;QAEA5G,OAAOqG,IAAI,CAOR,6BAA6B,OAAOnF,SAAS4H;YAC9C,OAAO,MAAM5I,oBAAoB;gBAC/B,MAAM,EAAEiB,QAAQ,EAAE4H,gBAAgB,EAAEC,SAAS,EAAEC,MAAM,EAAE,GAAG/H,QAAQqF,IAAI;gBAEtE,IAAI;oBACF,kBAAkB;oBAClB,MAAM/E,SAASjD,cAAcyC,GAAG,CAACG;oBAEjC,kDAAkD;oBAClD,MAAM+H,kBACJD,UAAW/K,OAAO+C,MAAM,CAACkI,IAAI,CAACC,aAAa;oBAE7C,UAAU;oBACV,MAAMrI,SAAS,MAAMS,OAAO6H,aAAa,CAAC;wBACxCN,kBAAkBA,oBAAoB;wBACtCC,WAAWA,aAAa;wBACxBC,QAAQC;oBACV;oBAEA,OAAOnI;gBACT,EAAE,OAAOuI,OAAgB;oBACvB,MAAMC,UAAUD,iBAAiBzB,QAAQyB,MAAMC,OAAO,GAAGC,OAAOF;oBAEhE,mBAAmB;oBACnB,IAAIC,QAAQ3F,QAAQ,CAAC,mBAAmB;wBACtCkF,MAAMW,MAAM,CAAC;wBACb,OAAO;4BACLC,SAAS;4BACTJ,OAAO,CAAC,kBAAkB,EAAEnI,UAAU;wBACxC;oBACF;oBAEA,WAAW;oBACX,IAAIoI,QAAQ3F,QAAQ,CAAC,gCAAgC;wBACnDkF,MAAMW,MAAM,CAAC;wBACb,OAAO;4BACLC,SAAS;4BACTJ,OAAO;wBACT;oBACF;oBAEA,aAAa;oBACb,IAAIC,QAAQ3F,QAAQ,CAAC,wBAAwB;wBAC3CkF,MAAMW,MAAM,CAAC;wBACb,OAAO;4BACLC,SAAS;4BACTJ,OAAO;wBACT;oBACF;oBAEA,QAAQ;oBACRR,MAAMW,MAAM,CAAC;oBACb,OAAO;wBACLC,SAAS;wBACTJ,OAAO,CAAC,wBAAwB,EAAEC,SAAS;oBAC7C;gBACF;YACF;QACF;QAEAvJ,OAAOgB,GAAG,CAIP,+BAA+B,OAAOE;YACvC,MAAM,EAAEC,QAAQ,EAAE,GAAGD,QAAQI,KAAK;YAClC,MAAME,SAASjD,cAAcyC,GAAG,CAACG;YACjC,MAAMwI,UAAUnI,OAAOoI,eAAe;YACtC,OAAO;gBAAED;YAAQ;QACnB;QAEA3J,OAAOgB,GAAG,CAAC,0BAA0B;YACnC,MAAMyI,SAAS,MAAMxJ,SAAS4J,SAAS;YAEvC,OAAO;gBAAEJ;YAAO;QAClB;QAEAzJ,OAAOqG,IAAI,CAST,6BACA,OAAOnF;YACL,MAAM,EAAE4I,MAAM,EAAEC,OAAO,EAAEC,KAAK,EAAEC,WAAW,EAAEC,SAAS,EAAE,GAAGhJ,QAAQqF,IAAI;YAEvE,IAAIuD,WAAW,UAAU;gBACvB,OAAO7J,SAASkK,aAAa;YAC/B;YAEA,0BAA0B;YAC1B,IAAIL,WAAW,SAAS;gBACtB,MAAMM,eAAe,IAAIxL;gBACzB,MAAMyL,mBAAmBN,QAAQ5B,IAAI,CAAC,CAACmC,IAAMF,aAAaG,wBAAwB,CAACD;gBAEnF,kBAAkB;gBAClB,MAAME,aAAa;oBAAC;oBAAa;oBAAa;oBAAW;iBAAM;gBAC/D,MAAMC,gBAAgBV,QAAQW,KAAK,CAAC,CAACnC;oBACnC,MAAMoC,eAAezM,OAAO0M,QAAQ,CAACrC,OAAO;oBAC5C,MAAMsC,OAAO,AAACF,cAAcG,YAAkCD,QAAQ;oBACtE,OAAOL,WAAW5G,QAAQ,CAACiH,KAAKE,WAAW;gBAC7C;gBAEA,IAAIV,oBAAoBD,aAAaY,YAAY,MAAM,CAACP,eAAe;oBACrE,MAAM,EAAEQ,KAAK,EAAE,GAAG,MAAMhL,SAAS4J,SAAS;oBAE1C,yCAAyC;oBACzC,MAAMqB,oBAAoB;2BACrB,IAAIC,IACLF,MACGhG,MAAM,CAAC,CAACmG,OAASrB,QAAQnG,QAAQ,CAACwH,KAAKC,OAAO,GAC9CnI,OAAO,CAAC,CAACkI,OAASA,KAAKE,OAAO;qBAEpC;oBAED,IAAIJ,kBAAkBjI,MAAM,GAAG,GAAG;wBAChC,cAAc;wBACd,MAAMsI,WAAW,MAAMnB,aAAaoB,kBAAkB,CAACN;wBAEvD,IAAIK,UAAU;4BACZ,sBAAsB;4BACtB,MAAM,EAAEE,QAAQ,EAAEC,QAAQ,EAAE,GAAG,MAAMtB,aAAauB,aAAa,CAC7DJ,SAASK,OAAO,EAChBL,SAASM,EAAE;4BAGb,IAAIJ,UAAU;gCACZ,WAAW;gCACX,MAAM1K,SAAS,MAAMd,SAAS6L,SAAS,CAAChC,QAAQC;gCAChD,IAAIhJ,OAAOkC,MAAM,GAAG,GAAG;oCACrB,MAAMmH,aAAa2B,YAAY,CAC7BR,SAASK,OAAO,EAChBL,SAASM,EAAE,EACX9K,QACAmJ;gCAEJ;gCACA,OAAOnJ;4BACT,OAAO,IAAI2K,UAAU;gCACnB,MAAM,IAAIlN,oBAAoBH,GAAG;4BACnC,OAAO,IAAI2L,OAAO;gCAChB,WAAW;gCACX,MAAMI,aAAa4B,aAAa,CAC9BT,SAASK,OAAO,EAChBL,SAASM,EAAE,EACX5B,eAAe,SACfC;gCAEF,MAAMnJ,SAAS,MAAMd,SAAS6L,SAAS,CAAChC,QAAQC;gCAChD,IAAIhJ,OAAOkC,MAAM,GAAG,GAAG;oCACrB,MAAMmH,aAAa2B,YAAY,CAC7BR,SAASK,OAAO,EAChBL,SAASM,EAAE,EACX9K,QACAmJ;gCAEJ;gCACA,OAAOnJ;4BACT,OAAO;gCACL,MAAM;gCACN,OAAO;oCACL8E,MAAM;oCACN+F,SAASL,SAASK,OAAO;oCACzBC,IAAIN,SAASM,EAAE;gCACjB;4BACF;wBACF,OAAO;4BACL,aAAa;4BACb,MAAM,EAAED,OAAO,EAAEC,EAAE,EAAE,GAAG,MAAMzB,aAAa6B,mBAAmB,CAC5Df,mBACAnB,SACAG;4BAEF,MAAME,aAAa8B,WAAW,CAAChB,mBAAmBU,SAASC;4BAE3D,OAAO;gCACLhG,MAAM;gCACN+F;gCACAC;4BACF;wBACF;oBACF;gBACF;YACF;YAEA,OAAO5L,SAAS6L,SAAS,CAAChC,QAAQC;QACpC;QAGF/J,OAAOqG,IAAI,CAKR,iCAAiC,OAAOnF;YACzC,MAAM,EAAE0K,OAAO,EAAEC,EAAE,EAAE,GAAG3K,QAAQqF,IAAI;YACpC,MAAM6D,eAAe,IAAIxL;YAEzB,IAAI,CAACwL,aAAaY,YAAY,IAAI;gBAChC,OAAO;oBAAES,UAAU;oBAAMC,UAAU;gBAAM;YAC3C;YAEA,OAAOtB,aAAauB,aAAa,CAACC,SAASC;QAC7C;QAEA7L,OAAOqG,IAAI,CAOR,iCAAiC,OAAOnF;YACzC,MAAM,EAAE0K,OAAO,EAAEC,EAAE,EAAEM,MAAM,EAAEjC,SAAS,EAAE,GAAGhJ,QAAQqF,IAAI;YACvD,MAAM6D,eAAe,IAAIxL;YAEzB,IAAI,CAACwL,aAAaY,YAAY,IAAI;gBAChC,MAAM,IAAIxM,oBAAoBH,GAAG;YACnC;YAEA,MAAM+L,aAAa4B,aAAa,CAACJ,SAASC,IAAIM,QAAQjC;YACtD,OAAO;gBAAER,SAAS;YAAK;QACzB;QAEA1J,OAAOqG,IAAI,CAIR,4BAA4B,OAAOnF;YACpC,MAAM,EAAEkL,SAAS,EAAE,GAAGlL,QAAQqF,IAAI;YAClC,OAAO,MAAMtG,SAASoM,QAAQ,CAACD;QACjC;QAEApM,OAAOqG,IAAI,CAAC,yCAAyC,OAAOiG;YAC1D,OAAO,MAAMrM,SAASsM,qBAAqB;QAC7C;QAEAvM,OAAOqG,IAAI,CAKR,8BAA8B,OAAOnF;YACtC,MAAM,EAAE+C,SAAS,EAAEuI,cAAcC,aAAa,EAAE,GAAGvL,QAAQqF,IAAI;YAC/D,IAAI,AAACtC,CAAAA,aAAa,EAAE,AAAD,EAAGhB,MAAM,KAAK,GAAG;gBAClC,MAAM,IAAIzE,oBAAoBH,GAAG;YACnC,OAAO,IAAI,AAACoO,CAAAA,iBAAiB,EAAE,AAAD,EAAGxJ,MAAM,KAAK,GAAG;gBAC7C,MAAM,IAAIzE,oBAAoBH,GAAG;YACnC;YAEA,UAAU;YACV4F,UAAUS,IAAI,CAAC,CAACC,GAAGC,IAAOD,IAAIC,IAAI,CAAC,IAAID,IAAIC,IAAI,IAAI;YACnD,MAAM4H,eAAetN,YAAYwN,OAAO,CAACzH,MAAM,CAAC,CAAC0H,KAAOF,cAAc7I,QAAQ,CAAC+I;YAE/E,MAAM3J,eAAeiB,UAAUf,OAAO,CAAC,CAAC/B;gBACtC,OAAOqL,aAAa3I,GAAG,CAAC,CAAC+I,cAAgB;wBAACzL;wBAAUyL;qBAAY;YAClE;YAEA,MAAMC,WAAW,MAAMxM,QAAQmE,GAAG,CAChCxB,aAAaa,GAAG,CAAC,OAAO,CAAC1C,UAAUyL,YAAY;gBAC7C,MAAM,EAAEE,OAAO,EAAEC,QAAQ,EAAE7E,QAAQ,EAAE,GAAG,MAAMhK,OAAOyC,MAAM,CAACqM,kBAAkB,CAC5E7L,UACAyL;gBAEF,OAAO;oBACLzL;oBACAyL;oBACAE;oBACAC;oBACA7E;gBACF;YACF;YAEF,OAAO;gBAAE2E;YAAS;QACpB;QAEA7M,OAAOqG,IAAI,CASR,6BAA6B,OAAOnF;YACrC,MAAM,EAAEwL,OAAO,EAAE,GAAGxL,QAAQqF,IAAI;YAChC,IAAImG,QAAQzJ,MAAM,KAAK,GAAG;gBACxB,MAAM,IAAIzE,oBAAoBH,GAAG;YACnC;YAEA,6BAA6B;YAC7B,MAAM8H,OAAOuG,QAAQxJ,OAAO,CAAC,CAAC,EAAE0J,WAAW,EAAE;gBAC3C,MAAMK,WAAWpO,gBAAgBmC,GAAG,CAAC4L;gBACrC,OAAOK,SAASC,mBAAmB,MAAM,EAAE;YAC7C;YAEA,wCAAwC;YACxC,MAAM5O,iBAAiB6O,cAAc,CAAC;mBAAI,IAAIhC,IAAIhF;aAAM;YAExD,oBAAoB;YACpB,MAAMpF,SAAS,MAAMV,QAAQmE,GAAG,CAC9BkI,QAAQ7I,GAAG,CAAC,OAAO,EAAE1C,QAAQ,EAAEyL,WAAW,EAAE3E,MAAM,EAAEmF,SAAS,EAAE;gBAC7D,IAAI;oBACF,OAAO,MAAMlP,OAAOyC,MAAM,CAAC0M,gBAAgB,CACzCT,aACA;wBACEzL;wBACA8G;oBACF,GAIA;wBACEmF;oBACF;gBAEJ,EAAE,OAAOE,GAAG;oBACV,IAAI7O,cAAc6O,MAAMA,EAAEC,UAAU,KAAK,KAAK;wBAC5C,OAAO;oBACT,OAAO;wBACLC,QAAQlE,KAAK,CAACgE;wBACd,MAAMA;oBACR;gBACF;YACF;YAGF,IAAIvM,OAAOkE,MAAM,CAAC9F,aAAa8D,MAAM,KAAK,GAAG;gBAC3C,MAAM,IAAIvE,4BAA4BL,GAAG;YAC3C;YACA,OAAO0C;QACT;QAEAf,OAAOqG,IAAI,CAQR,4BAA4B,OAAOnF;YACpC,MAAM,EAAEuM,MAAM,EAAE,GAAGvM,QAAQqF,IAAI;YAE/B,IAAI;gBACF,MAAM,EAAEqG,WAAW,EAAE,GAAGc,iBAAiB,GAAGD;gBAC5C,MAAME,eAAe,MAAMzP,OAAOyC,MAAM,CAACiN,cAAc,CACrDhB,aACAc;gBAGF,OAAO;oBAAEC;gBAAa;YACxB,EAAE,OAAOL,GAAG;gBACVE,QAAQlE,KAAK,CAACgE;gBACd,MAAMA;YACR;QACF;QAEAtN,OAAOqG,IAAI,CAAC,gBAAgB,OAAOnF;YACjC,MAAM,EAAE2M,QAAQ,EAAEC,QAAQ,EAAEC,MAAM,EAAEC,cAAc,EAAE,GAAG9M,QAAQqF,IAAI;YAOnE,OAAOvH,eAAeiP,WAAW,CAACJ,UAAUC,UAAUC,QAAQC;QAChE;QAEAhO,OAAOqG,IAAI,CAAC,uBAAuB,OAAOnF;YACxC,MAAM,EAAEgN,EAAE,EAAEC,QAAQ,EAAE,GAAGjN,QAAQqF,IAAI;YAKrC,OAAOvH,eAAeoP,cAAc,CAACF,IAAIC;QAC3C;QAEAnO,OAAOqG,IAAI,CAAC,iCAAiC,OAAOnF;YAClD,MAAM,EAAEmN,IAAI,EAAE,GAAGnN,QAAQqF,IAAI;YAE7B,OAAOvH,eAAesP,gBAAgB,CAACD;QACzC;QAEArO,OAAOgB,GAAG,CAAC,wBAAwB;YACjC,OAAO1C,iBAAiBiQ,aAAa;QACvC;QAEAvO,OAAOgB,GAAG,CAAC,oBAAoB,OAAOwN,UAAU1F;YAC9C,MAAM,EAAEnH,QAAQ,EAAE8M,MAAM,EAAE,GAAG,MAAMnQ,iBAAiBoQ,aAAa;YACjE5F,MACG6F,MAAM,CACL,gBACA,qEAEDA,MAAM,CAAC,uBAAuB,CAAC,sBAAsB,EAAEhN,SAAS,CAAC,CAAC,EAClEiN,IAAI,CAACH;QACV;QAEAzO,OAAOqG,IAAI,CAAC,oBAAoB,OAAOnF;YACrC,MAAM2N,OAAO,MAAM3N,QAAQ4N,IAAI;YAC/B,IAAI,CAACD,MAAM;gBACT,MAAM,IAAIrQ,oBAAoBH,GAAG;YACnC;YACA,MAAMoQ,SAAS,MAAMI,KAAKE,QAAQ;YAClC,OAAOzQ,iBAAiB0Q,eAAe,CAACP;QAC1C;QAEAzO,OAAOqG,IAAI,CAOR,oBAAoB,OAAOnF;YAC5B,MAAM5C,iBAAiB2Q,WAAW,CAAC/N,QAAQqF,IAAI;YAC/C,OAAO;gBAAEmD,SAAS;YAAK;QACzB;QAEA1J,OAAOqG,IAAI,CAKR,oBAAoB,OAAOnF;YAC5B,MAAM5C,iBAAiB4Q,WAAW,CAAChO,QAAQqF,IAAI;YAC/C,OAAO;gBAAEmD,SAAS;YAAK;QACzB;QAEA1J,OAAOqG,IAAI,CAIR,oBAAoB,OAAOnF;YAC5B,MAAM5C,iBAAiB6Q,WAAW,CAACjO,QAAQqF,IAAI,CAAC6I,GAAG;YACnD,OAAO;gBAAE1F,SAAS;YAAK;QACzB;QAEA1J,OAAOqG,IAAI,CAA+B,wBAAwB,OAAOnF;YACvE,OAAO5C,iBAAiB+Q,UAAU,CAACnO,QAAQqF,IAAI,CAACJ,IAAI;QACtD;QAEA,YAAY;QACZnG,OAAOgB,GAAG,CAAC,qBAAqB;YAC9B,OAAO;gBAAE+G,QAAQ7J,OAAOoR,SAAS,KAAK;YAAK;QAC7C;QAEAtP,OAAOgB,GAAG,CAAC,kCAAkC;YAC3C,IAAI,CAAC9C,OAAOoR,SAAS,EACnB,MAAM,IAAIzH,MAAM;YAClB,MAAM0H,cAAcrR,OAAOoR,SAAS,CAACE,mBAAmB;YACxD,OAAO;gBAAED;YAAY;QACvB;QAEAvP,OAAOgB,GAAG,CAWP,2BAA2B,OAAOE;YACnC,IAAI,CAAChD,OAAOoR,SAAS,EACnB,MAAM,IAAIzH,MAAM;YAClB,MAAM4H,UAAUvR,OAAOoR,SAAS,CAACG,OAAO;YACxC,MAAM,EAAEC,KAAK,EAAEtH,KAAK,EAAEC,MAAM,EAAEsH,KAAK,EAAElG,MAAM,EAAEmG,YAAY,EAAEC,YAAY,EAAEC,aAAa,EAAE,GACtF5O,QAAQI,KAAK;YACf,OAAOmO,QAAQM,gBAAgB,CAAC;gBAC9BL,OAAOA,QAAQM,OAAOC,QAAQ,CAACP,OAAO,MAAM1K;gBAC5CoD;gBACAC;gBACAsH;gBACAlG,QAAQA,SAASA,OAAO1G,KAAK,CAAC,OAAOiC;gBACrC4K,cAAcA,gBAAgB5K;gBAC9B6K,cAAcA,eAAe,IAAIK,KAAKL,gBAAgB7K;gBACtD8K,eAAeA,gBAAgB,IAAII,KAAKJ,iBAAiB9K;YAC3D;QACF;QAEAhF,OAAOgB,GAAG,CAEP,+BAA+B,OAAOE;YACvC,IAAI,CAAChD,OAAOoR,SAAS,EACnB,MAAM,IAAIzH,MAAM;YAClB,MAAM4H,UAAUvR,OAAOoR,SAAS,CAACG,OAAO;YACxC,MAAMU,cAAc,MAAMV,QAAQW,cAAc,CAAC;gBAC/CC,eAAenP,QAAQoP,MAAM,CAAClO,EAAE;YAClC;YACA,IAAI,CAAC+N,aAAa;gBAChB,MAAM,IAAItI,MAAM,CAAC,wBAAwB,EAAE3G,QAAQoP,MAAM,CAAClO,EAAE,EAAE;YAChE;YACA,OAAO+N;QACT;QAEAnQ,OAAOqG,IAAI,CAER,sCAAsC,OAAOnF;YAC9C,IAAI,CAAChD,OAAOoR,SAAS,EACnB,MAAM,IAAIzH,MAAM;YAClB,MAAM4H,UAAUvR,OAAOoR,SAAS,CAACG,OAAO;YACxC,OAAOA,QAAQc,iBAAiB,CAAC;gBAC/BF,eAAenP,QAAQoP,MAAM,CAAClO,EAAE;YAClC;QACF;QAEApC,OAAOqG,IAAI,CAER,qCAAqC,OAAOnF;YAC7C,IAAI,CAAChD,OAAOoR,SAAS,EACnB,MAAM,IAAIzH,MAAM;YAClB,MAAM4H,UAAUvR,OAAOoR,SAAS,CAACG,OAAO;YACxC,OAAOA,QAAQe,gBAAgB,CAAC;gBAC9BH,eAAenP,QAAQoP,MAAM,CAAClO,EAAE;YAClC;QACF;QAEApC,OAAOqG,IAAI,CAER,sCAAsC,OAAOnF;YAC9C,IAAI,CAAChD,OAAOoR,SAAS,EACnB,MAAM,IAAIzH,MAAM;YAClB,MAAM4H,UAAUvR,OAAOoR,SAAS,CAACG,OAAO;YACxC,OAAOA,QAAQgB,iBAAiB,CAAC;gBAC/BJ,eAAenP,QAAQoP,MAAM,CAAClO,EAAE;YAClC;QACF;QAEApC,OAAOgB,GAAG,CAOP,qCAAqC,OAAOE;YAC7C,IAAI,CAAChD,OAAOoR,SAAS,EACnB,MAAM,IAAIzH,MAAM;YAClB,MAAM4H,UAAUvR,OAAOoR,SAAS,CAACG,OAAO;YACxC,MAAM,EAAEC,KAAK,EAAEtH,KAAK,EAAEC,MAAM,EAAE,GAAGnH,QAAQI,KAAK;YAC9C,OAAOmO,QAAQiB,gBAAgB,CAAC;gBAC9BL,eAAenP,QAAQoP,MAAM,CAAClO,EAAE;gBAChCsN,OAAOA,QAAQM,OAAOC,QAAQ,CAACP,OAAO,MAAM1K;gBAC5CoD;gBACAC;YACF;QACF;QAEA;;;OAGC,GACDrI,OAAOgB,GAAG,CAAC,sBAAsB,OAAOE;YACtC,MAAMyP,UAAUzP,QAAQlB,MAAM,CAACA,MAAM,CAAC2Q,OAAO;YAC7C,MAAMC,OAAOD,WAAW,OAAOA,YAAY,WAAW,AAACA,QAAwBC,IAAI,GAAG;YAEtF,OAAO;gBACLC,IAAI;gBACJC,SAASC,QAAQC,GAAG,GAAGjO,KAAK,CAAC,KAAKkO,GAAG,MAAM;gBAC3CL;gBACAM,WAAW,IAAIhB,OAAOiB,WAAW;YACnC;QACF;QAEA;;OAEC,GACDnR,OAAOqG,IAAI,CAOR,gCAAgC,OAAOnF,SAAS4H;YACjD,MAAM,EAAEtH,MAAM,EAAE4P,QAAQ,CAAC,EAAEC,SAAS,EAAEC,WAAW,SAAS,EAAE,GAAGpQ,QAAQqF,IAAI;YAE3E,gBAAgB;YAChB,MAAMqE,WAAW0G,aAAa,YAAYpT,OAAO0M,QAAQ,CAAC2G,OAAO,GAAGrT,OAAO0M,QAAQ,CAAC4G,IAAI;YAExF,eAAe;YACf,MAAMtD,KAAK9P,mBAAmBwM;YAE9B,IAAI;gBACF,sBAAsB;gBACtB,MAAM6G,YAAY,IAAI1S,iBAAiBmP,IAAIA,IAAIoD,UAAU/S;gBAEzD,kBAAkB;gBAClB,MAAM4P,WAAW,MAAMsD,UAAUC,aAAa,CAAC;oBAC7C;wBACElQ;wBACA4P;wBACAC,WAAWA,aAAa,CAAC;oBAC3B;iBACD;gBAED,OAAO;oBACL3H,SAAS;oBACTlI;oBACA4P,OAAOjD,SAASlL,MAAM;oBACtBkL;oBACAmD;gBACF;YACF,EAAE,OAAOhI,OAAO;gBACdR,MAAMW,MAAM,CAAC;gBACb,OAAO;oBACLC,SAAS;oBACTJ,OAAOA,iBAAiBzB,QAAQyB,MAAMC,OAAO,GAAGC,OAAOF;gBACzD;YACF,SAAU;gBACR,MAAM4E,GAAGyD,OAAO;YAClB;QACF;QAEA;;OAEC,GACD3R,OAAOqG,IAAI,CAOR,+BAA+B,OAAOnF,SAAS4H;YAChD,MAAM,EAAEtH,MAAM,EAAEoQ,QAAQ,EAAElC,QAAQ,EAAE,EAAEmC,KAAK,EAAE,GAAG3Q,QAAQqF,IAAI;YAE5D,qBAAqB;YACrB,MAAMuL,kBAAkB5T,OAAO0M,QAAQ,CAAC2G,OAAO;YAE/C,eAAe;YACf,MAAMQ,YAAY3T,mBAAmB0T;YAErC,IAAI;gBACF,kBAAkB;gBAClB,MAAME,WAAW,IAAIlT,aAAaiT,WAAWxT;gBAE7C,MAAMsQ,OAAO,MAAMmD,SAASC,OAAO,CAACzQ,QAAQ;oBAC1CoQ;oBACAlC;oBACAmC;gBACF;gBAEA,OAAO;oBACLnI,SAAS;oBACTlI;oBACAoQ;oBACAR,OAAOvC,KAAK5L,MAAM;oBAClB4L;gBACF;YACF,EAAE,OAAOvF,OAAO;gBACdR,MAAMW,MAAM,CAAC;gBACb,OAAO;oBACLC,SAAS;oBACTJ,OAAOA,iBAAiBzB,QAAQyB,MAAMC,OAAO,GAAGC,OAAOF;gBACzD;YACF,SAAU;gBACR,MAAMyI,UAAUJ,OAAO;YACzB;QACF;QAEA;;;OAGC,GACD3R,OAAOqG,IAAI,CAQR,6BAA6B,OAAOnF,SAAS4H;YAC9C,MAAM,EACJtH,MAAM,EACNoQ,WAAW,QAAQ,EACnBlC,QAAQ,EAAE,EACVwC,mBAAmB,IAAI,EACvBC,WAAW,CAAC,EACb,GAAGjR,QAAQqF,IAAI;YAEhB,6CAA6C;YAC7C,MAAM6L,WAAWjU,GAAGkU,KAAK,CAAC;YAE1B,sBAAsB;YACtB,MAAMN,YAAY3T,mBAAmBF,OAAO0M,QAAQ,CAAC2G,OAAO;YAE5D,IAAI;gBACF,sBAAsB;gBACtB,MAAME,YAAY,IAAI1S,iBAAiBqT,UAAUL,WAAW,WAAWxT;gBAEvE,qCAAqC;gBACrC,MAAM+T,UAAU,MAAMb,UAAUc,gBAAgB,CAAC/Q,QAAQ;oBACvDoQ;oBACAlC;oBACAwC;oBACAC;gBACF;gBAEA,OAAO;oBACLzI,SAAS;oBACTlI;oBACAoQ;oBACAR,OAAOkB,QAAQrP,MAAM;oBACrBuP,UAAUF;gBACZ;YACF,EAAE,OAAOhJ,OAAO;gBACdR,MAAMW,MAAM,CAAC;gBACb,OAAO;oBACLC,SAAS;oBACTJ,OAAOA,iBAAiBzB,QAAQyB,MAAMC,OAAO,GAAGC,OAAOF;gBACzD;YACF,SAAU;gBACR,uCAAuC;gBACvC,MAAMyI,UAAUJ,OAAO;YACzB;QACF;QAEA;;;OAGC,GACD3R,OAAOqG,IAAI,CAIR,6BAA6B,OAAOnF,SAAS4H;YAC9C,MAAM,EAAEvE,QAAQ,EAAE,GAAGrD,QAAQqF,IAAI;YAEjC,gBAAgB;YAChB,MAAMwL,YAAY3T,mBAAmBF,OAAO0M,QAAQ,CAAC2G,OAAO;YAE5D,IAAI;gBACF,mBAAmB;gBACnB,MAAMkB,iBACJlO,YAAYA,SAAStB,MAAM,GAAG,IAAIsB,WAAWhG,cAAcyD,SAAS;gBAEtE,wCAAwC;gBACxC,MAAM0Q,aAAaD,eAAe5O,GAAG,CAAC,CAAC1C;oBACrC,MAAMK,SAASjD,cAAcyC,GAAG,CAACG;oBACjC,OAAOK,OAAOmF,KAAK;gBACrB;gBAEA,mDAAmD;gBACnD,iCAAiC;gBACjC,MAAMoL,UAAUY,GAAG,CACjB,CAAC,eAAe,EAAED,WAAW7O,GAAG,CAAC,CAACyG,IAAM,CAAC,CAAC,EAAEA,EAAE,CAAC,CAAC,EAAE/G,IAAI,CAAC,MAAM,yBAAyB,CAAC;gBAGzF,OAAO;oBACLmG,SAAS;oBACTkJ,SAASF;oBACTtB,OAAOsB,WAAWzP,MAAM;gBAC1B;YACF,EAAE,OAAOqG,OAAO;gBACdR,MAAMW,MAAM,CAAC;gBACb,OAAO;oBACLC,SAAS;oBACTJ,OAAOA,iBAAiBzB,QAAQyB,MAAMC,OAAO,GAAGC,OAAOF;gBACzD;YACF,SAAU;gBACR,MAAMyI,UAAUJ,OAAO;YACzB;QACF;QAEA,UAAU;QACV3R,OAAOgB,GAAG,CAAC,sBAAsB;YAC/B,OAAOxB;QACT;QAEAQ,OAAOgB,GAAG,CAAC,iBAAiB;YAC1B,OAAOzB;QACT;QAEAS,OAAOqG,IAAI,CAAiC,wBAAwB,OAAOnF;YACzE,MAAM,EAAE2R,QAAQ,EAAE,GAAG3R,QAAQqF,IAAI;YACjC,OAAO5G,YAAYkT;QACrB;QAEA7S,OAAOqG,IAAI,CAAiC,wBAAwB,OAAOnF;YACzE,MAAM,EAAE2R,QAAQ,EAAE,GAAG3R,QAAQqF,IAAI;YACjC,OAAOlH,YAAYwT;QACrB;QAEA7S,OAAOqG,IAAI,CAAiC,uBAAuB,OAAOnF;YACxE,MAAM,EAAE2R,QAAQ,EAAE,GAAG3R,QAAQqF,IAAI;YACjC7G,eAAemT;YACf,OAAO;gBAAEnJ,SAAS;YAAK;QACzB;QAEA,iBAAiB;QACjB1J,OAAOgB,GAAG,CAAC,oBAAoB;YAC7B,OAAOvB;QACT;QAEAO,OAAOqG,IAAI,CAAkC,uBAAuB,OAAOnF;YACzE,MAAM,EAAE4R,SAAS,EAAE,GAAG5R,QAAQqF,IAAI;YAClC,OAAO3G,WAAWkT;QACpB;QAEA9S,OAAOqG,IAAI,CAAkC,uBAAuB,OAAOnF;YACzE,MAAM,EAAE4R,SAAS,EAAE,GAAG5R,QAAQqF,IAAI;YAClC,OAAOjH,WAAWwT;QACpB;QAEA,kBAAkB;QAClB,MAAMC,aAAa/U,KAAKsC,OAAO,CAAC,YAAY0S,OAAO,EAAE;QAErD,6CAA6C;QAC7ChT,OAAOD,QAAQ,CAAC,MAAM,MAAM,CAAC,oBAAoB;YAC/CkT,MAAMF;YACNG,QAAQ;YACRC,eAAe;YACfC,UAAU;QACZ;QAEA,8CAA8C;QAC9CpT,OAAOgB,GAAG,CAAC,KAAK,OAAOwN,UAAU1F;YAC/BA,MAAMuK,OAAO,CAAC;gBAAE,gBAAgB;YAAY,GAAGzE,IAAI,CACjD9Q,GACGwV,YAAY,CAACtV,KAAKsC,OAAO,CAACyS,YAAY,eACtCQ,QAAQ,GACR3Q,OAAO,CAAC,mBAAmB1E,OAAO+C,MAAM,CAACuS,WAAW,IAAI;QAE/D;IACF,GACA;QAAEN,QAAQ;IAAa;AAE3B"}
1149
+ //# 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 {\n  editContent,\n  editSchema,\n  getCddTree,\n  getDashboard,\n  listSchemas,\n  openSourceFile,\n  readContent,\n  readSchema,\n} 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/dashboard\", async () => {\n        return getDashboard();\n      });\n\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      // CDD Schema API\n      server.get(\"/api/cdd/schemas\", async () => {\n        return listSchemas();\n      });\n\n      server.post<{ Body: { schemaKey: string } }>(\"/api/cdd/readSchema\", async (request) => {\n        const { schemaKey } = request.body;\n        return readSchema(schemaKey);\n      });\n\n      server.post<{ Body: { schemaKey: string } }>(\"/api/cdd/editSchema\", async (request) => {\n        const { schemaKey } = request.body;\n        return editSchema(schemaKey);\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","editSchema","getCddTree","getDashboard","listSchemas","openSourceFile","readContent","readSchema","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","schemaKey","uiDistPath","dirname","root","prefix","decorateReply","wildcard","headers","readFileSync","toString","projectName"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "sonamu",
3
- "version": "0.8.21",
3
+ "version": "0.8.22",
4
4
  "description": "Sonamu — TypeScript Fullstack API Framework",
5
5
  "keywords": [
6
6
  "typescript",
@@ -123,10 +123,10 @@
123
123
  "tsicli": "^1.0.5",
124
124
  "vite": "7.3.0",
125
125
  "vitest": "^4.0.10",
126
- "@sonamu-kit/hmr-hook": "^0.4.1",
126
+ "@sonamu-kit/hmr-runner": "^0.1.1",
127
127
  "@sonamu-kit/tasks": "^0.2.0",
128
- "@sonamu-kit/ts-loader": "^2.1.3",
129
- "@sonamu-kit/hmr-runner": "^0.1.1"
128
+ "@sonamu-kit/hmr-hook": "^0.4.1",
129
+ "@sonamu-kit/ts-loader": "^2.1.3"
130
130
  },
131
131
  "devDependencies": {
132
132
  "@biomejs/biome": "^2.3.13",
package/src/api/config.ts CHANGED
@@ -83,7 +83,7 @@ export type SonamuConfig<TSinkId extends string = string, TFilterId extends stri
83
83
  targets: string[]; // "web", "app" 등
84
84
  };
85
85
 
86
- database?: {
86
+ database: {
87
87
  // 데이터베이스(pg는 pg 모듈, pgnative는 pg-native 모듈의 설치가 필요합니다.)
88
88
  database?: "pg" | "pgnative";
89
89
  // 기본 데이터베이스 이름