sonamu 0.8.9 → 0.8.10
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/migration/migrator.d.ts.map +1 -1
- package/dist/migration/migrator.js +37 -33
- package/dist/ui/api.js +5 -5
- package/package.json +1 -1
- package/src/migration/migrator.ts +42 -41
- package/src/shared/web.shared.ts.txt +5 -0
- package/src/ui/api.ts +5 -5
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"migrator.d.ts","sourceRoot":"","sources":["../../src/migration/migrator.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAIjC,OAAO,EAAM,KAAK,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAMzD,OAAO,KAAK,EAAE,gBAAgB,EAAgB,MAAM,gBAAgB,CAAC;AAMrE,OAAO,KAAK,EAA6B,eAAe,EAAE,MAAM,SAAS,CAAC;AAE1E,MAAM,MAAM,eAAe,GAAG;IAC5B,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,EAAE,CAAC;CACnB,EAAE,CAAC;AAEJ,qBAAa,QAAQ;YACL,iBAAiB;IAqB/B;;;;;;;;OAQG;IACG,SAAS,IAAI,OAAO,CAAC,eAAe,CAAC;IAiG3C;;;;;;;;;OASG;IACG,SAAS,CACb,MAAM,EAAE,OAAO,GAAG,UAAU,EAC5B,OAAO,EAAE,CAAC,MAAM,cAAc,CAAC,EAAE,GAChC,OAAO,CAAC,eAAe,CAAC;
|
|
1
|
+
{"version":3,"file":"migrator.d.ts","sourceRoot":"","sources":["../../src/migration/migrator.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAIjC,OAAO,EAAM,KAAK,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAMzD,OAAO,KAAK,EAAE,gBAAgB,EAAgB,MAAM,gBAAgB,CAAC;AAMrE,OAAO,KAAK,EAA6B,eAAe,EAAE,MAAM,SAAS,CAAC;AAE1E,MAAM,MAAM,eAAe,GAAG;IAC5B,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,EAAE,CAAC;CACnB,EAAE,CAAC;AAEJ,qBAAa,QAAQ;YACL,iBAAiB;IAqB/B;;;;;;;;OAQG;IACG,SAAS,IAAI,OAAO,CAAC,eAAe,CAAC;IAiG3C;;;;;;;;;OASG;IACG,SAAS,CACb,MAAM,EAAE,OAAO,GAAG,UAAU,EAC5B,OAAO,EAAE,CAAC,MAAM,cAAc,CAAC,EAAE,GAChC,OAAO,CAAC,eAAe,CAAC;IAmE3B;;;;;;OAMG;IACH,iBAAiB,CAAC,KAAK,EAAE,eAAe,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,MAAM,EAAE;;;;IAWtE;;;;;;;OAOG;IACG,QAAQ,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;IAuBpD,OAAO,CAAC,UAAU;IAalB;;;;;;OAMG;IACG,qBAAqB,IAAI,OAAO,CAAC,MAAM,CAAC;IAuBxC,iBAAiB,CAAC,SAAS,EAAE,IAAI,GAAG,OAAO,CAAC,gBAAgB,EAAE,CAAC;IAgErE;;;;;;OAMG;IACG,aAAa,IAAI,OAAO,CAAC,eAAe,CAAC;CAkEhD"}
|
|
@@ -95,9 +95,11 @@ export class Migrator {
|
|
|
95
95
|
return [];
|
|
96
96
|
}
|
|
97
97
|
const compareDBconn = createKnexInstance(Sonamu.dbConfig[status0conn.connKey]);
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
98
|
+
try {
|
|
99
|
+
return await this.compareMigrations(compareDBconn);
|
|
100
|
+
} finally{
|
|
101
|
+
await compareDBconn.destroy();
|
|
102
|
+
}
|
|
101
103
|
})();
|
|
102
104
|
Naite.t("migrator:getStatus:preparedCodes", preparedCodes);
|
|
103
105
|
return {
|
|
@@ -129,35 +131,37 @@ export class Migrator {
|
|
|
129
131
|
connKey: config.connKey,
|
|
130
132
|
knex: createKnexInstance(config.options)
|
|
131
133
|
})));
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
return
|
|
158
|
-
}
|
|
159
|
-
|
|
160
|
-
|
|
134
|
+
try {
|
|
135
|
+
// action
|
|
136
|
+
const result = await (async ()=>{
|
|
137
|
+
switch(action){
|
|
138
|
+
case "apply":
|
|
139
|
+
return Promise.all(conns.map(async ({ connKey, knex })=>{
|
|
140
|
+
const [batchNo, applied] = await knex.migrate.latest();
|
|
141
|
+
return {
|
|
142
|
+
connKey,
|
|
143
|
+
batchNo,
|
|
144
|
+
applied
|
|
145
|
+
};
|
|
146
|
+
}));
|
|
147
|
+
case "rollback":
|
|
148
|
+
return Promise.all(conns.map(async ({ connKey, knex })=>{
|
|
149
|
+
const [batchNo, applied] = await knex.migrate.rollback();
|
|
150
|
+
return {
|
|
151
|
+
connKey,
|
|
152
|
+
batchNo,
|
|
153
|
+
applied
|
|
154
|
+
};
|
|
155
|
+
}));
|
|
156
|
+
}
|
|
157
|
+
})();
|
|
158
|
+
Naite.t("migrator:runAction:result", result);
|
|
159
|
+
return result;
|
|
160
|
+
} finally{
|
|
161
|
+
await Promise.all(conns.map(({ knex })=>{
|
|
162
|
+
return knex.destroy();
|
|
163
|
+
}));
|
|
164
|
+
}
|
|
161
165
|
}
|
|
162
166
|
/**
|
|
163
167
|
* 삭제 가능한 마이그레이션 코드 파일을 검증합니다.
|
|
@@ -340,4 +344,4 @@ export class Migrator {
|
|
|
340
344
|
}
|
|
341
345
|
}
|
|
342
346
|
|
|
343
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/migration/migrator.ts"],"sourcesContent":["import assert from \"assert\";\nimport chalk from \"chalk\";\nimport { mkdir, readdir, unlink, writeFile } from \"fs/promises\";\nimport type { Knex } from \"knex\";\nimport path from \"path\";\nimport { group, sum, unique } from \"radashi\";\nimport { Sonamu } from \"../api\";\nimport { DB, type SonamuDBConfig } from \"../database/db\";\nimport { createKnexInstance } from \"../database/knex\";\nimport { SD } from \"../dict/sd\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { ServiceUnavailableException } from \"../exceptions/so-exceptions\";\nimport { Naite } from \"../naite/naite\";\nimport type { GenMigrationCode, MigrationSet } from \"../types/types\";\nimport { isTest } from \"../utils/controller\";\nimport { exists } from \"../utils/fs-utils\";\nimport { generateAlterCode, generateCreateCode } from \"./code-generation\";\nimport { getMigrationSetFromEntity } from \"./migration-set\";\nimport { PostgreSQLSchemaReader } from \"./postgresql-schema-reader\";\nimport type { ConnString, MigrationCode, MigrationStatus } from \"./types\";\n\nexport type MigrationResult = {\n  connKey: string;\n  batchNo: number;\n  applied: string[];\n}[];\n\nexport class Migrator {\n  private async getMigrationCodes(): Promise<MigrationCode[]> {\n    const srcMigrationsDir = path.join(Sonamu.apiRootPath, \"src\", \"migrations\"); // 이건 환경에 관계없이 항상 src에서 찾아야 해요.\n\n    if (!(await exists(srcMigrationsDir))) {\n      await mkdir(srcMigrationsDir, {\n        recursive: true,\n      });\n    }\n\n    const codes = (await readdir(srcMigrationsDir))\n      .filter((f) => f.endsWith(\".ts\"))\n      .map((f) => ({\n        name: f.replace(\".ts\", \"\"),\n        path: path.join(srcMigrationsDir, f),\n      }))\n      .sort((a, b) => (a.name < b.name ? 1 : -1)); // 이름 내림차순 정렬(최신순)\n\n    Naite.t(\"migrator:getMigrationCodes:results\", codes);\n    return codes;\n  }\n\n  /**\n   * 타겟별 마이그레이션 상태와 코드 생성/준비 상태를 구해옵니다.\n   * 실제로 DB에 접근도 하고 마이그레이션 코드 파일도 확인하고,\n   * 필요하다면 적용할 수 있는 코드를 생성까지 해옵니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @returns\n   */\n  async getStatus(): Promise<MigrationStatus> {\n    const codes = await this.getMigrationCodes();\n    Naite.t(\"migrator:getStatus:codes\", codes);\n\n    const connKeys = Object.keys(Sonamu.dbConfig).filter(\n      (key) => key.endsWith(\"_slave\") === false,\n    ) as (keyof typeof Sonamu.dbConfig)[];\n\n    let migrationStatusError: string | undefined;\n\n    const statuses = await Promise.all(\n      connKeys.map(async (connKey) => {\n        const knexOptions = Sonamu.dbConfig[connKey];\n        const tConn = createKnexInstance(knexOptions);\n\n        try {\n          const status = await (async () => {\n            try {\n              return await tConn.migrate.status();\n            } catch (err) {\n              console.warn(\n                chalk.yellow(\n                  `${connKey}의 마이그레이션 상태를 가져오는 데에 실패하였습니다. 데이터베이스가 올바르게 구성되지 않은 것 같습니다. 확인하시고 다시 시도해주세요.\\n시도한 연결 설정:\\n${JSON.stringify(knexOptions.connection, null, 2)}\\n발생한 에러:\\n${err}\\n`,\n                ),\n              );\n              migrationStatusError = err instanceof Error ? err.message : String(err);\n              return \"error\";\n            }\n          })();\n          const pending: string[] = await (async () => {\n            try {\n              const [, fdList] = await tConn.migrate.list();\n              return fdList.map((fd: { file: string }) => fd.file.replace(\".ts\", \"\"));\n            } catch (err) {\n              migrationStatusError = err instanceof Error ? err.message : String(err);\n              return [];\n            }\n          })();\n          const currentVersion = await (async () => {\n            try {\n              return await tConn.migrate.currentVersion();\n            } catch (_err) {\n              migrationStatusError = _err instanceof Error ? _err.message : String(_err);\n              return \"error\";\n            }\n          })();\n          Naite.t(\"migrator:getStatus:status\", status);\n\n          const connection = knexOptions.connection as Knex.PgConnectionConfig;\n\n          return {\n            name: connKey.replace(\"_master\", \"\"),\n            connKey,\n            connString: `pg://${connection.user ?? \"\"}@${connection.host}:${\n              connection.port\n            }/${connection.database}` as ConnString,\n            currentVersion,\n            status: status as number | \"error\",\n            pending,\n          };\n        } finally {\n          await tConn.destroy();\n        }\n      }),\n    );\n\n    Naite.t(\"migrator:getStatus:conns\", statuses);\n\n    const preparedCodes: GenMigrationCode[] = await (async () => {\n      const status0conn = statuses.find((status) => status.status === 0);\n      if (status0conn === undefined) {\n        console.warn(\n          chalk.yellow(\n            `While trying to prepare migration codes, we found that there is no database to compare migrations. We need at least one database where every migration is applied(status === 0). You might want to apply your existing migrations to one of the databases.`,\n          ),\n        );\n        return [];\n      }\n\n      const compareDBconn = createKnexInstance(Sonamu.dbConfig[status0conn.connKey]);\n      const genCodes = await this.compareMigrations(compareDBconn);\n\n      await compareDBconn.destroy();\n\n      return genCodes;\n    })();\n\n    Naite.t(\"migrator:getStatus:preparedCodes\", preparedCodes);\n\n    return {\n      conns: statuses,\n      codes,\n      preparedCodes,\n      error: migrationStatusError,\n    };\n  }\n\n  /**\n   * 마이그레이션을 적용하거나 롤백합니다.\n   * Sonamu UI에서 마이그레이션 작업을 수행할 때 사용됩니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @param action 작업 유형 (apply/rollback)\n   * @param targets 작업 대상 DB 설정 키 (keyof SonamuDBConfig)\n   * @returns 작업 결과\n   */\n  async runAction(\n    action: \"apply\" | \"rollback\",\n    targets: (keyof SonamuDBConfig)[],\n  ): Promise<MigrationResult> {\n    Naite.t(\"migrator:runAction:action\", action);\n    Naite.t(\"migrator:runAction:targets\", targets);\n\n    // get uniq knex configs\n    const configs = unique(\n      targets\n        .map((target) => ({\n          connKey: target,\n          options: Sonamu.dbConfig[target as keyof typeof Sonamu.dbConfig],\n        }))\n        .filter((c) => c.options !== undefined),\n      ({ options }) =>\n        `${(options.connection as Knex.PgConnectionConfig).host}:${\n          (options.connection as Knex.PgConnectionConfig).port ?? 5432\n        }/${(options.connection as Knex.PgConnectionConfig).database}`,\n    );\n\n    // get connections\n    const conns = await Promise.all(\n      configs.map(async (config) => ({\n        connKey: config.connKey,\n        knex: createKnexInstance(config.options),\n      })),\n    );\n\n    // action\n    const result = await (async () => {\n      switch (action) {\n        case \"apply\":\n          return Promise.all(\n            conns.map(async ({ connKey, knex }) => {\n              const [batchNo, applied] = await knex.migrate.latest();\n              return {\n                connKey,\n                batchNo,\n                applied, // 이번 latest 호출로 인해 \"up\"이 적용된 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L560\n              };\n            }),\n          );\n        case \"rollback\":\n          return Promise.all(\n            conns.map(async ({ connKey, knex }) => {\n              const [batchNo, applied] = await knex.migrate.rollback();\n              return {\n                connKey,\n                batchNo,\n                applied, // 이번 rollback 호출로 인해 \"down\"이 적용된(=롤백된) 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L611\n              };\n            }),\n          );\n      }\n    })();\n\n    // destroy\n    await Promise.all(\n      conns.map(({ knex }) => {\n        return knex.destroy();\n      }),\n    );\n\n    Naite.t(\"migrator:runAction:result\", result);\n\n    return result;\n  }\n\n  /**\n   * 삭제 가능한 마이그레이션 코드 파일을 검증합니다.\n   *\n   * @param conns 마이그레이션 상태 배열\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제 가능 여부 및 적용된 마이그레이션 코드 파일 이름\n   */\n  validateDeletable(conns: MigrationStatus[\"conns\"], codeNames: string[]) {\n    const appliedCodes = codeNames.filter((codeName) =>\n      conns.some((conn) => conn.pending.includes(codeName) === false),\n    );\n\n    return {\n      canDelete: appliedCodes.length === 0,\n      appliedCodes,\n    };\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 삭제합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제된 마이그레이션 코드 파일 개수\n   */\n  async delCodes(codeNames: string[]): Promise<number> {\n    const { conns } = await this.getStatus();\n    const { canDelete, appliedCodes } = this.validateDeletable(conns, codeNames);\n    if (!canDelete) {\n      throw new Error(\n        `You cannot delete a migration file if there is already applied. Applied codes: ${appliedCodes.join(\", \")}`,\n      );\n    }\n\n    return sum(\n      await Promise.all(\n        codeNames.map(async (codeName) => {\n          const filePath = `${Sonamu.apiRootPath}/src/migrations/${codeName}.ts`;\n          if (await exists(filePath)) {\n            await unlink(filePath);\n            return 1;\n          }\n          return 0;\n        }),\n      ),\n    );\n  }\n\n  private genDateTag(index: number, baseDate: Date = new Date()): string {\n    const date = new Date(baseDate.getTime() + index * 1000);\n    const pad = (num: number, size: number = 2) => num.toString().padStart(size, \"0\");\n    return (\n      date.getFullYear().toString() +\n      pad(date.getMonth() + 1) +\n      pad(date.getDate()) +\n      pad(date.getHours()) +\n      pad(date.getMinutes()) +\n      pad(date.getSeconds())\n    );\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 생성합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns 생성된 마이그레이션 코드 파일 개수\n   */\n  async generatePreparedCodes(): Promise<number> {\n    const { preparedCodes } = await this.getStatus();\n    Naite.t(\"migrator:generatePreparedCodes:preparedCodes\", preparedCodes);\n    if (preparedCodes.length === 0) {\n      console.log(chalk.green(\"\\n현재 모두 싱크된 상태입니다.\"));\n      return 0;\n    }\n\n    // 실제 코드 생성\n    const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;\n\n    for (const [index, pcode] of preparedCodes.entries()) {\n      if (pcode.formatted) {\n        const dateTag = this.genDateTag(index);\n        const filePath = `${migrationsDir}/${dateTag}_${pcode.title}.ts`;\n        await writeFile(filePath, pcode.formatted);\n        !isTest() && console.log(chalk.green(`MIGRATION CREATED ${filePath}`));\n      }\n    }\n\n    return preparedCodes.length;\n  }\n\n  async compareMigrations(compareDB: Knex): Promise<GenMigrationCode[]> {\n    // Entity 순회하여 싱크\n    const entityIds = EntityManager.getAllIds();\n\n    // 조인테이블 포함하여 Entity에서 MigrationSet 추출\n    const entitySetsWithJoinTable = entityIds\n      .filter((entityId) => EntityManager.get(entityId).props.length > 0)\n      .map((entityId) => getMigrationSetFromEntity(EntityManager.get(entityId)));\n\n    // 조인테이블만 추출\n    const joinTablesWithDup = entitySetsWithJoinTable.flatMap((entitySet) => entitySet.joinTables);\n    // 중복 제거 (중복인 경우 indexes를 병합)\n    const joinTables = Object.values(group(joinTablesWithDup, (jt) => jt.table)).map((tables) => {\n      assert(tables !== undefined, \"tables is undefined\");\n      if (tables.length === 1) {\n        return tables[0];\n      }\n      return {\n        ...tables[0],\n        indexes: unique(\n          tables.flatMap((t) => t.indexes),\n          (index) => [index.type, ...index.columns].join(\"-\"),\n        ),\n      };\n    });\n\n    // 조인테이블 포함하여 MigrationSet 배열\n    const entitySets: MigrationSet[] = [...entitySetsWithJoinTable, ...joinTables];\n\n    const codes: GenMigrationCode[] = (\n      await Promise.all(\n        entitySets.map(async (entitySet) => {\n          const dbSet = await PostgreSQLSchemaReader.getMigrationSetFromDB(\n            compareDB,\n            entitySet.table,\n          );\n          Naite.t(`migrator:compareMigrations:entitySet:${entitySet.table}`, entitySet);\n          Naite.t(`migrator:compareMigrations:dbSet:${entitySet.table}`, dbSet);\n\n          if (dbSet === null) {\n            // 기존 테이블 없음, 새로 테이블 생성\n            return await generateCreateCode(entitySet);\n          } else {\n            // 기존 테이블 존재하는 케이스\n            return await generateAlterCode(entitySet, dbSet, compareDB);\n          }\n        }),\n      )\n    ).flat();\n\n    // normal 타입이 앞으로, foreign이 뒤로\n    codes.sort((codeA, codeB) => {\n      if (codeA.type === \"foreign\" && codeB.type === \"normal\") {\n        return 1;\n      } else if (codeA.type === \"normal\" && codeB.type === \"foreign\") {\n        return -1;\n      } else {\n        return 0;\n      }\n    });\n\n    return codes;\n  }\n\n  /**\n   * Shadow DB 테스트를 진행합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns Shadow DB 테스트 결과\n   */\n  async runShadowTest(): Promise<MigrationResult> {\n    const tdbConn = Sonamu.dbConfig.test.connection as Knex.PgConnectionConfig;\n    const shadowDatabase = `${tdbConn.database}__migration_shadow`;\n\n    // 테스트 상황에서는 트랜잭션을 초기화하고, 새 데이터베이스 커넥션을 가져와야 함\n    if (isTest()) {\n      await DB.clearTestTransaction();\n      // 병렬 테스트 모드에서는 worker DB 연결 유지\n      if (process.env.SONAMU_WORKER_DB !== \"true\") {\n        await DB.destroy();\n      }\n    }\n\n    // 기존 Shadow DB 삭제 후 Shadow DB 생성\n    const tdb = createKnexInstance(Sonamu.dbConfig.test);\n    !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n    await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n    await tdb.raw(`\n      SELECT pg_terminate_backend(pg_stat_activity.pid)\n      FROM pg_stat_activity\n      WHERE datname = '${tdbConn.database}'\n        AND pid <> pg_backend_pid();\n    `);\n    await tdb.raw(`CREATE DATABASE ${shadowDatabase} TEMPLATE ${tdbConn.database}`);\n\n    // Shadow DB에 연결\n    const sdb = createKnexInstance({\n      ...Sonamu.dbConfig.test,\n      connection: {\n        ...tdbConn,\n        database: shadowDatabase,\n        password: tdbConn.password,\n      },\n    });\n\n    // shadow DB 테스트 진행\n    try {\n      const [batchNo, applied] = await sdb.migrate.latest();\n      !isTest() &&\n        console.log(chalk.green(\"Shadow DB 테스트에 성공했습니다!\"), {\n          batchNo,\n          applied,\n        });\n\n      return [\n        {\n          connKey: \"shadow\",\n          batchNo,\n          applied,\n        },\n      ];\n    } catch (e) {\n      console.error(e);\n      throw new ServiceUnavailableException(SD(\"sonamu.error.shadowDbTestFailed\"));\n    } finally {\n      // Shadow DB 연결 종료\n      await sdb.destroy();\n\n      // Shadow DB 삭제\n      !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n      await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n\n      // Test DB 연결 종료\n      await tdb.destroy();\n    }\n  }\n}\n"],"names":["assert","chalk","mkdir","readdir","unlink","writeFile","path","group","sum","unique","Sonamu","DB","createKnexInstance","SD","EntityManager","ServiceUnavailableException","Naite","isTest","exists","generateAlterCode","generateCreateCode","getMigrationSetFromEntity","PostgreSQLSchemaReader","Migrator","getMigrationCodes","srcMigrationsDir","join","apiRootPath","recursive","codes","filter","f","endsWith","map","name","replace","sort","a","b","t","getStatus","connKeys","Object","keys","dbConfig","key","migrationStatusError","statuses","Promise","all","connKey","knexOptions","tConn","status","migrate","err","console","warn","yellow","JSON","stringify","connection","Error","message","String","pending","fdList","list","fd","file","currentVersion","_err","connString","user","host","port","database","destroy","preparedCodes","status0conn","find","undefined","compareDBconn","genCodes","compareMigrations","conns","error","runAction","action","targets","configs","target","options","c","config","knex","result","batchNo","applied","latest","rollback","validateDeletable","codeNames","appliedCodes","codeName","some","conn","includes","canDelete","length","delCodes","filePath","genDateTag","index","baseDate","Date","date","getTime","pad","num","size","toString","padStart","getFullYear","getMonth","getDate","getHours","getMinutes","getSeconds","generatePreparedCodes","log","green","migrationsDir","pcode","entries","formatted","dateTag","title","compareDB","entityIds","getAllIds","entitySetsWithJoinTable","entityId","get","props","joinTablesWithDup","flatMap","entitySet","joinTables","values","jt","table","tables","indexes","type","columns","entitySets","dbSet","getMigrationSetFromDB","flat","codeA","codeB","runShadowTest","tdbConn","test","shadowDatabase","clearTestTransaction","process","env","SONAMU_WORKER_DB","tdb","magenta","raw","sdb","password","e"],"mappings":"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"}
|
|
347
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/migration/migrator.ts"],"sourcesContent":["import assert from \"assert\";\nimport chalk from \"chalk\";\nimport { mkdir, readdir, unlink, writeFile } from \"fs/promises\";\nimport type { Knex } from \"knex\";\nimport path from \"path\";\nimport { group, sum, unique } from \"radashi\";\nimport { Sonamu } from \"../api\";\nimport { DB, type SonamuDBConfig } from \"../database/db\";\nimport { createKnexInstance } from \"../database/knex\";\nimport { SD } from \"../dict/sd\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { ServiceUnavailableException } from \"../exceptions/so-exceptions\";\nimport { Naite } from \"../naite/naite\";\nimport type { GenMigrationCode, MigrationSet } from \"../types/types\";\nimport { isTest } from \"../utils/controller\";\nimport { exists } from \"../utils/fs-utils\";\nimport { generateAlterCode, generateCreateCode } from \"./code-generation\";\nimport { getMigrationSetFromEntity } from \"./migration-set\";\nimport { PostgreSQLSchemaReader } from \"./postgresql-schema-reader\";\nimport type { ConnString, MigrationCode, MigrationStatus } from \"./types\";\n\nexport type MigrationResult = {\n  connKey: string;\n  batchNo: number;\n  applied: string[];\n}[];\n\nexport class Migrator {\n  private async getMigrationCodes(): Promise<MigrationCode[]> {\n    const srcMigrationsDir = path.join(Sonamu.apiRootPath, \"src\", \"migrations\"); // 이건 환경에 관계없이 항상 src에서 찾아야 해요.\n\n    if (!(await exists(srcMigrationsDir))) {\n      await mkdir(srcMigrationsDir, {\n        recursive: true,\n      });\n    }\n\n    const codes = (await readdir(srcMigrationsDir))\n      .filter((f) => f.endsWith(\".ts\"))\n      .map((f) => ({\n        name: f.replace(\".ts\", \"\"),\n        path: path.join(srcMigrationsDir, f),\n      }))\n      .sort((a, b) => (a.name < b.name ? 1 : -1)); // 이름 내림차순 정렬(최신순)\n\n    Naite.t(\"migrator:getMigrationCodes:results\", codes);\n    return codes;\n  }\n\n  /**\n   * 타겟별 마이그레이션 상태와 코드 생성/준비 상태를 구해옵니다.\n   * 실제로 DB에 접근도 하고 마이그레이션 코드 파일도 확인하고,\n   * 필요하다면 적용할 수 있는 코드를 생성까지 해옵니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @returns\n   */\n  async getStatus(): Promise<MigrationStatus> {\n    const codes = await this.getMigrationCodes();\n    Naite.t(\"migrator:getStatus:codes\", codes);\n\n    const connKeys = Object.keys(Sonamu.dbConfig).filter(\n      (key) => key.endsWith(\"_slave\") === false,\n    ) as (keyof typeof Sonamu.dbConfig)[];\n\n    let migrationStatusError: string | undefined;\n\n    const statuses = await Promise.all(\n      connKeys.map(async (connKey) => {\n        const knexOptions = Sonamu.dbConfig[connKey];\n        const tConn = createKnexInstance(knexOptions);\n\n        try {\n          const status = await (async () => {\n            try {\n              return await tConn.migrate.status();\n            } catch (err) {\n              console.warn(\n                chalk.yellow(\n                  `${connKey}의 마이그레이션 상태를 가져오는 데에 실패하였습니다. 데이터베이스가 올바르게 구성되지 않은 것 같습니다. 확인하시고 다시 시도해주세요.\\n시도한 연결 설정:\\n${JSON.stringify(knexOptions.connection, null, 2)}\\n발생한 에러:\\n${err}\\n`,\n                ),\n              );\n              migrationStatusError = err instanceof Error ? err.message : String(err);\n              return \"error\";\n            }\n          })();\n          const pending: string[] = await (async () => {\n            try {\n              const [, fdList] = await tConn.migrate.list();\n              return fdList.map((fd: { file: string }) => fd.file.replace(\".ts\", \"\"));\n            } catch (err) {\n              migrationStatusError = err instanceof Error ? err.message : String(err);\n              return [];\n            }\n          })();\n          const currentVersion = await (async () => {\n            try {\n              return await tConn.migrate.currentVersion();\n            } catch (_err) {\n              migrationStatusError = _err instanceof Error ? _err.message : String(_err);\n              return \"error\";\n            }\n          })();\n          Naite.t(\"migrator:getStatus:status\", status);\n\n          const connection = knexOptions.connection as Knex.PgConnectionConfig;\n\n          return {\n            name: connKey.replace(\"_master\", \"\"),\n            connKey,\n            connString: `pg://${connection.user ?? \"\"}@${connection.host}:${\n              connection.port\n            }/${connection.database}` as ConnString,\n            currentVersion,\n            status: status as number | \"error\",\n            pending,\n          };\n        } finally {\n          await tConn.destroy();\n        }\n      }),\n    );\n\n    Naite.t(\"migrator:getStatus:conns\", statuses);\n\n    const preparedCodes: GenMigrationCode[] = await (async () => {\n      const status0conn = statuses.find((status) => status.status === 0);\n      if (status0conn === undefined) {\n        console.warn(\n          chalk.yellow(\n            `While trying to prepare migration codes, we found that there is no database to compare migrations. We need at least one database where every migration is applied(status === 0). You might want to apply your existing migrations to one of the databases.`,\n          ),\n        );\n        return [];\n      }\n\n      const compareDBconn = createKnexInstance(Sonamu.dbConfig[status0conn.connKey]);\n      try {\n        return await this.compareMigrations(compareDBconn);\n      } finally {\n        await compareDBconn.destroy();\n      }\n    })();\n\n    Naite.t(\"migrator:getStatus:preparedCodes\", preparedCodes);\n\n    return {\n      conns: statuses,\n      codes,\n      preparedCodes,\n      error: migrationStatusError,\n    };\n  }\n\n  /**\n   * 마이그레이션을 적용하거나 롤백합니다.\n   * Sonamu UI에서 마이그레이션 작업을 수행할 때 사용됩니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @param action 작업 유형 (apply/rollback)\n   * @param targets 작업 대상 DB 설정 키 (keyof SonamuDBConfig)\n   * @returns 작업 결과\n   */\n  async runAction(\n    action: \"apply\" | \"rollback\",\n    targets: (keyof SonamuDBConfig)[],\n  ): Promise<MigrationResult> {\n    Naite.t(\"migrator:runAction:action\", action);\n    Naite.t(\"migrator:runAction:targets\", targets);\n\n    // get uniq knex configs\n    const configs = unique(\n      targets\n        .map((target) => ({\n          connKey: target,\n          options: Sonamu.dbConfig[target as keyof typeof Sonamu.dbConfig],\n        }))\n        .filter((c) => c.options !== undefined),\n      ({ options }) =>\n        `${(options.connection as Knex.PgConnectionConfig).host}:${\n          (options.connection as Knex.PgConnectionConfig).port ?? 5432\n        }/${(options.connection as Knex.PgConnectionConfig).database}`,\n    );\n\n    // get connections\n    const conns = await Promise.all(\n      configs.map(async (config) => ({\n        connKey: config.connKey,\n        knex: createKnexInstance(config.options),\n      })),\n    );\n\n    try {\n      // action\n      const result = await (async () => {\n        switch (action) {\n          case \"apply\":\n            return Promise.all(\n              conns.map(async ({ connKey, knex }) => {\n                const [batchNo, applied] = await knex.migrate.latest();\n                return {\n                  connKey,\n                  batchNo,\n                  applied, // 이번 latest 호출로 인해 \"up\"이 적용된 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L560\n                };\n              }),\n            );\n          case \"rollback\":\n            return Promise.all(\n              conns.map(async ({ connKey, knex }) => {\n                const [batchNo, applied] = await knex.migrate.rollback();\n                return {\n                  connKey,\n                  batchNo,\n                  applied, // 이번 rollback 호출로 인해 \"down\"이 적용된(=롤백된) 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L611\n                };\n              }),\n            );\n        }\n      })();\n\n      Naite.t(\"migrator:runAction:result\", result);\n\n      return result;\n    } finally {\n      await Promise.all(\n        conns.map(({ knex }) => {\n          return knex.destroy();\n        }),\n      );\n    }\n  }\n\n  /**\n   * 삭제 가능한 마이그레이션 코드 파일을 검증합니다.\n   *\n   * @param conns 마이그레이션 상태 배열\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제 가능 여부 및 적용된 마이그레이션 코드 파일 이름\n   */\n  validateDeletable(conns: MigrationStatus[\"conns\"], codeNames: string[]) {\n    const appliedCodes = codeNames.filter((codeName) =>\n      conns.some((conn) => conn.pending.includes(codeName) === false),\n    );\n\n    return {\n      canDelete: appliedCodes.length === 0,\n      appliedCodes,\n    };\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 삭제합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제된 마이그레이션 코드 파일 개수\n   */\n  async delCodes(codeNames: string[]): Promise<number> {\n    const { conns } = await this.getStatus();\n    const { canDelete, appliedCodes } = this.validateDeletable(conns, codeNames);\n    if (!canDelete) {\n      throw new Error(\n        `You cannot delete a migration file if there is already applied. Applied codes: ${appliedCodes.join(\", \")}`,\n      );\n    }\n\n    return sum(\n      await Promise.all(\n        codeNames.map(async (codeName) => {\n          const filePath = `${Sonamu.apiRootPath}/src/migrations/${codeName}.ts`;\n          if (await exists(filePath)) {\n            await unlink(filePath);\n            return 1;\n          }\n          return 0;\n        }),\n      ),\n    );\n  }\n\n  private genDateTag(index: number, baseDate: Date = new Date()): string {\n    const date = new Date(baseDate.getTime() + index * 1000);\n    const pad = (num: number, size: number = 2) => num.toString().padStart(size, \"0\");\n    return (\n      date.getFullYear().toString() +\n      pad(date.getMonth() + 1) +\n      pad(date.getDate()) +\n      pad(date.getHours()) +\n      pad(date.getMinutes()) +\n      pad(date.getSeconds())\n    );\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 생성합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns 생성된 마이그레이션 코드 파일 개수\n   */\n  async generatePreparedCodes(): Promise<number> {\n    const { preparedCodes } = await this.getStatus();\n    Naite.t(\"migrator:generatePreparedCodes:preparedCodes\", preparedCodes);\n    if (preparedCodes.length === 0) {\n      console.log(chalk.green(\"\\n현재 모두 싱크된 상태입니다.\"));\n      return 0;\n    }\n\n    // 실제 코드 생성\n    const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;\n\n    for (const [index, pcode] of preparedCodes.entries()) {\n      if (pcode.formatted) {\n        const dateTag = this.genDateTag(index);\n        const filePath = `${migrationsDir}/${dateTag}_${pcode.title}.ts`;\n        await writeFile(filePath, pcode.formatted);\n        !isTest() && console.log(chalk.green(`MIGRATION CREATED ${filePath}`));\n      }\n    }\n\n    return preparedCodes.length;\n  }\n\n  async compareMigrations(compareDB: Knex): Promise<GenMigrationCode[]> {\n    // Entity 순회하여 싱크\n    const entityIds = EntityManager.getAllIds();\n\n    // 조인테이블 포함하여 Entity에서 MigrationSet 추출\n    const entitySetsWithJoinTable = entityIds\n      .filter((entityId) => EntityManager.get(entityId).props.length > 0)\n      .map((entityId) => getMigrationSetFromEntity(EntityManager.get(entityId)));\n\n    // 조인테이블만 추출\n    const joinTablesWithDup = entitySetsWithJoinTable.flatMap((entitySet) => entitySet.joinTables);\n    // 중복 제거 (중복인 경우 indexes를 병합)\n    const joinTables = Object.values(group(joinTablesWithDup, (jt) => jt.table)).map((tables) => {\n      assert(tables !== undefined, \"tables is undefined\");\n      if (tables.length === 1) {\n        return tables[0];\n      }\n      return {\n        ...tables[0],\n        indexes: unique(\n          tables.flatMap((t) => t.indexes),\n          (index) => [index.type, ...index.columns].join(\"-\"),\n        ),\n      };\n    });\n\n    // 조인테이블 포함하여 MigrationSet 배열\n    const entitySets: MigrationSet[] = [...entitySetsWithJoinTable, ...joinTables];\n\n    const codes: GenMigrationCode[] = (\n      await Promise.all(\n        entitySets.map(async (entitySet) => {\n          const dbSet = await PostgreSQLSchemaReader.getMigrationSetFromDB(\n            compareDB,\n            entitySet.table,\n          );\n          Naite.t(`migrator:compareMigrations:entitySet:${entitySet.table}`, entitySet);\n          Naite.t(`migrator:compareMigrations:dbSet:${entitySet.table}`, dbSet);\n\n          if (dbSet === null) {\n            // 기존 테이블 없음, 새로 테이블 생성\n            return await generateCreateCode(entitySet);\n          } else {\n            // 기존 테이블 존재하는 케이스\n            return await generateAlterCode(entitySet, dbSet, compareDB);\n          }\n        }),\n      )\n    ).flat();\n\n    // normal 타입이 앞으로, foreign이 뒤로\n    codes.sort((codeA, codeB) => {\n      if (codeA.type === \"foreign\" && codeB.type === \"normal\") {\n        return 1;\n      } else if (codeA.type === \"normal\" && codeB.type === \"foreign\") {\n        return -1;\n      } else {\n        return 0;\n      }\n    });\n\n    return codes;\n  }\n\n  /**\n   * Shadow DB 테스트를 진행합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns Shadow DB 테스트 결과\n   */\n  async runShadowTest(): Promise<MigrationResult> {\n    const tdbConn = Sonamu.dbConfig.test.connection as Knex.PgConnectionConfig;\n    const shadowDatabase = `${tdbConn.database}__migration_shadow`;\n\n    // 테스트 상황에서는 트랜잭션을 초기화하고, 새 데이터베이스 커넥션을 가져와야 함\n    if (isTest()) {\n      await DB.clearTestTransaction();\n      // 병렬 테스트 모드에서는 worker DB 연결 유지\n      if (process.env.SONAMU_WORKER_DB !== \"true\") {\n        await DB.destroy();\n      }\n    }\n\n    // 기존 Shadow DB 삭제 후 Shadow DB 생성\n    const tdb = createKnexInstance(Sonamu.dbConfig.test);\n    !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n    await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n    await tdb.raw(`\n      SELECT pg_terminate_backend(pg_stat_activity.pid)\n      FROM pg_stat_activity\n      WHERE datname = '${tdbConn.database}'\n        AND pid <> pg_backend_pid();\n    `);\n    await tdb.raw(`CREATE DATABASE ${shadowDatabase} TEMPLATE ${tdbConn.database}`);\n\n    // Shadow DB에 연결\n    const sdb = createKnexInstance({\n      ...Sonamu.dbConfig.test,\n      connection: {\n        ...tdbConn,\n        database: shadowDatabase,\n        password: tdbConn.password,\n      },\n    });\n\n    // shadow DB 테스트 진행\n    try {\n      const [batchNo, applied] = await sdb.migrate.latest();\n      !isTest() &&\n        console.log(chalk.green(\"Shadow DB 테스트에 성공했습니다!\"), {\n          batchNo,\n          applied,\n        });\n\n      return [\n        {\n          connKey: \"shadow\",\n          batchNo,\n          applied,\n        },\n      ];\n    } catch (e) {\n      console.error(e);\n      throw new ServiceUnavailableException(SD(\"sonamu.error.shadowDbTestFailed\"));\n    } finally {\n      // Shadow DB 연결 종료\n      await sdb.destroy();\n\n      // Shadow DB 삭제\n      !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n      await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n\n      // Test DB 연결 종료\n      await tdb.destroy();\n    }\n  }\n}\n"],"names":["assert","chalk","mkdir","readdir","unlink","writeFile","path","group","sum","unique","Sonamu","DB","createKnexInstance","SD","EntityManager","ServiceUnavailableException","Naite","isTest","exists","generateAlterCode","generateCreateCode","getMigrationSetFromEntity","PostgreSQLSchemaReader","Migrator","getMigrationCodes","srcMigrationsDir","join","apiRootPath","recursive","codes","filter","f","endsWith","map","name","replace","sort","a","b","t","getStatus","connKeys","Object","keys","dbConfig","key","migrationStatusError","statuses","Promise","all","connKey","knexOptions","tConn","status","migrate","err","console","warn","yellow","JSON","stringify","connection","Error","message","String","pending","fdList","list","fd","file","currentVersion","_err","connString","user","host","port","database","destroy","preparedCodes","status0conn","find","undefined","compareDBconn","compareMigrations","conns","error","runAction","action","targets","configs","target","options","c","config","knex","result","batchNo","applied","latest","rollback","validateDeletable","codeNames","appliedCodes","codeName","some","conn","includes","canDelete","length","delCodes","filePath","genDateTag","index","baseDate","Date","date","getTime","pad","num","size","toString","padStart","getFullYear","getMonth","getDate","getHours","getMinutes","getSeconds","generatePreparedCodes","log","green","migrationsDir","pcode","entries","formatted","dateTag","title","compareDB","entityIds","getAllIds","entitySetsWithJoinTable","entityId","get","props","joinTablesWithDup","flatMap","entitySet","joinTables","values","jt","table","tables","indexes","type","columns","entitySets","dbSet","getMigrationSetFromDB","flat","codeA","codeB","runShadowTest","tdbConn","test","shadowDatabase","clearTestTransaction","process","env","SONAMU_WORKER_DB","tdb","magenta","raw","sdb","password","e"],"mappings":"AAAA,OAAOA,YAAY,SAAS;AAC5B,OAAOC,WAAW,QAAQ;AAC1B,SAASC,KAAK,EAAEC,OAAO,EAAEC,MAAM,EAAEC,SAAS,QAAQ,mBAAc;AAEhE,OAAOC,UAAU,OAAO;AACxB,SAASC,KAAK,EAAEC,GAAG,EAAEC,MAAM,QAAQ,UAAU;AAC7C,SAASC,MAAM,QAAQ,kBAAS;AAChC,SAASC,EAAE,QAA6B,oBAAiB;AACzD,SAASC,kBAAkB,QAAQ,sBAAmB;AACtD,SAASC,EAAE,QAAQ,gBAAa;AAChC,SAASC,aAAa,QAAQ,8BAA2B;AACzD,SAASC,2BAA2B,QAAQ,iCAA8B;AAC1E,SAASC,KAAK,QAAQ,oBAAiB;AAEvC,SAASC,MAAM,QAAQ,yBAAsB;AAC7C,SAASC,MAAM,QAAQ,uBAAoB;AAC3C,SAASC,iBAAiB,EAAEC,kBAAkB,QAAQ,uBAAoB;AAC1E,SAASC,yBAAyB,QAAQ,qBAAkB;AAC5D,SAASC,sBAAsB,QAAQ,gCAA6B;AASpE,OAAO,MAAMC;IACX,MAAcC,oBAA8C;QAC1D,MAAMC,mBAAmBnB,KAAKoB,IAAI,CAAChB,OAAOiB,WAAW,EAAE,OAAO,eAAe,+BAA+B;QAE5G,IAAI,CAAE,MAAMT,OAAOO,mBAAoB;YACrC,MAAMvB,MAAMuB,kBAAkB;gBAC5BG,WAAW;YACb;QACF;QAEA,MAAMC,QAAQ,AAAC,CAAA,MAAM1B,QAAQsB,iBAAgB,EAC1CK,MAAM,CAAC,CAACC,IAAMA,EAAEC,QAAQ,CAAC,QACzBC,GAAG,CAAC,CAACF,IAAO,CAAA;gBACXG,MAAMH,EAAEI,OAAO,CAAC,OAAO;gBACvB7B,MAAMA,KAAKoB,IAAI,CAACD,kBAAkBM;YACpC,CAAA,GACCK,IAAI,CAAC,CAACC,GAAGC,IAAOD,EAAEH,IAAI,GAAGI,EAAEJ,IAAI,GAAG,IAAI,CAAC,IAAK,kBAAkB;QAEjElB,MAAMuB,CAAC,CAAC,sCAAsCV;QAC9C,OAAOA;IACT;IAEA;;;;;;;;GAQC,GACD,MAAMW,YAAsC;QAC1C,MAAMX,QAAQ,MAAM,IAAI,CAACL,iBAAiB;QAC1CR,MAAMuB,CAAC,CAAC,4BAA4BV;QAEpC,MAAMY,WAAWC,OAAOC,IAAI,CAACjC,OAAOkC,QAAQ,EAAEd,MAAM,CAClD,CAACe,MAAQA,IAAIb,QAAQ,CAAC,cAAc;QAGtC,IAAIc;QAEJ,MAAMC,WAAW,MAAMC,QAAQC,GAAG,CAChCR,SAASR,GAAG,CAAC,OAAOiB;YAClB,MAAMC,cAAczC,OAAOkC,QAAQ,CAACM,QAAQ;YAC5C,MAAME,QAAQxC,mBAAmBuC;YAEjC,IAAI;gBACF,MAAME,SAAS,MAAM,AAAC,CAAA;oBACpB,IAAI;wBACF,OAAO,MAAMD,MAAME,OAAO,CAACD,MAAM;oBACnC,EAAE,OAAOE,KAAK;wBACZC,QAAQC,IAAI,CACVxD,MAAMyD,MAAM,CACV,GAAGR,QAAQ,yFAAyF,EAAES,KAAKC,SAAS,CAACT,YAAYU,UAAU,EAAE,MAAM,GAAG,WAAW,EAAEN,IAAI,EAAE,CAAC;wBAG9KT,uBAAuBS,eAAeO,QAAQP,IAAIQ,OAAO,GAAGC,OAAOT;wBACnE,OAAO;oBACT;gBACF,CAAA;gBACA,MAAMU,UAAoB,MAAM,AAAC,CAAA;oBAC/B,IAAI;wBACF,MAAM,GAAGC,OAAO,GAAG,MAAMd,MAAME,OAAO,CAACa,IAAI;wBAC3C,OAAOD,OAAOjC,GAAG,CAAC,CAACmC,KAAyBA,GAAGC,IAAI,CAAClC,OAAO,CAAC,OAAO;oBACrE,EAAE,OAAOoB,KAAK;wBACZT,uBAAuBS,eAAeO,QAAQP,IAAIQ,OAAO,GAAGC,OAAOT;wBACnE,OAAO,EAAE;oBACX;gBACF,CAAA;gBACA,MAAMe,iBAAiB,MAAM,AAAC,CAAA;oBAC5B,IAAI;wBACF,OAAO,MAAMlB,MAAME,OAAO,CAACgB,cAAc;oBAC3C,EAAE,OAAOC,MAAM;wBACbzB,uBAAuByB,gBAAgBT,QAAQS,KAAKR,OAAO,GAAGC,OAAOO;wBACrE,OAAO;oBACT;gBACF,CAAA;gBACAvD,MAAMuB,CAAC,CAAC,6BAA6Bc;gBAErC,MAAMQ,aAAaV,YAAYU,UAAU;gBAEzC,OAAO;oBACL3B,MAAMgB,QAAQf,OAAO,CAAC,WAAW;oBACjCe;oBACAsB,YAAY,CAAC,KAAK,EAAEX,WAAWY,IAAI,IAAI,GAAG,CAAC,EAAEZ,WAAWa,IAAI,CAAC,CAAC,EAC5Db,WAAWc,IAAI,CAChB,CAAC,EAAEd,WAAWe,QAAQ,EAAE;oBACzBN;oBACAjB,QAAQA;oBACRY;gBACF;YACF,SAAU;gBACR,MAAMb,MAAMyB,OAAO;YACrB;QACF;QAGF7D,MAAMuB,CAAC,CAAC,4BAA4BQ;QAEpC,MAAM+B,gBAAoC,MAAM,AAAC,CAAA;YAC/C,MAAMC,cAAchC,SAASiC,IAAI,CAAC,CAAC3B,SAAWA,OAAOA,MAAM,KAAK;YAChE,IAAI0B,gBAAgBE,WAAW;gBAC7BzB,QAAQC,IAAI,CACVxD,MAAMyD,MAAM,CACV,CAAC,0PAA0P,CAAC;gBAGhQ,OAAO,EAAE;YACX;YAEA,MAAMwB,gBAAgBtE,mBAAmBF,OAAOkC,QAAQ,CAACmC,YAAY7B,OAAO,CAAC;YAC7E,IAAI;gBACF,OAAO,MAAM,IAAI,CAACiC,iBAAiB,CAACD;YACtC,SAAU;gBACR,MAAMA,cAAcL,OAAO;YAC7B;QACF,CAAA;QAEA7D,MAAMuB,CAAC,CAAC,oCAAoCuC;QAE5C,OAAO;YACLM,OAAOrC;YACPlB;YACAiD;YACAO,OAAOvC;QACT;IACF;IAEA;;;;;;;;;GASC,GACD,MAAMwC,UACJC,MAA4B,EAC5BC,OAAiC,EACP;QAC1BxE,MAAMuB,CAAC,CAAC,6BAA6BgD;QACrCvE,MAAMuB,CAAC,CAAC,8BAA8BiD;QAEtC,wBAAwB;QACxB,MAAMC,UAAUhF,OACd+E,QACGvD,GAAG,CAAC,CAACyD,SAAY,CAAA;gBAChBxC,SAASwC;gBACTC,SAASjF,OAAOkC,QAAQ,CAAC8C,OAAuC;YAClE,CAAA,GACC5D,MAAM,CAAC,CAAC8D,IAAMA,EAAED,OAAO,KAAKV,YAC/B,CAAC,EAAEU,OAAO,EAAE,GACV,GAAG,AAACA,QAAQ9B,UAAU,CAA6Ba,IAAI,CAAC,CAAC,EACvD,AAACiB,QAAQ9B,UAAU,CAA6Bc,IAAI,IAAI,KACzD,CAAC,EAAE,AAACgB,QAAQ9B,UAAU,CAA6Be,QAAQ,EAAE;QAGlE,kBAAkB;QAClB,MAAMQ,QAAQ,MAAMpC,QAAQC,GAAG,CAC7BwC,QAAQxD,GAAG,CAAC,OAAO4D,SAAY,CAAA;gBAC7B3C,SAAS2C,OAAO3C,OAAO;gBACvB4C,MAAMlF,mBAAmBiF,OAAOF,OAAO;YACzC,CAAA;QAGF,IAAI;YACF,SAAS;YACT,MAAMI,SAAS,MAAM,AAAC,CAAA;gBACpB,OAAQR;oBACN,KAAK;wBACH,OAAOvC,QAAQC,GAAG,CAChBmC,MAAMnD,GAAG,CAAC,OAAO,EAAEiB,OAAO,EAAE4C,IAAI,EAAE;4BAChC,MAAM,CAACE,SAASC,QAAQ,GAAG,MAAMH,KAAKxC,OAAO,CAAC4C,MAAM;4BACpD,OAAO;gCACLhD;gCACA8C;gCACAC;4BACF;wBACF;oBAEJ,KAAK;wBACH,OAAOjD,QAAQC,GAAG,CAChBmC,MAAMnD,GAAG,CAAC,OAAO,EAAEiB,OAAO,EAAE4C,IAAI,EAAE;4BAChC,MAAM,CAACE,SAASC,QAAQ,GAAG,MAAMH,KAAKxC,OAAO,CAAC6C,QAAQ;4BACtD,OAAO;gCACLjD;gCACA8C;gCACAC;4BACF;wBACF;gBAEN;YACF,CAAA;YAEAjF,MAAMuB,CAAC,CAAC,6BAA6BwD;YAErC,OAAOA;QACT,SAAU;YACR,MAAM/C,QAAQC,GAAG,CACfmC,MAAMnD,GAAG,CAAC,CAAC,EAAE6D,IAAI,EAAE;gBACjB,OAAOA,KAAKjB,OAAO;YACrB;QAEJ;IACF;IAEA;;;;;;GAMC,GACDuB,kBAAkBhB,KAA+B,EAAEiB,SAAmB,EAAE;QACtE,MAAMC,eAAeD,UAAUvE,MAAM,CAAC,CAACyE,WACrCnB,MAAMoB,IAAI,CAAC,CAACC,OAASA,KAAKxC,OAAO,CAACyC,QAAQ,CAACH,cAAc;QAG3D,OAAO;YACLI,WAAWL,aAAaM,MAAM,KAAK;YACnCN;QACF;IACF;IAEA;;;;;;;GAOC,GACD,MAAMO,SAASR,SAAmB,EAAmB;QACnD,MAAM,EAAEjB,KAAK,EAAE,GAAG,MAAM,IAAI,CAAC5C,SAAS;QACtC,MAAM,EAAEmE,SAAS,EAAEL,YAAY,EAAE,GAAG,IAAI,CAACF,iBAAiB,CAAChB,OAAOiB;QAClE,IAAI,CAACM,WAAW;YACd,MAAM,IAAI7C,MACR,CAAC,+EAA+E,EAAEwC,aAAa5E,IAAI,CAAC,OAAO;QAE/G;QAEA,OAAOlB,IACL,MAAMwC,QAAQC,GAAG,CACfoD,UAAUpE,GAAG,CAAC,OAAOsE;YACnB,MAAMO,WAAW,GAAGpG,OAAOiB,WAAW,CAAC,gBAAgB,EAAE4E,SAAS,GAAG,CAAC;YACtE,IAAI,MAAMrF,OAAO4F,WAAW;gBAC1B,MAAM1G,OAAO0G;gBACb,OAAO;YACT;YACA,OAAO;QACT;IAGN;IAEQC,WAAWC,KAAa,EAAEC,WAAiB,IAAIC,MAAM,EAAU;QACrE,MAAMC,OAAO,IAAID,KAAKD,SAASG,OAAO,KAAKJ,QAAQ;QACnD,MAAMK,MAAM,CAACC,KAAaC,OAAe,CAAC,GAAKD,IAAIE,QAAQ,GAAGC,QAAQ,CAACF,MAAM;QAC7E,OACEJ,KAAKO,WAAW,GAAGF,QAAQ,KAC3BH,IAAIF,KAAKQ,QAAQ,KAAK,KACtBN,IAAIF,KAAKS,OAAO,MAChBP,IAAIF,KAAKU,QAAQ,MACjBR,IAAIF,KAAKW,UAAU,MACnBT,IAAIF,KAAKY,UAAU;IAEvB;IAEA;;;;;;GAMC,GACD,MAAMC,wBAAyC;QAC7C,MAAM,EAAElD,aAAa,EAAE,GAAG,MAAM,IAAI,CAACtC,SAAS;QAC9CxB,MAAMuB,CAAC,CAAC,gDAAgDuC;QACxD,IAAIA,cAAc8B,MAAM,KAAK,GAAG;YAC9BpD,QAAQyE,GAAG,CAAChI,MAAMiI,KAAK,CAAC;YACxB,OAAO;QACT;QAEA,WAAW;QACX,MAAMC,gBAAgB,GAAGzH,OAAOiB,WAAW,CAAC,eAAe,CAAC;QAE5D,KAAK,MAAM,CAACqF,OAAOoB,MAAM,IAAItD,cAAcuD,OAAO,GAAI;YACpD,IAAID,MAAME,SAAS,EAAE;gBACnB,MAAMC,UAAU,IAAI,CAACxB,UAAU,CAACC;gBAChC,MAAMF,WAAW,GAAGqB,cAAc,CAAC,EAAEI,QAAQ,CAAC,EAAEH,MAAMI,KAAK,CAAC,GAAG,CAAC;gBAChE,MAAMnI,UAAUyG,UAAUsB,MAAME,SAAS;gBACzC,CAACrH,YAAYuC,QAAQyE,GAAG,CAAChI,MAAMiI,KAAK,CAAC,CAAC,kBAAkB,EAAEpB,UAAU;YACtE;QACF;QAEA,OAAOhC,cAAc8B,MAAM;IAC7B;IAEA,MAAMzB,kBAAkBsD,SAAe,EAA+B;QACpE,iBAAiB;QACjB,MAAMC,YAAY5H,cAAc6H,SAAS;QAEzC,sCAAsC;QACtC,MAAMC,0BAA0BF,UAC7B5G,MAAM,CAAC,CAAC+G,WAAa/H,cAAcgI,GAAG,CAACD,UAAUE,KAAK,CAACnC,MAAM,GAAG,GAChE3E,GAAG,CAAC,CAAC4G,WAAaxH,0BAA0BP,cAAcgI,GAAG,CAACD;QAEjE,YAAY;QACZ,MAAMG,oBAAoBJ,wBAAwBK,OAAO,CAAC,CAACC,YAAcA,UAAUC,UAAU;QAC7F,6BAA6B;QAC7B,MAAMA,aAAazG,OAAO0G,MAAM,CAAC7I,MAAMyI,mBAAmB,CAACK,KAAOA,GAAGC,KAAK,GAAGrH,GAAG,CAAC,CAACsH;YAChFvJ,OAAOuJ,WAAWtE,WAAW;YAC7B,IAAIsE,OAAO3C,MAAM,KAAK,GAAG;gBACvB,OAAO2C,MAAM,CAAC,EAAE;YAClB;YACA,OAAO;gBACL,GAAGA,MAAM,CAAC,EAAE;gBACZC,SAAS/I,OACP8I,OAAON,OAAO,CAAC,CAAC1G,IAAMA,EAAEiH,OAAO,GAC/B,CAACxC,QAAU;wBAACA,MAAMyC,IAAI;2BAAKzC,MAAM0C,OAAO;qBAAC,CAAChI,IAAI,CAAC;YAEnD;QACF;QAEA,6BAA6B;QAC7B,MAAMiI,aAA6B;eAAIf;eAA4BO;SAAW;QAE9E,MAAMtH,QAA4B,AAChC,CAAA,MAAMmB,QAAQC,GAAG,CACf0G,WAAW1H,GAAG,CAAC,OAAOiH;YACpB,MAAMU,QAAQ,MAAMtI,uBAAuBuI,qBAAqB,CAC9DpB,WACAS,UAAUI,KAAK;YAEjBtI,MAAMuB,CAAC,CAAC,CAAC,qCAAqC,EAAE2G,UAAUI,KAAK,EAAE,EAAEJ;YACnElI,MAAMuB,CAAC,CAAC,CAAC,iCAAiC,EAAE2G,UAAUI,KAAK,EAAE,EAAEM;YAE/D,IAAIA,UAAU,MAAM;gBAClB,uBAAuB;gBACvB,OAAO,MAAMxI,mBAAmB8H;YAClC,OAAO;gBACL,kBAAkB;gBAClB,OAAO,MAAM/H,kBAAkB+H,WAAWU,OAAOnB;YACnD;QACF,GACF,EACAqB,IAAI;QAEN,8BAA8B;QAC9BjI,MAAMO,IAAI,CAAC,CAAC2H,OAAOC;YACjB,IAAID,MAAMN,IAAI,KAAK,aAAaO,MAAMP,IAAI,KAAK,UAAU;gBACvD,OAAO;YACT,OAAO,IAAIM,MAAMN,IAAI,KAAK,YAAYO,MAAMP,IAAI,KAAK,WAAW;gBAC9D,OAAO,CAAC;YACV,OAAO;gBACL,OAAO;YACT;QACF;QAEA,OAAO5H;IACT;IAEA;;;;;;GAMC,GACD,MAAMoI,gBAA0C;QAC9C,MAAMC,UAAUxJ,OAAOkC,QAAQ,CAACuH,IAAI,CAACtG,UAAU;QAC/C,MAAMuG,iBAAiB,GAAGF,QAAQtF,QAAQ,CAAC,kBAAkB,CAAC;QAE9D,8CAA8C;QAC9C,IAAI3D,UAAU;YACZ,MAAMN,GAAG0J,oBAAoB;YAC7B,+BAA+B;YAC/B,IAAIC,QAAQC,GAAG,CAACC,gBAAgB,KAAK,QAAQ;gBAC3C,MAAM7J,GAAGkE,OAAO;YAClB;QACF;QAEA,iCAAiC;QACjC,MAAM4F,MAAM7J,mBAAmBF,OAAOkC,QAAQ,CAACuH,IAAI;QACnD,CAAClJ,YAAYuC,QAAQyE,GAAG,CAAChI,MAAMyK,OAAO,CAAC,GAAGN,eAAe,GAAG,CAAC;QAC7D,MAAMK,IAAIE,GAAG,CAAC,CAAC,wBAAwB,EAAEP,gBAAgB;QACzD,MAAMK,IAAIE,GAAG,CAAC,CAAC;;;uBAGI,EAAET,QAAQtF,QAAQ,CAAC;;IAEtC,CAAC;QACD,MAAM6F,IAAIE,GAAG,CAAC,CAAC,gBAAgB,EAAEP,eAAe,UAAU,EAAEF,QAAQtF,QAAQ,EAAE;QAE9E,gBAAgB;QAChB,MAAMgG,MAAMhK,mBAAmB;YAC7B,GAAGF,OAAOkC,QAAQ,CAACuH,IAAI;YACvBtG,YAAY;gBACV,GAAGqG,OAAO;gBACVtF,UAAUwF;gBACVS,UAAUX,QAAQW,QAAQ;YAC5B;QACF;QAEA,mBAAmB;QACnB,IAAI;YACF,MAAM,CAAC7E,SAASC,QAAQ,GAAG,MAAM2E,IAAItH,OAAO,CAAC4C,MAAM;YACnD,CAACjF,YACCuC,QAAQyE,GAAG,CAAChI,MAAMiI,KAAK,CAAC,2BAA2B;gBACjDlC;gBACAC;YACF;YAEF,OAAO;gBACL;oBACE/C,SAAS;oBACT8C;oBACAC;gBACF;aACD;QACH,EAAE,OAAO6E,GAAG;YACVtH,QAAQ6B,KAAK,CAACyF;YACd,MAAM,IAAI/J,4BAA4BF,GAAG;QAC3C,SAAU;YACR,kBAAkB;YAClB,MAAM+J,IAAI/F,OAAO;YAEjB,eAAe;YACf,CAAC5D,YAAYuC,QAAQyE,GAAG,CAAChI,MAAMyK,OAAO,CAAC,GAAGN,eAAe,GAAG,CAAC;YAC7D,MAAMK,IAAIE,GAAG,CAAC,CAAC,wBAAwB,EAAEP,gBAAgB;YAEzD,gBAAgB;YAChB,MAAMK,IAAI5F,OAAO;QACnB;IACF;AACF"}
|
package/dist/ui/api.js
CHANGED
|
@@ -26,14 +26,14 @@ export async function sonamuUIApiPlugin(fastify) {
|
|
|
26
26
|
// waitForHMRCompleted
|
|
27
27
|
async function waitForHMRCompleted(fn) {
|
|
28
28
|
const waitPromise = new Promise((resolve)=>{
|
|
29
|
-
const timeout = setTimeout(()=>{
|
|
30
|
-
resolve();
|
|
31
|
-
}, 1500);
|
|
32
29
|
const handler = ()=>{
|
|
33
30
|
clearTimeout(timeout);
|
|
34
|
-
Sonamu.syncer.eventEmitter.off("onHMRCompleted", handler);
|
|
35
31
|
resolve();
|
|
36
32
|
};
|
|
33
|
+
const timeout = setTimeout(()=>{
|
|
34
|
+
Sonamu.syncer.eventEmitter.off("onHMRCompleted", handler);
|
|
35
|
+
resolve();
|
|
36
|
+
}, 1500);
|
|
37
37
|
Sonamu.syncer.eventEmitter.once("onHMRCompleted", handler);
|
|
38
38
|
});
|
|
39
39
|
const result = await fn();
|
|
@@ -1111,4 +1111,4 @@ export async function sonamuUIApiPlugin(fastify) {
|
|
|
1111
1111
|
});
|
|
1112
1112
|
}
|
|
1113
1113
|
|
|
1114
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/ui/api.ts"],"sourcesContent":["import { execSync } from \"child_process\";\nimport type { FastifyInstance } from \"fastify\";\nimport fs from \"fs\";\nimport inflection from \"inflection\";\nimport type { AddressInfo } from \"net\";\nimport path from \"path\";\nimport { range } from \"radashi\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { DB, type SonamuDBConfig } from \"../database/db\";\nimport { createKnexInstance } from \"../database/knex\";\nimport { SD } from \"../dict/sd\";\nimport { sonamuDictionary } from \"../dict/sonamu-dictionary\";\nimport type { Entity } from \"../entity/entity\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport {\n  BadRequestException,\n  isSoException,\n  ServiceUnavailableException,\n} from \"../exceptions/so-exceptions\";\nimport { type MigrationResult, Migrator } from \"../migration/migrator\";\nimport { SlackConfirm, type SlackConfirmPendingResult } from \"../migration/slack-confirm\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport { DataExplorer } from \"../testing/data-explorer\";\nimport { FixtureGenerator } from \"../testing/fixture-generator\";\nimport { type DuplicateCheckOptions, FixtureManager } from \"../testing/fixture-manager\";\nimport {\n  BUILT_IN_TYPE_IDS,\n  type Cone,\n  type EntityIndex,\n  type EntityProp,\n  type EntitySubsetRow,\n  type FixtureRecord,\n  type FixtureSearchOptions,\n  type FlattenSubsetRow,\n  type PathAndCode,\n  TemplateKey,\n} from \"../types/types\";\nimport { nonNullable } from \"../utils/utils\";\nimport { setAiApi } from \"./ai-api\";\n\nexport async function sonamuUIApiPlugin(fastify: FastifyInstance) {\n  fastify.register(\n    async (server) => {\n      // migrator\n      const migrator = new Migrator();\n\n      // waitForHMRCompleted\n      async function waitForHMRCompleted<T>(fn: () => Promise<T>): Promise<T> {\n        const waitPromise = new Promise<void>((resolve) => {\n          const timeout = setTimeout(() => {\n            resolve();\n          }, 1500);\n\n          const handler = () => {\n            clearTimeout(timeout);\n            Sonamu.syncer.eventEmitter.off(\"onHMRCompleted\", handler);\n            resolve();\n          };\n\n          Sonamu.syncer.eventEmitter.once(\"onHMRCompleted\", handler);\n        });\n\n        const result = await fn();\n        await waitPromise;\n        return result;\n      }\n\n      await setAiApi(server);\n\n      server.get(\"/api/sonamu/config\", async () => {\n        return Sonamu.config;\n      });\n\n      server.get<{\n        Querystring: {\n          entityId?: string;\n          preset?: \"types\" | \"entity.json\" | \"generated\";\n          absPath?: string;\n        };\n      }>(\"/api/tools/openVscode\", async (request) => {\n        const { entityId, preset, absPath } = request.query;\n\n        const targetPath = (() => {\n          if (entityId && preset) {\n            const entity = EntityManager.get(entityId);\n            const { names } = entity;\n\n            const { apiRootPath } = Sonamu;\n            const filename = (() => {\n              switch (preset) {\n                case \"types\":\n                  return `${names.fs}.types.ts`;\n                case \"entity.json\":\n                  return `${names.fs}.entity.json`;\n                case \"generated\":\n                  return `${names.fs}.generated.ts`;\n              }\n            })();\n            return `${apiRootPath}/src/application/${entity.names.parentFs}/${filename}`;\n          } else {\n            if (!absPath) {\n              throw new BadRequestException(SD(\"sonamu.error.presetOrAbsPathRequired\"));\n            }\n            return absPath;\n          }\n        })();\n        execSync(`code ${targetPath}`);\n      });\n\n      server.get<{\n        Querystring: {\n          origin: string;\n          entityId?: string;\n        };\n      }>(\"/api/tools/getSuggestion\", async (request) => {\n        const { origin, entityId } = request.query;\n\n        // 치환 용어집\n        const glossary = new Map<string, string>([\n          [\"status\", \"상태\"],\n          [\"type\", \"타입\"],\n          [\"image\", \"이미지\"],\n          [\"images\", \"이미지리스트\"],\n          [\"url\", \"URL\"],\n          [\"id\", \"ID\"],\n          [\"name\", `{EntityID}명`],\n          [\"title\", \"{EntityID}명\"],\n          [\"parent\", \"상위{EntityID}\"],\n          [\"desc\", \"설명\"],\n          [\"at\", \"일시\"],\n          [\"created\", \"등록\"],\n          [\"updated\", \"수정\"],\n          [\"deleted\", \"삭제\"],\n          [\"by\", \"유저\"],\n          [\"date\", \"일자\"],\n          [\"time\", \"시간\"],\n          [\"ko\", \"(한글)\"],\n          [\"en\", \"(영문)\"],\n          [\"krw\", \"(원)\"],\n          [\"usd\", \"(USD)\"],\n          [\"color\", \"컬러\"],\n          [\"code\", \"코드\"],\n          [\"x\", \"X좌표\"],\n          [\"y\", \"Y좌표\"],\n          [\"current\", \"현재\"],\n          [\"stock\", \"재고\"],\n          [\"total\", \"총\"],\n          [\"admin\", \"관리자\"],\n          [\"group\", \"그룹\"],\n          [\"item\", \"아이템\"],\n          [\"cnt\", \"수량\"],\n          [\"price\", \"가격\"],\n          [\"preset\", \"프리셋\"],\n          [\"acct\", \"계좌\"],\n          [\"tel\", \"전화번호\"],\n          [\"no\", \"번호\"],\n          [\"body\", \"내용\"],\n          [\"content\", \"내용\"],\n          [\"orderno\", \"정렬순서\"],\n          [\"priority\", \"우선순위\"],\n          [\"text\", \"텍스트\"],\n          [\"key\", \"키\"],\n          [\"sum\", \"합산\"],\n          [\"expected\", \"예상\"],\n          [\"actual\", \"실제\"],\n        ]);\n        // 전체 엔티티 순회하며, 엔티티 타이틀과 프롭 설명을 치환 용어집에 추가\n        for (const entityId of EntityManager.getAllIds()) {\n          const entity = EntityManager.get(entityId);\n          if ((entity.title ?? \"\") !== \"\") {\n            glossary.set(inflection.underscore(entity.id), entity.title);\n            glossary.set(\n              inflection.underscore(inflection.pluralize(entity.id)),\n              `${entity.title}리스트`,\n            );\n          }\n\n          entity.props.forEach((prop) => {\n            if (glossary.has(prop.name)) {\n              return;\n            }\n            if (prop.desc) {\n              glossary.set(prop.name, prop.desc.replace(entity.title ?? \"\", \"{EntityID}\"));\n            }\n          });\n        }\n\n        const suggested = (() => {\n          // 단어 분리, 가능한 조합 생성\n          const words = origin.split(\"_\");\n          const combinations = [...range(words.length, 0, -1)].flatMap((len) => {\n            return [\n              ...range(0, words.length - len + 1, (idx) => {\n                return {\n                  len,\n                  w: words.slice(idx, idx + len).join(\"_\"),\n                };\n              }),\n            ];\n          });\n\n          // 조합을 순회하며, 치환 용어집에 있는 단어가 포함된 경우, 치환 용어로 치환\n          const REPLACED_PREFIX = \"#REPLACED//\"; // 치환된 단어를 join 이후에도 식별하기 위해 prefix 추가\n          let remainArr: string[] = [...words];\n          for (const comb of combinations) {\n            const remainStr = remainArr.join(\"_\");\n            if (remainStr.includes(comb.w) && glossary.has(comb.w)) {\n              remainArr = remainStr\n                .replace(comb.w, REPLACED_PREFIX + glossary.get(comb.w))\n                .split(\"_\");\n            }\n          }\n\n          return remainArr\n            .map((r) => {\n              if (r.startsWith(REPLACED_PREFIX)) {\n                return r.replace(REPLACED_PREFIX, \"\");\n              } else {\n                return r.toUpperCase();\n              }\n            })\n            .join(\"\")\n            .replace(/{EntityID}/g, entityId ? EntityManager.get(entityId).title : \"\");\n        })();\n\n        return { suggested };\n      });\n\n      server.get(\"/api/entity/findMany\", async () => {\n        const entityIds = EntityManager.getAllIds();\n\n        function flattenSubsetRows(subsetRows: EntitySubsetRow[]): FlattenSubsetRow[] {\n          return subsetRows.flatMap((subsetRow) => {\n            const { children, ...sRow } = subsetRow;\n            return [sRow, ...flattenSubsetRows(children)];\n          });\n        }\n\n        const entities = await Promise.all(\n          entityIds.map((entityId) => {\n            const entity = EntityManager.get(entityId);\n            const subsetRows = entity.getSubsetRows();\n\n            return {\n              ...entity,\n              flattenSubsetRows: flattenSubsetRows(subsetRows),\n            };\n          }),\n        );\n\n        entities.sort((a, b) => {\n          const aId = a.parentId ?? a.id;\n          const bId = b.parentId ?? b.id;\n          if (aId < bId) return -1;\n          if (aId > bId) return 1;\n          if (aId === bId) {\n            if (a.parentId === undefined) return -1;\n            if (b.parentId === undefined) return 1;\n            return 0;\n          }\n          return 0;\n        });\n        return { entities };\n      });\n\n      server.get<{\n        Querystring: {\n          filter?: \"enums\" | \"types\";\n          reload?: \"1\";\n        };\n      }>(\"/api/entity/typeIds\", async (request): Promise<{ typeIds: string[] }> => {\n        const { filter, reload } = request.query;\n\n        if (reload === \"1\") {\n          await Sonamu.syncer.autoloadTypes();\n        }\n\n        const typeIds = (() => {\n          // 프로젝트에서 정의한 타입들\n          const projectTypeIds = Object.entries(Sonamu.syncer.types)\n            .filter(([_typeId, zodType]) => (zodType._zod.def.type as string) !== \"enum\")\n            .map(([typeId, _zodType]) => typeId);\n\n          // 내장 타입들 (sonamu 코어에서 제공)\n          const builtInTypeIds = [...BUILT_IN_TYPE_IDS];\n\n          // 모든 타입 병합\n          const allTypeIds = [...builtInTypeIds, ...projectTypeIds];\n\n          if (filter === \"types\") {\n            return allTypeIds;\n          }\n\n          const enumIds = EntityManager.getAllIds().flatMap((entityId) => {\n            const entity = EntityManager.get(entityId);\n            return Object.keys(entity.enumLabels);\n          });\n\n          if (filter === \"enums\") {\n            return enumIds;\n          } else {\n            return [...allTypeIds, ...enumIds];\n          }\n        })();\n\n        return {\n          typeIds,\n        };\n      });\n\n      server.post<{\n        Body: {\n          form: {\n            id: string;\n            title: string;\n            table: string;\n            parentId?: string;\n          };\n        };\n      }>(\"/api/entity/create\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { form } = request.body;\n          await Sonamu.syncer.createEntity({ ...form, entityId: form.id });\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n        };\n      }>(\"/api/entity/del\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId } = request.body;\n          return await Sonamu.syncer.delEntity(entityId);\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newValues: {\n            title: string;\n            table: string;\n            parentId?: string;\n          };\n        };\n      }>(\"/api/entity/modifyEntityBase\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, newValues } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.title = newValues.title;\n          entity.table = newValues.table;\n          entity.parentId = newValues.parentId;\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          subsetKey: string;\n          fields: string[];\n          fieldsInternal?: string[];\n        };\n      }>(\"/api/entity/modifySubset\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, subsetKey, fields, fieldsInternal } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.subsets[subsetKey] = fields;\n          if (fieldsInternal !== undefined) {\n            if (fieldsInternal.length > 0) {\n              entity.subsetsInternal[subsetKey] = fieldsInternal;\n            } else {\n              delete entity.subsetsInternal[subsetKey];\n            }\n          }\n          await entity.save();\n\n          return { updated: fields, updatedInternal: fieldsInternal };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          subsetKey: string;\n        };\n      }>(\"/api/entity/delSubset\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, subsetKey } = request.body;\n          const entity = EntityManager.get(entityId);\n          delete entity.subsets[subsetKey];\n          delete entity.subsetsInternal[subsetKey];\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newProp: EntityProp;\n          at?: number;\n        };\n      }>(\"/api/entity/createProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, newProp } = request.body;\n          const entity = EntityManager.get(entityId);\n          await entity.createProp(newProp, at);\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newProp: EntityProp;\n          at: number;\n        };\n      }>(\"/api/entity/modifyProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, newProp } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.modifyProp(newProp, at);\n\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          at: number;\n        };\n      }>(\"/api/entity/delProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.delProp(at);\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          at: number;\n          to: number;\n        };\n      }>(\"/api/entity/moveProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, to } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.moveProp(at, to);\n\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          indexes: EntityIndex[];\n        };\n      }>(\"/api/entity/modifyIndexes\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, indexes } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.indexes = indexes;\n          await entity.save();\n\n          return { updated: indexes };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumLabels: Entity[\"enumLabels\"];\n        };\n      }>(\"/api/entity/modifyEnumLabels\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumLabels } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.enumLabels = enumLabels;\n          await entity.save();\n\n          return { updated: enumLabels };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newEnumId: string;\n        };\n      }>(\"/api/entity/createEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, newEnumId } = request.body;\n          const entity = EntityManager.get(entityId);\n\n          if (entity.enumLabels[newEnumId]) {\n            throw new Error(`이미 존재하는 enumId입니다: ${newEnumId}`);\n          }\n\n          entity.enumLabels[newEnumId] = {\n            ...(newEnumId.endsWith(\"Status\")\n              ? {\n                  active: \"노출\",\n                  hidden: \"숨김\",\n                }\n              : {\n                  \"\": \"\",\n                }),\n          };\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumId: {\n            before: string;\n            after: string;\n          };\n        };\n      }>(\"/api/entity/modifyEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumId } = request.body;\n          const entityIds = EntityManager.getAllIds();\n          const isExists = entityIds.some((entityId) => {\n            const entity = EntityManager.get(entityId);\n            return Object.keys(entity.enumLabels).includes(enumId.after);\n          });\n          if (isExists) {\n            throw new Error(`이미 존재하는 EnumId입니다: ${enumId.after}`);\n          }\n\n          const entity = EntityManager.get(entityId);\n          entity.enumLabels[enumId.after] = entity.enumLabels[enumId.before];\n          delete entity.enumLabels[enumId.before];\n\n          await entity.save();\n\n          for (const entityId of entityIds) {\n            const entity = EntityManager.get(entityId);\n            for (const prop of entity.props) {\n              if (prop.type === \"enum\" && prop.id === enumId.before) {\n                prop.id = enumId.after;\n              }\n            }\n            await entity.save();\n          }\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumId: string;\n        };\n      }>(\"/api/entity/deleteEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumId } = request.body;\n\n          const entityIds = EntityManager.getAllIds();\n          const isReferenced = entityIds\n            .flatMap((entityId) => EntityManager.get(entityId).props)\n            .some((prop) => prop.type === \"enum\" && prop.id === enumId);\n          if (isReferenced) {\n            throw new Error(`${enumId}를 참조하는 프로퍼티가 존재합니다.`);\n          }\n\n          const entity = EntityManager.get(entityId);\n          delete entity.enumLabels[enumId];\n          await entity.save();\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          target: \"entity\" | \"prop\" | \"enum\" | \"subset\";\n          propName?: string;\n          enumId?: string;\n          subsetKey?: string;\n          cone: Cone;\n        };\n      }>(\"/api/entity/updateCone\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, target, propName, enumId, subsetKey, cone } = request.body;\n          const entity = EntityManager.get(entityId);\n\n          if (target === \"entity\") {\n            entity.cone = cone;\n          } else if (target === \"prop\" && propName) {\n            const prop = entity.props.find((p) => p.name === propName);\n            if (prop) {\n              (prop as { cone?: Cone }).cone = cone;\n            }\n          } else if (target === \"enum\" && enumId) {\n            entity.enumCones[enumId] = cone;\n          } else if (target === \"subset\" && subsetKey) {\n            entity.subsetCones[subsetKey] = cone;\n          }\n\n          await entity.save();\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          preserveExisting?: boolean;\n          onlyEmpty?: boolean;\n          locale?: \"ko\" | \"en\" | \"ja\";\n        };\n      }>(\"/api/entity/generateCones\", async (request, reply) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, preserveExisting, onlyEmpty, locale } = request.body;\n\n          try {\n            // Entity 존재 여부 확인\n            const entity = EntityManager.get(entityId);\n\n            // locale 기본값: Sonamu.config.i18n.defaultLocale 사용\n            const effectiveLocale =\n              locale ?? (Sonamu.config.i18n.defaultLocale as \"ko\" | \"en\" | \"ja\");\n\n            // Cone 생성\n            const result = await entity.generateCones({\n              preserveExisting: preserveExisting ?? true,\n              onlyEmpty: onlyEmpty ?? false,\n              locale: effectiveLocale,\n            });\n\n            return result;\n          } catch (error: unknown) {\n            const message = error instanceof Error ? error.message : String(error);\n\n            // Entity not found\n            if (message.includes(\"존재하지 않는 Entity\")) {\n              reply.status(404);\n              return {\n                success: false,\n                error: `Entity not found: ${entityId}`,\n              };\n            }\n\n            // API 키 없음\n            if (message.includes(\"ANTHROPIC_API_KEY not found\")) {\n              reply.status(500);\n              return {\n                success: false,\n                error: \"API key not configured\",\n              };\n            }\n\n            // Rate limit\n            if (message.includes(\"Rate limit exceeded\")) {\n              reply.status(429);\n              return {\n                success: false,\n                error: \"Rate limit exceeded. Please try again later.\",\n              };\n            }\n\n            // 기타 에러\n            reply.status(500);\n            return {\n              success: false,\n              error: `Cone generation failed: ${message}`,\n            };\n          }\n        });\n      });\n\n      server.get<{\n        Querystring: {\n          entityId: string;\n        };\n      }>(\"/api/entity/getTableColumns\", async (request) => {\n        const { entityId } = request.query;\n        const entity = EntityManager.get(entityId);\n        const columns = entity.getTableColumns();\n        return { columns };\n      });\n\n      server.get(\"/api/migrations/status\", async () => {\n        const status = await migrator.getStatus();\n\n        return { status };\n      });\n\n      server.post<{\n        Body: {\n          action: \"apply\" | \"rollback\" | \"shadow\";\n          targets: (keyof SonamuDBConfig)[];\n          force?: boolean;\n          forceReason?: string;\n          requestor?: string;\n        };\n      }>(\n        \"/api/migrations/runAction\",\n        async (request): Promise<MigrationResult | SlackConfirmPendingResult> => {\n          const { action, targets, force, forceReason, requestor } = request.body;\n\n          if (action === \"shadow\") {\n            return migrator.runShadowTest();\n          }\n\n          // Slack 승인 체크 (apply 시에만)\n          if (action === \"apply\") {\n            const slackConfirm = new SlackConfirm();\n            const requiresApproval = targets.some((t) => slackConfirm.isTargetRequiresApproval(t));\n\n            // 로컬 DB인 경우 승인 스킵\n            const localHosts = [\"localhost\", \"127.0.0.1\", \"0.0.0.0\", \"::1\"];\n            const isLocalTarget = targets.every((target) => {\n              const targetConfig = Sonamu.dbConfig[target];\n              const host = (targetConfig?.connection as { host?: string })?.host ?? \"localhost\";\n              return localHosts.includes(host.toLowerCase());\n            });\n\n            if (requiresApproval && slackConfirm.isConfigured() && !isLocalTarget) {\n              const { conns } = await migrator.getStatus();\n\n              // 모든 타겟 DB에서 pending인 마이그레이션의 합집합을 구합니다.\n              const pendingMigrations = [\n                ...new Set(\n                  conns\n                    .filter((conn) => targets.includes(conn.connKey as keyof SonamuDBConfig))\n                    .flatMap((conn) => conn.pending),\n                ),\n              ];\n\n              if (pendingMigrations.length > 0) {\n                // 기존 승인 요청 확인\n                const existing = await slackConfirm.getExistingRequest(pendingMigrations);\n\n                if (existing) {\n                  // 기존 요청이 있으면 승인 상태 확인\n                  const { approved, rejected } = await slackConfirm.checkApproval(\n                    existing.channel,\n                    existing.ts,\n                  );\n\n                  if (approved) {\n                    // 승인됨 → 실행\n                    const result = await migrator.runAction(action, targets);\n                    if (result.length > 0) {\n                      await slackConfirm.logExecution(\n                        existing.channel,\n                        existing.ts,\n                        result,\n                        requestor,\n                      );\n                    }\n                    return result;\n                  } else if (rejected) {\n                    throw new BadRequestException(SD(\"sonamu.error.migrationRejected\"));\n                  } else if (force) {\n                    // Force 진행\n                    await slackConfirm.forceApproval(\n                      existing.channel,\n                      existing.ts,\n                      forceReason ?? \"사유 없음\",\n                      requestor,\n                    );\n                    const result = await migrator.runAction(action, targets);\n                    if (result.length > 0) {\n                      await slackConfirm.logExecution(\n                        existing.channel,\n                        existing.ts,\n                        result,\n                        requestor,\n                      );\n                    }\n                    return result;\n                  } else {\n                    // 대기중\n                    return {\n                      type: \"pending\",\n                      channel: existing.channel,\n                      ts: existing.ts,\n                    };\n                  }\n                } else {\n                  // 새 승인 요청 발송\n                  const { channel, ts } = await slackConfirm.postApprovalRequest(\n                    pendingMigrations,\n                    targets,\n                    requestor,\n                  );\n                  await slackConfirm.saveRequest(pendingMigrations, channel, ts);\n\n                  return {\n                    type: \"pending\",\n                    channel,\n                    ts,\n                  };\n                }\n              }\n            }\n          }\n\n          return migrator.runAction(action, targets);\n        },\n      );\n\n      server.post<{\n        Body: {\n          channel: string;\n          ts: string;\n        };\n      }>(\"/api/migrations/checkApproval\", async (request) => {\n        const { channel, ts } = request.body;\n        const slackConfirm = new SlackConfirm();\n\n        if (!slackConfirm.isConfigured()) {\n          return { approved: true, rejected: false };\n        }\n\n        return slackConfirm.checkApproval(channel, ts);\n      });\n\n      server.post<{\n        Body: {\n          channel: string;\n          ts: string;\n          reason: string;\n          requestor?: string;\n        };\n      }>(\"/api/migrations/forceApproval\", async (request) => {\n        const { channel, ts, reason, requestor } = request.body;\n        const slackConfirm = new SlackConfirm();\n\n        if (!slackConfirm.isConfigured()) {\n          throw new BadRequestException(SD(\"sonamu.error.slackConfirmNotConfigured\"));\n        }\n\n        await slackConfirm.forceApproval(channel, ts, reason, requestor);\n        return { success: true };\n      });\n\n      server.post<{\n        Body: {\n          codeNames: string[];\n        };\n      }>(\"/api/migrations/delCodes\", async (request) => {\n        const { codeNames } = request.body;\n        return await migrator.delCodes(codeNames);\n      });\n\n      server.post(\"/api/migrations/generatePreparedCodes\", async (_requestt) => {\n        return await migrator.generatePreparedCodes();\n      });\n\n      server.post<{\n        Body: {\n          entityIds: string[];\n          templateKeys: string[];\n        };\n      }>(\"/api/scaffolding/getStatus\", async (request) => {\n        const { entityIds, templateKeys: _templateKeys } = request.body;\n        if ((entityIds ?? []).length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.entityIdsRequired\"));\n        } else if ((_templateKeys ?? []).length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.templateKeysRequired\"));\n        }\n\n        // sorting\n        entityIds.sort((a, b) => (a < b ? -1 : a > b ? 1 : 0));\n        const templateKeys = TemplateKey.options.filter((tk) => _templateKeys.includes(tk));\n\n        const combinations = entityIds.flatMap((entityId) => {\n          return templateKeys.map((templateKey) => [entityId, templateKey]);\n        });\n\n        const statuses = await Promise.all(\n          combinations.map(async ([entityId, templateKey]) => {\n            const { subPath, fullPath, isExists } = await Sonamu.syncer.checkExistsGenCode(\n              entityId,\n              templateKey as TemplateKey,\n            );\n            return {\n              entityId,\n              templateKey,\n              subPath,\n              fullPath,\n              isExists,\n            };\n          }),\n        );\n        return { statuses };\n      });\n\n      server.post<{\n        Body: {\n          options: {\n            entityId: string;\n            templateKey: string;\n            enumId?: string;\n            overwrite?: boolean;\n          }[];\n        };\n      }>(\"/api/scaffolding/generate\", async (request) => {\n        const { options } = request.body;\n        if (options.length === 0) {\n          throw new BadRequestException(SD(\"sonamu.error.optionsRequired\"));\n        }\n\n        // 1. 모든 템플릿에서 필요한 dict 키를 수집\n        const keys = options.flatMap(({ templateKey }) => {\n          const template = TemplateManager.get(templateKey);\n          return template.getRequiredDictKeys() ?? [];\n        });\n\n        // 2. target별로 ensureDictKeys 호출 (순차 처리)\n        await sonamuDictionary.ensureDictKeys([...new Set(keys)]);\n\n        // 3. 템플릿 생성 (병렬 처리)\n        const result = await Promise.all(\n          options.map(async ({ entityId, templateKey, enumId, overwrite }) => {\n            try {\n              return await Sonamu.syncer.generateTemplate(\n                templateKey as TemplateKey,\n                {\n                  entityId,\n                  enumId,\n                } as {\n                  entityId: string;\n                  enumId?: string;\n                },\n                {\n                  overwrite,\n                },\n              );\n            } catch (e) {\n              if (isSoException(e) && e.statusCode === 541) {\n                return null;\n              } else {\n                console.error(e);\n                throw e;\n              }\n            }\n          }),\n        );\n\n        if (result.filter(nonNullable).length === 0) {\n          throw new ServiceUnavailableException(SD(\"sonamu.error.allFilesGenerated\"));\n        }\n        return result;\n      });\n\n      server.post<{\n        Body: {\n          option: {\n            entityId: string;\n            templateKey: string;\n            enumId?: string;\n          };\n        };\n      }>(\"/api/scaffolding/preview\", async (request): Promise<{ pathAndCodes: PathAndCode[] }> => {\n        const { option } = request.body;\n\n        try {\n          const { templateKey, ...templateOptions } = option;\n          const pathAndCodes = await Sonamu.syncer.renderTemplate(\n            templateKey as TemplateKey,\n            templateOptions,\n          );\n\n          return { pathAndCodes };\n        } catch (e) {\n          console.error(e);\n          throw e;\n        }\n      });\n\n      server.post(\"/api/fixture\", async (request) => {\n        const { sourceDB, targetDB, search, duplicateCheck } = request.body as {\n          sourceDB: keyof SonamuDBConfig;\n          targetDB: keyof SonamuDBConfig;\n          search: FixtureSearchOptions;\n          duplicateCheck?: DuplicateCheckOptions;\n        };\n\n        return FixtureManager.getFixtures(sourceDB, targetDB, search, duplicateCheck);\n      });\n\n      server.post(\"/api/fixture/import\", async (request) => {\n        const { db, fixtures } = request.body as {\n          db: keyof SonamuDBConfig;\n          fixtures: FixtureRecord[];\n        };\n\n        return FixtureManager.insertFixtures(db, fixtures);\n      });\n\n      server.post(\"/api/fixture/addFixtureLoader\", async (request) => {\n        const { code } = request.body as { code: string };\n\n        return FixtureManager.addFixtureLoader(code);\n      });\n\n      server.get(\"/api/i18n/dictionary\", async () => {\n        return sonamuDictionary.getDictionary();\n      });\n\n      server.get(\"/api/i18n/export\", async (_request, reply) => {\n        const { filename, buffer } = await sonamuDictionary.exportToExcel();\n        reply\n          .header(\n            \"Content-Type\",\n            \"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet\",\n          )\n          .header(\"Content-Disposition\", `attachment; filename=\"${filename}\"`)\n          .send(buffer);\n      });\n\n      server.post(\"/api/i18n/import\", async (request) => {\n        const data = await request.file();\n        if (!data) {\n          throw new BadRequestException(SD(\"sonamu.error.fileNotUploaded\"));\n        }\n        const buffer = await data.toBuffer();\n        return sonamuDictionary.importFromExcel(buffer);\n      });\n\n      server.post<{\n        Body: {\n          oldKey: string;\n          newKey: string;\n          source: \"entity\" | \"project\" | \"sonamu\";\n          values: Record<string, string>;\n        };\n      }>(\"/api/i18n/update\", async (request) => {\n        await sonamuDictionary.updateEntry(request.body);\n        return { success: true };\n      });\n\n      server.post<{\n        Body: {\n          key: string;\n          values: Record<string, string>;\n        };\n      }>(\"/api/i18n/create\", async (request) => {\n        await sonamuDictionary.createEntry(request.body);\n        return { success: true };\n      });\n\n      server.post<{\n        Body: {\n          key: string;\n        };\n      }>(\"/api/i18n/delete\", async (request) => {\n        await sonamuDictionary.deleteEntry(request.body.key);\n        return { success: true };\n      });\n\n      server.post<{ Body: { keys: string[] } }>(\"/api/i18n/checkUsage\", async (request) => {\n        return sonamuDictionary.checkUsage(request.body.keys);\n      });\n\n      // Tasks API\n      server.get(\"/api/tasks/status\", async () => {\n        try {\n          Sonamu.workflows;\n          return { active: true };\n        } catch {\n          return { active: false };\n        }\n      });\n\n      server.get(\"/api/tasks/workflowDefinitions\", async () => {\n        const definitions = Sonamu.workflows.workflowDefinitions;\n        return { definitions };\n      });\n\n      server.get<{\n        Querystring: {\n          limit?: string;\n          after?: string;\n          before?: string;\n          order?: \"asc\" | \"desc\";\n          status?: string;\n          workflowName?: string;\n          createdAfter?: string;\n          createdBefore?: string;\n        };\n      }>(\"/api/tasks/workflowRuns\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        const { limit, after, before, order, status, workflowName, createdAfter, createdBefore } =\n          request.query;\n        return backend.listWorkflowRuns({\n          limit: limit ? Number.parseInt(limit, 10) : undefined,\n          after,\n          before,\n          order,\n          status: status ? status.split(\",\") : undefined,\n          workflowName: workflowName || undefined,\n          createdAfter: createdAfter ? new Date(createdAfter) : undefined,\n          createdBefore: createdBefore ? new Date(createdBefore) : undefined,\n        });\n      });\n\n      server.get<{\n        Params: { id: string };\n      }>(\"/api/tasks/workflowRuns/:id\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        const workflowRun = await backend.getWorkflowRun({\n          workflowRunId: request.params.id,\n        });\n        if (!workflowRun) {\n          throw new Error(`Workflow run not found: ${request.params.id}`);\n        }\n        return workflowRun;\n      });\n\n      server.post<{\n        Params: { id: string };\n      }>(\"/api/tasks/workflowRuns/:id/cancel\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        return backend.cancelWorkflowRun({\n          workflowRunId: request.params.id,\n        });\n      });\n\n      server.post<{\n        Params: { id: string };\n      }>(\"/api/tasks/workflowRuns/:id/pause\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        return backend.pauseWorkflowRun({\n          workflowRunId: request.params.id,\n        });\n      });\n\n      server.post<{\n        Params: { id: string };\n      }>(\"/api/tasks/workflowRuns/:id/resume\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        return backend.resumeWorkflowRun({\n          workflowRunId: request.params.id,\n        });\n      });\n\n      server.get<{\n        Params: { id: string };\n        Querystring: {\n          limit?: string;\n          after?: string;\n          before?: string;\n        };\n      }>(\"/api/tasks/workflowRuns/:id/steps\", async (request) => {\n        const backend = Sonamu.workflows.backend;\n        const { limit, after, before } = request.query;\n        return backend.listStepAttempts({\n          workflowRunId: request.params.id,\n          limit: limit ? Number.parseInt(limit, 10) : undefined,\n          after,\n          before,\n        });\n      });\n\n      /**\n       * Health Check API\n       * MCP 도구가 Sonamu 서버를 자동 감지하기 위한 엔드포인트\n       */\n      server.get(\"/api/sonamu/health\", async (request) => {\n        const address = request.server.server.address();\n        const port = address && typeof address === \"object\" ? (address as AddressInfo).port : 0;\n\n        return {\n          ok: true,\n          project: process.cwd().split(\"/\").pop() || \"unknown\",\n          port,\n          timestamp: new Date().toISOString(),\n        };\n      });\n\n      /**\n       * Fixture 생성 API\n       */\n      server.post<{\n        Body: {\n          entity: string;\n          count?: number;\n          overrides?: Record<string, unknown>;\n          targetDb?: \"fixture\" | \"test\";\n        };\n      }>(\"/api/sonamu/fixture/generate\", async (request, reply) => {\n        const { entity, count = 1, overrides, targetDb = \"fixture\" } = request.body;\n\n        // 타겟 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      // ui-web 빌드 파일 서빙\n      const uiDistPath = path.resolve(import.meta.dirname, \"../ui-web\");\n\n      // 정적 파일 서빙: 루트 폴더 전체 (assets, setting.svg 등)\n      server.register(await import(\"@fastify/static\"), {\n        root: uiDistPath,\n        prefix: \"/\",\n        decorateReply: false,\n        wildcard: false,\n      });\n\n      // SPA fallback - 정적 파일이 없는 모든 경로는 index.html로\n      server.get(\"*\", async (_request, reply) => {\n        reply.headers({ \"Content-type\": \"text/html\" }).send(\n          fs\n            .readFileSync(path.resolve(uiDistPath, \"index.html\"))\n            .toString()\n            .replace(\"{{projectName}}\", Sonamu.config.projectName ?? \"UnknownSonamuProject\"),\n        );\n      });\n    },\n    { prefix: \"/sonamu-ui\" },\n  );\n}\n"],"names":["execSync","fs","inflection","path","range","Sonamu","DB","createKnexInstance","SD","sonamuDictionary","EntityManager","BadRequestException","isSoException","ServiceUnavailableException","Migrator","SlackConfirm","TemplateManager","DataExplorer","FixtureGenerator","FixtureManager","BUILT_IN_TYPE_IDS","TemplateKey","nonNullable","setAiApi","sonamuUIApiPlugin","fastify","register","server","migrator","waitForHMRCompleted","fn","waitPromise","Promise","resolve","timeout","setTimeout","handler","clearTimeout","syncer","eventEmitter","off","once","result","get","config","request","entityId","preset","absPath","query","targetPath","entity","names","apiRootPath","filename","parentFs","origin","glossary","Map","getAllIds","title","set","underscore","id","pluralize","props","forEach","prop","has","name","desc","replace","suggested","words","split","combinations","length","flatMap","len","idx","w","slice","join","REPLACED_PREFIX","remainArr","comb","remainStr","includes","map","r","startsWith","toUpperCase","entityIds","flattenSubsetRows","subsetRows","subsetRow","children","sRow","entities","all","getSubsetRows","sort","a","b","aId","parentId","bId","undefined","filter","reload","autoloadTypes","typeIds","projectTypeIds","Object","entries","types","_typeId","zodType","_zod","def","type","typeId","_zodType","builtInTypeIds","allTypeIds","enumIds","keys","enumLabels","post","form","body","createEntity","delEntity","newValues","table","save","subsetKey","fields","fieldsInternal","subsets","subsetsInternal","updated","updatedInternal","at","newProp","createProp","modifyProp","delProp","to","moveProp","indexes","newEnumId","Error","endsWith","active","hidden","enumId","isExists","some","after","before","isReferenced","target","propName","cone","find","p","enumCones","subsetCones","reply","preserveExisting","onlyEmpty","locale","effectiveLocale","i18n","defaultLocale","generateCones","error","message","String","status","success","columns","getTableColumns","getStatus","action","targets","force","forceReason","requestor","runShadowTest","slackConfirm","requiresApproval","t","isTargetRequiresApproval","localHosts","isLocalTarget","every","targetConfig","dbConfig","host","connection","toLowerCase","isConfigured","conns","pendingMigrations","Set","conn","connKey","pending","existing","getExistingRequest","approved","rejected","checkApproval","channel","ts","runAction","logExecution","forceApproval","postApprovalRequest","saveRequest","reason","codeNames","delCodes","_requestt","generatePreparedCodes","templateKeys","_templateKeys","options","tk","templateKey","statuses","subPath","fullPath","checkExistsGenCode","template","getRequiredDictKeys","ensureDictKeys","overwrite","generateTemplate","e","statusCode","console","option","templateOptions","pathAndCodes","renderTemplate","sourceDB","targetDB","search","duplicateCheck","getFixtures","db","fixtures","insertFixtures","code","addFixtureLoader","getDictionary","_request","buffer","exportToExcel","header","send","data","file","toBuffer","importFromExcel","updateEntry","createEntry","deleteEntry","key","checkUsage","workflows","definitions","workflowDefinitions","backend","limit","order","workflowName","createdAfter","createdBefore","listWorkflowRuns","Number","parseInt","Date","workflowRun","getWorkflowRun","workflowRunId","params","cancelWorkflowRun","pauseWorkflowRun","resumeWorkflowRun","listStepAttempts","address","port","ok","project","process","cwd","pop","timestamp","toISOString","count","overrides","targetDb","fixture","test","generator","generateBatch","destroy","strategy","where","fixtureDbConfig","fixtureDb","explorer","explore","includeRelations","maxDepth","sourceDb","getDB","results","importFromSource","imported","targetEntities","tableNames","raw","cleaned","uiDistPath","dirname","root","prefix","decorateReply","wildcard","headers","readFileSync","toString","projectName"],"mappings":"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"}
|
|
1114
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/ui/api.ts"],"sourcesContent":["import { execSync } from \"child_process\";\nimport type { FastifyInstance } from \"fastify\";\nimport fs from \"fs\";\nimport inflection from \"inflection\";\nimport type { AddressInfo } from \"net\";\nimport path from \"path\";\nimport { range } from \"radashi\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { DB, type SonamuDBConfig } from \"../database/db\";\nimport { createKnexInstance } from \"../database/knex\";\nimport { SD } from \"../dict/sd\";\nimport { sonamuDictionary } from \"../dict/sonamu-dictionary\";\nimport type { Entity } from \"../entity/entity\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport {\n  BadRequestException,\n  isSoException,\n  ServiceUnavailableException,\n} from \"../exceptions/so-exceptions\";\nimport { type MigrationResult, Migrator } from \"../migration/migrator\";\nimport { SlackConfirm, type SlackConfirmPendingResult } from \"../migration/slack-confirm\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport { DataExplorer } from \"../testing/data-explorer\";\nimport { FixtureGenerator } from \"../testing/fixture-generator\";\nimport { type DuplicateCheckOptions, FixtureManager } from \"../testing/fixture-manager\";\nimport {\n  BUILT_IN_TYPE_IDS,\n  type Cone,\n  type EntityIndex,\n  type EntityProp,\n  type EntitySubsetRow,\n  type FixtureRecord,\n  type FixtureSearchOptions,\n  type FlattenSubsetRow,\n  type PathAndCode,\n  TemplateKey,\n} from \"../types/types\";\nimport { nonNullable } from \"../utils/utils\";\nimport { setAiApi } from \"./ai-api\";\n\nexport async function sonamuUIApiPlugin(fastify: FastifyInstance) {\n  fastify.register(\n    async (server) => {\n      // migrator\n      const migrator = new Migrator();\n\n      // waitForHMRCompleted\n      async function waitForHMRCompleted<T>(fn: () => Promise<T>): Promise<T> {\n        const waitPromise = new Promise<void>((resolve) => {\n          const 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      // ui-web 빌드 파일 서빙\n      const uiDistPath = path.resolve(import.meta.dirname, \"../ui-web\");\n\n      // 정적 파일 서빙: 루트 폴더 전체 (assets, setting.svg 등)\n      server.register(await import(\"@fastify/static\"), {\n        root: uiDistPath,\n        prefix: \"/\",\n        decorateReply: false,\n        wildcard: false,\n      });\n\n      // SPA fallback - 정적 파일이 없는 모든 경로는 index.html로\n      server.get(\"*\", async (_request, reply) => {\n        reply.headers({ \"Content-type\": \"text/html\" }).send(\n          fs\n            .readFileSync(path.resolve(uiDistPath, \"index.html\"))\n            .toString()\n            .replace(\"{{projectName}}\", Sonamu.config.projectName ?? \"UnknownSonamuProject\"),\n        );\n      });\n    },\n    { prefix: \"/sonamu-ui\" },\n  );\n}\n"],"names":["execSync","fs","inflection","path","range","Sonamu","DB","createKnexInstance","SD","sonamuDictionary","EntityManager","BadRequestException","isSoException","ServiceUnavailableException","Migrator","SlackConfirm","TemplateManager","DataExplorer","FixtureGenerator","FixtureManager","BUILT_IN_TYPE_IDS","TemplateKey","nonNullable","setAiApi","sonamuUIApiPlugin","fastify","register","server","migrator","waitForHMRCompleted","fn","waitPromise","Promise","resolve","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","uiDistPath","dirname","root","prefix","decorateReply","wildcard","headers","readFileSync","toString","projectName"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -136,11 +136,11 @@ export class Migrator {
|
|
|
136
136
|
}
|
|
137
137
|
|
|
138
138
|
const compareDBconn = createKnexInstance(Sonamu.dbConfig[status0conn.connKey]);
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
139
|
+
try {
|
|
140
|
+
return await this.compareMigrations(compareDBconn);
|
|
141
|
+
} finally {
|
|
142
|
+
await compareDBconn.destroy();
|
|
143
|
+
}
|
|
144
144
|
})();
|
|
145
145
|
|
|
146
146
|
Naite.t("migrator:getStatus:preparedCodes", preparedCodes);
|
|
@@ -192,44 +192,45 @@ export class Migrator {
|
|
|
192
192
|
})),
|
|
193
193
|
);
|
|
194
194
|
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
// destroy
|
|
224
|
-
await Promise.all(
|
|
225
|
-
conns.map(({ knex }) => {
|
|
226
|
-
return knex.destroy();
|
|
227
|
-
}),
|
|
228
|
-
);
|
|
195
|
+
try {
|
|
196
|
+
// action
|
|
197
|
+
const result = await (async () => {
|
|
198
|
+
switch (action) {
|
|
199
|
+
case "apply":
|
|
200
|
+
return Promise.all(
|
|
201
|
+
conns.map(async ({ connKey, knex }) => {
|
|
202
|
+
const [batchNo, applied] = await knex.migrate.latest();
|
|
203
|
+
return {
|
|
204
|
+
connKey,
|
|
205
|
+
batchNo,
|
|
206
|
+
applied, // 이번 latest 호출로 인해 "up"이 적용된 마이그레이션 이름(e.g. "20251124233557_create__companies.ts")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L560
|
|
207
|
+
};
|
|
208
|
+
}),
|
|
209
|
+
);
|
|
210
|
+
case "rollback":
|
|
211
|
+
return Promise.all(
|
|
212
|
+
conns.map(async ({ connKey, knex }) => {
|
|
213
|
+
const [batchNo, applied] = await knex.migrate.rollback();
|
|
214
|
+
return {
|
|
215
|
+
connKey,
|
|
216
|
+
batchNo,
|
|
217
|
+
applied, // 이번 rollback 호출로 인해 "down"이 적용된(=롤백된) 마이그레이션 이름(e.g. "20251124233557_create__companies.ts")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L611
|
|
218
|
+
};
|
|
219
|
+
}),
|
|
220
|
+
);
|
|
221
|
+
}
|
|
222
|
+
})();
|
|
229
223
|
|
|
230
|
-
|
|
224
|
+
Naite.t("migrator:runAction:result", result);
|
|
231
225
|
|
|
232
|
-
|
|
226
|
+
return result;
|
|
227
|
+
} finally {
|
|
228
|
+
await Promise.all(
|
|
229
|
+
conns.map(({ knex }) => {
|
|
230
|
+
return knex.destroy();
|
|
231
|
+
}),
|
|
232
|
+
);
|
|
233
|
+
}
|
|
233
234
|
}
|
|
234
235
|
|
|
235
236
|
/**
|
|
@@ -49,6 +49,11 @@ axios.defaults.transformResponse = [
|
|
|
49
49
|
},
|
|
50
50
|
];
|
|
51
51
|
|
|
52
|
+
axios.interceptors.request.use((config) => {
|
|
53
|
+
config.headers["Accept-Language"] = getCurrentLocale();
|
|
54
|
+
return config;
|
|
55
|
+
});
|
|
56
|
+
|
|
52
57
|
export async function fetch(options: AxiosRequestConfig) {
|
|
53
58
|
try {
|
|
54
59
|
const res = await axios({
|
package/src/ui/api.ts
CHANGED
|
@@ -47,16 +47,16 @@ export async function sonamuUIApiPlugin(fastify: FastifyInstance) {
|
|
|
47
47
|
// waitForHMRCompleted
|
|
48
48
|
async function waitForHMRCompleted<T>(fn: () => Promise<T>): Promise<T> {
|
|
49
49
|
const waitPromise = new Promise<void>((resolve) => {
|
|
50
|
-
const timeout = setTimeout(() => {
|
|
51
|
-
resolve();
|
|
52
|
-
}, 1500);
|
|
53
|
-
|
|
54
50
|
const handler = () => {
|
|
55
51
|
clearTimeout(timeout);
|
|
56
|
-
Sonamu.syncer.eventEmitter.off("onHMRCompleted", handler);
|
|
57
52
|
resolve();
|
|
58
53
|
};
|
|
59
54
|
|
|
55
|
+
const timeout = setTimeout(() => {
|
|
56
|
+
Sonamu.syncer.eventEmitter.off("onHMRCompleted", handler);
|
|
57
|
+
resolve();
|
|
58
|
+
}, 1500);
|
|
59
|
+
|
|
60
60
|
Sonamu.syncer.eventEmitter.once("onHMRCompleted", handler);
|
|
61
61
|
});
|
|
62
62
|
|