sonamu 0.9.4 → 0.9.5
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/ai/providers/rtzr/utils.js +2 -2
- package/dist/database/upsert-builder.js +4 -4
- package/dist/dict/sonamu-dictionary.js +2 -2
- package/dist/migration/code-generation.d.ts.map +1 -1
- package/dist/migration/code-generation.js +2 -3
- package/dist/testing/data-explorer.d.ts.map +1 -1
- package/dist/testing/data-explorer.js +5 -3
- package/dist/ui/api.d.ts.map +1 -1
- package/dist/ui/api.js +3 -2
- package/dist/ui-web/assets/index-D4rYm-Xz.css +1 -0
- package/dist/ui-web/assets/{index-C5KUjXm0.js → index-DzZ7vBk4.js} +4 -4
- package/dist/ui-web/index.html +2 -2
- package/package.json +4 -4
- package/src/ai/providers/rtzr/utils.ts +1 -1
- package/src/database/upsert-builder.ts +3 -3
- package/src/dict/sonamu-dictionary.ts +1 -1
- package/src/migration/code-generation.ts +1 -6
- package/src/shared/app.shared.ts.txt +58 -3
- package/src/shared/web.shared.ts.txt +58 -3
- package/src/testing/data-explorer.ts +3 -2
- package/src/ui/api.ts +10 -1
- package/dist/ui-web/assets/index-Dr8pRJC_.css +0 -1
|
@@ -9,7 +9,7 @@ function handleFetchError({ error, url, requestBodyValues }) {
|
|
|
9
9
|
}
|
|
10
10
|
if (error instanceof TypeError && FETCH_FAILED_ERROR_MESSAGES.includes(error.message.toLowerCase())) {
|
|
11
11
|
const cause = error.cause;
|
|
12
|
-
if (cause
|
|
12
|
+
if (cause) {
|
|
13
13
|
return new APICallError({
|
|
14
14
|
message: `Cannot connect to API: ${cause.message}`,
|
|
15
15
|
cause,
|
|
@@ -85,4 +85,4 @@ const getFromApi = async ({ url, headers = {}, successfulResponseHandler, failed
|
|
|
85
85
|
|
|
86
86
|
//#endregion
|
|
87
87
|
export { getFromApi, handleFetchError };
|
|
88
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJuYW1lcyI6WyJlcnJvckluZm9ybWF0aW9uOiB7XG4gICAgICAgIHZhbHVlOiBFcnJvcjtcbiAgICAgICAgcmVzcG9uc2VIZWFkZXJzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPiB8IHVuZGVmaW5lZDtcbiAgICAgIH0iXSwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvYWkvcHJvdmlkZXJzL3J0enIvdXRpbHMudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQVBJQ2FsbEVycm9yIH0gZnJvbSBcIkBhaS1zZGsvcHJvdmlkZXJcIjtcbmltcG9ydCB7IGV4dHJhY3RSZXNwb25zZUhlYWRlcnMsIGlzQWJvcnRFcnJvciwgd2l0aFVzZXJBZ2VudFN1ZmZpeCB9IGZyb20gXCJAYWktc2RrL3Byb3ZpZGVyLXV0aWxzXCI7XG5pbXBvcnQgeyB0eXBlIEZldGNoRnVuY3Rpb24sIHR5cGUgUmVzcG9uc2VIYW5kbGVyIH0gZnJvbSBcIkBhaS1zZGsvcHJvdmlkZXItdXRpbHNcIjtcblxuY29uc3QgRkVUQ0hfRkFJTEVEX0VSUk9SX01FU1NBR0VTID0gW1wiZmV0Y2ggZmFpbGVkXCIsIFwiZmFpbGVkIHRvIGZldGNoXCJdO1xuXG5leHBvcnQgZnVuY3Rpb24gaGFuZGxlRmV0Y2hFcnJvcih7XG4gIGVycm9yLFxuICB1cmwsXG4gIHJlcXVlc3RCb2R5VmFsdWVzLFxufToge1xuICBlcnJvcjogdW5rbm93bjtcbiAgdXJsOiBzdHJpbmc7XG4gIHJlcXVlc3RCb2R5VmFsdWVzOiB1bmtub3duO1xufSkge1xuICBpZiAoaXNBYm9ydEVycm9yKGVycm9yKSkge1xuICAgIHJldHVybiBlcnJvcjtcbiAgfVxuXG4gIGlmIChcbiAgICBlcnJvciBpbnN0YW5jZW9mIFR5cGVFcnJvciAmJlxuICAgIEZFVENIX0ZBSUxFRF9FUlJPUl9NRVNTQUdFUy5pbmNsdWRlcyhlcnJvci5tZXNzYWdlLnRvTG93ZXJDYXNlKCkpXG4gICkge1xuICAgIGNvbnN0IGNhdXNlID0gKGVycm9yIGFzIHsgY2F1c2U/
|
|
88
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJuYW1lcyI6WyJlcnJvckluZm9ybWF0aW9uOiB7XG4gICAgICAgIHZhbHVlOiBFcnJvcjtcbiAgICAgICAgcmVzcG9uc2VIZWFkZXJzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPiB8IHVuZGVmaW5lZDtcbiAgICAgIH0iXSwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvYWkvcHJvdmlkZXJzL3J0enIvdXRpbHMudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQVBJQ2FsbEVycm9yIH0gZnJvbSBcIkBhaS1zZGsvcHJvdmlkZXJcIjtcbmltcG9ydCB7IGV4dHJhY3RSZXNwb25zZUhlYWRlcnMsIGlzQWJvcnRFcnJvciwgd2l0aFVzZXJBZ2VudFN1ZmZpeCB9IGZyb20gXCJAYWktc2RrL3Byb3ZpZGVyLXV0aWxzXCI7XG5pbXBvcnQgeyB0eXBlIEZldGNoRnVuY3Rpb24sIHR5cGUgUmVzcG9uc2VIYW5kbGVyIH0gZnJvbSBcIkBhaS1zZGsvcHJvdmlkZXItdXRpbHNcIjtcblxuY29uc3QgRkVUQ0hfRkFJTEVEX0VSUk9SX01FU1NBR0VTID0gW1wiZmV0Y2ggZmFpbGVkXCIsIFwiZmFpbGVkIHRvIGZldGNoXCJdO1xuXG5leHBvcnQgZnVuY3Rpb24gaGFuZGxlRmV0Y2hFcnJvcih7XG4gIGVycm9yLFxuICB1cmwsXG4gIHJlcXVlc3RCb2R5VmFsdWVzLFxufToge1xuICBlcnJvcjogdW5rbm93bjtcbiAgdXJsOiBzdHJpbmc7XG4gIHJlcXVlc3RCb2R5VmFsdWVzOiB1bmtub3duO1xufSkge1xuICBpZiAoaXNBYm9ydEVycm9yKGVycm9yKSkge1xuICAgIHJldHVybiBlcnJvcjtcbiAgfVxuXG4gIGlmIChcbiAgICBlcnJvciBpbnN0YW5jZW9mIFR5cGVFcnJvciAmJlxuICAgIEZFVENIX0ZBSUxFRF9FUlJPUl9NRVNTQUdFUy5pbmNsdWRlcyhlcnJvci5tZXNzYWdlLnRvTG93ZXJDYXNlKCkpXG4gICkge1xuICAgIGNvbnN0IGNhdXNlID0gKGVycm9yIGFzIHsgY2F1c2U/OiBFcnJvciB9KS5jYXVzZTtcblxuICAgIGlmIChjYXVzZSkge1xuICAgICAgcmV0dXJuIG5ldyBBUElDYWxsRXJyb3Ioe1xuICAgICAgICBtZXNzYWdlOiBgQ2Fubm90IGNvbm5lY3QgdG8gQVBJOiAke2NhdXNlLm1lc3NhZ2V9YCxcbiAgICAgICAgY2F1c2UsXG4gICAgICAgIHVybCxcbiAgICAgICAgcmVxdWVzdEJvZHlWYWx1ZXMsXG4gICAgICAgIGlzUmV0cnlhYmxlOiB0cnVlLCAvLyByZXRyeSB3aGVuIG5ldHdvcmsgZXJyb3JcbiAgICAgIH0pO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiBlcnJvcjtcbn1cblxuLy8gdXNlIGZ1bmN0aW9uIHRvIGFsbG93IGZvciBtb2NraW5nIGluIHRlc3RzOlxuY29uc3QgZ2V0T3JpZ2luYWxGZXRjaCA9ICgpID0+IGdsb2JhbFRoaXMuZmV0Y2g7XG5cbmV4cG9ydCBjb25zdCBnZXRGcm9tQXBpID0gYXN5bmMgPFQ+KHtcbiAgdXJsLFxuICBoZWFkZXJzID0ge30sXG4gIHN1Y2Nlc3NmdWxSZXNwb25zZUhhbmRsZXIsXG4gIGZhaWxlZFJlc3BvbnNlSGFuZGxlcixcbiAgYWJvcnRTaWduYWwsXG4gIGZldGNoID0gZ2V0T3JpZ2luYWxGZXRjaCgpLFxufToge1xuICB1cmw6IHN0cmluZztcbiAgaGVhZGVycz86IFJlY29yZDxzdHJpbmcsIHN0cmluZyB8IHVuZGVmaW5lZD47XG4gIGZhaWxlZFJlc3BvbnNlSGFuZGxlcjogUmVzcG9uc2VIYW5kbGVyPEVycm9yPjtcbiAgc3VjY2Vzc2Z1bFJlc3BvbnNlSGFuZGxlcjogUmVzcG9uc2VIYW5kbGVyPFQ+O1xuICBhYm9ydFNpZ25hbD86IEFib3J0U2lnbmFsO1xuICBmZXRjaD86IEZldGNoRnVuY3Rpb247XG59KSA9PiB7XG4gIHRyeSB7XG4gICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBmZXRjaCh1cmwsIHtcbiAgICAgIG1ldGhvZDogXCJHRVRcIixcbiAgICAgIGhlYWRlcnM6IHdpdGhVc2VyQWdlbnRTdWZmaXgoaGVhZGVycywgYGNhcnRhbm92YS1haS9ydHpyLWFwaS1wcm92aWRlcmApLFxuICAgICAgc2lnbmFsOiBhYm9ydFNpZ25hbCxcbiAgICB9KTtcblxuICAgIGNvbnN0IHJlc3BvbnNlSGVhZGVycyA9IGV4dHJhY3RSZXNwb25zZUhlYWRlcnMocmVzcG9uc2UpO1xuXG4gICAgaWYgKCFyZXNwb25zZS5vaykge1xuICAgICAgbGV0IGVycm9ySW5mb3JtYXRpb246IHtcbiAgICAgICAgdmFsdWU6IEVycm9yO1xuICAgICAgICByZXNwb25zZUhlYWRlcnM/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+IHwgdW5kZWZpbmVkO1xuICAgICAgfTtcblxuICAgICAgdHJ5IHtcbiAgICAgICAgZXJyb3JJbmZvcm1hdGlvbiA9IGF3YWl0IGZhaWxlZFJlc3BvbnNlSGFuZGxlcih7XG4gICAgICAgICAgcmVzcG9uc2UsXG4gICAgICAgICAgdXJsLFxuICAgICAgICAgIHJlcXVlc3RCb2R5VmFsdWVzOiB7fSxcbiAgICAgICAgfSk7XG4gICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICBpZiAoaXNBYm9ydEVycm9yKGVycm9yKSB8fCBBUElDYWxsRXJyb3IuaXNJbnN0YW5jZShlcnJvcikpIHtcbiAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgfVxuXG4gICAgICAgIHRocm93IG5ldyBBUElDYWxsRXJyb3Ioe1xuICAgICAgICAgIG1lc3NhZ2U6IFwiRmFpbGVkIHRvIHByb2Nlc3MgZXJyb3IgcmVzcG9uc2VcIixcbiAgICAgICAgICBjYXVzZTogZXJyb3IsXG4gICAgICAgICAgc3RhdHVzQ29kZTogcmVzcG9uc2Uuc3RhdHVzLFxuICAgICAgICAgIHVybCxcbiAgICAgICAgICByZXNwb25zZUhlYWRlcnMsXG4gICAgICAgICAgcmVxdWVzdEJvZHlWYWx1ZXM6IHt9LFxuICAgICAgICB9KTtcbiAgICAgIH1cblxuICAgICAgdGhyb3cgZXJyb3JJbmZvcm1hdGlvbi52YWx1ZTtcbiAgICB9XG5cbiAgICB0cnkge1xuICAgICAgcmV0dXJuIGF3YWl0IHN1Y2Nlc3NmdWxSZXNwb25zZUhhbmRsZXIoe1xuICAgICAgICByZXNwb25zZSxcbiAgICAgICAgdXJsLFxuICAgICAgICByZXF1ZXN0Qm9keVZhbHVlczoge30sXG4gICAgICB9KTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgaWYgKGVycm9yIGluc3RhbmNlb2YgRXJyb3IpIHtcbiAgICAgICAgaWYgKGlzQWJvcnRFcnJvcihlcnJvcikgfHwgQVBJQ2FsbEVycm9yLmlzSW5zdGFuY2UoZXJyb3IpKSB7XG4gICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgdGhyb3cgbmV3IEFQSUNhbGxFcnJvcih7XG4gICAgICAgIG1lc3NhZ2U6IFwiRmFpbGVkIHRvIHByb2Nlc3Mgc3VjY2Vzc2Z1bCByZXNwb25zZVwiLFxuICAgICAgICBjYXVzZTogZXJyb3IsXG4gICAgICAgIHN0YXR1c0NvZGU6IHJlc3BvbnNlLnN0YXR1cyxcbiAgICAgICAgdXJsLFxuICAgICAgICByZXNwb25zZUhlYWRlcnMsXG4gICAgICAgIHJlcXVlc3RCb2R5VmFsdWVzOiB7fSxcbiAgICAgIH0pO1xuICAgIH1cbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICB0aHJvdyBoYW5kbGVGZXRjaEVycm9yKHsgZXJyb3IsIHVybCwgcmVxdWVzdEJvZHlWYWx1ZXM6IHt9IH0pO1xuICB9XG59O1xuIl0sIm1hcHBpbmdzIjoiOzs7O0FBSUEsTUFBTSw4QkFBOEIsQ0FBQyxnQkFBZ0Isa0JBQWtCO0FBRXZFLFNBQWdCLGlCQUFpQixFQUMvQixPQUNBLEtBQ0EscUJBS0M7QUFDRCxLQUFJLGFBQWEsTUFBTSxFQUFFO0FBQ3ZCLFNBQU87O0FBR1QsS0FDRSxpQkFBaUIsYUFDakIsNEJBQTRCLFNBQVMsTUFBTSxRQUFRLGFBQWEsQ0FBQyxFQUNqRTtFQUNBLE1BQU0sUUFBUyxNQUE0QjtBQUUzQyxNQUFJLE9BQU87QUFDVCxVQUFPLElBQUksYUFBYTtJQUN0QixTQUFTLDBCQUEwQixNQUFNO0lBQ3pDO0lBQ0E7SUFDQTtJQUNBLGFBQWE7SUFDZCxDQUFDOzs7QUFJTixRQUFPOztBQUlULE1BQU0seUJBQXlCLFdBQVc7QUFFMUMsTUFBYSxhQUFhLE9BQVUsRUFDbEMsS0FDQSxVQUFVLEVBQUUsRUFDWiwyQkFDQSx1QkFDQSxhQUNBLFFBQVEsa0JBQWtCLE9BUXRCO0FBQ0osS0FBSTtFQUNGLE1BQU0sV0FBVyxNQUFNLE1BQU0sS0FBSztHQUNoQyxRQUFRO0dBQ1IsU0FBUyxvQkFBb0IsU0FBUyxpQ0FBaUM7R0FDdkUsUUFBUTtHQUNULENBQUM7RUFFRixNQUFNLGtCQUFrQix1QkFBdUIsU0FBUztBQUV4RCxNQUFJLENBQUMsU0FBUyxJQUFJO0dBQ2hCLElBQUlBO0FBS0osT0FBSTtBQUNGLHVCQUFtQixNQUFNLHNCQUFzQjtLQUM3QztLQUNBO0tBQ0EsbUJBQW1CLEVBQUU7S0FDdEIsQ0FBQztZQUNLLE9BQU87QUFDZCxRQUFJLGFBQWEsTUFBTSxJQUFJLGFBQWEsV0FBVyxNQUFNLEVBQUU7QUFDekQsV0FBTTs7QUFHUixVQUFNLElBQUksYUFBYTtLQUNyQixTQUFTO0tBQ1QsT0FBTztLQUNQLFlBQVksU0FBUztLQUNyQjtLQUNBO0tBQ0EsbUJBQW1CLEVBQUU7S0FDdEIsQ0FBQzs7QUFHSixTQUFNLGlCQUFpQjs7QUFHekIsTUFBSTtBQUNGLFVBQU8sTUFBTSwwQkFBMEI7SUFDckM7SUFDQTtJQUNBLG1CQUFtQixFQUFFO0lBQ3RCLENBQUM7V0FDSyxPQUFPO0FBQ2QsT0FBSSxpQkFBaUIsT0FBTztBQUMxQixRQUFJLGFBQWEsTUFBTSxJQUFJLGFBQWEsV0FBVyxNQUFNLEVBQUU7QUFDekQsV0FBTTs7O0FBSVYsU0FBTSxJQUFJLGFBQWE7SUFDckIsU0FBUztJQUNULE9BQU87SUFDUCxZQUFZLFNBQVM7SUFDckI7SUFDQTtJQUNBLG1CQUFtQixFQUFFO0lBQ3RCLENBQUM7O1VBRUcsT0FBTztBQUNkLFFBQU0saUJBQWlCO0dBQUU7R0FBTztHQUFLLG1CQUFtQixFQUFFO0dBQUUsQ0FBQyJ9
|
|
@@ -193,7 +193,7 @@ var init_upsert_builder = __esmMin((() => {
|
|
|
193
193
|
total: levelChunks.length
|
|
194
194
|
});
|
|
195
195
|
const originalUuids = dataChunk.map((r) => r.uuid);
|
|
196
|
-
const dataForDb = dataChunk.map(({ uuid, ...rest }) => rest);
|
|
196
|
+
const dataForDb = dataChunk.map(({ uuid: _, ...rest }) => rest);
|
|
197
197
|
let resultRows;
|
|
198
198
|
if (mode === "insert") {
|
|
199
199
|
resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);
|
|
@@ -205,7 +205,7 @@ var init_upsert_builder = __esmMin((() => {
|
|
|
205
205
|
const conditions = [];
|
|
206
206
|
for (const row of rowsWithoutId) {
|
|
207
207
|
const values = columns.map((col) => row[col]);
|
|
208
|
-
if (!values.some((v) => v
|
|
208
|
+
if (!values.some((v) => v === null || v === undefined)) {
|
|
209
209
|
conditions.push(values);
|
|
210
210
|
}
|
|
211
211
|
}
|
|
@@ -287,7 +287,7 @@ var init_upsert_builder = __esmMin((() => {
|
|
|
287
287
|
const cleanOrphans = options.cleanOrphans;
|
|
288
288
|
const fkColumns = isArray(cleanOrphans) ? cleanOrphans : [cleanOrphans];
|
|
289
289
|
const fkConditions = fkColumns.map((fkCol) => {
|
|
290
|
-
const fkValues = [...new Set(table.rows.map((row) => row[fkCol]).filter(
|
|
290
|
+
const fkValues = [...new Set(table.rows.map((row) => row[fkCol]).filter(Boolean))];
|
|
291
291
|
return {
|
|
292
292
|
column: fkCol,
|
|
293
293
|
values: fkValues
|
|
@@ -406,4 +406,4 @@ var init_upsert_builder = __esmMin((() => {
|
|
|
406
406
|
//#endregion
|
|
407
407
|
init_upsert_builder();
|
|
408
408
|
export { UpsertBuilder, init_upsert_builder, isRefField };
|
|
409
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"upsert-builder.js","names":["result: UBRef","table","allIds: (number | string)[]","resultRows: { id: number | string; [key: string]: unknown }[]","conditions: unknown[][]","levels: Record<string, unknown>[][]","currentLevel: Record<string, unknown>[]","nextPending: Record<string, unknown>[]"],"sources":["../../src/database/upsert-builder.ts"],"sourcesContent":["import { randomUUID } from \"crypto\";\n\nimport { getLogger } from \"@logtape/logtape\";\nimport { type Knex } from \"knex\";\nimport { cluster, isArray, unique } from \"radashi\";\n\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport {\n  type DatabaseForeignKeys,\n  type DatabaseSchemaExtend,\n  type EntityIndex,\n} from \"../types/types\";\nimport { assertDefined, nonNullable } from \"../utils/utils\";\nimport { batchUpdate } from \"./_batch_update\";\nimport { type RowWithId } from \"./_batch_update\";\nimport { type ColumnKeys, type ForeignKeyColumns, type IdType, type TableName } from \"./puri.types\";\n\nconst logger = getLogger([\"sonamu\", \"internal\", \"upsert-builder\"]);\n\n/**\n * FK 타입 추론을 위해 DatabaseForeignKeys export\n * (module augmentation 자동 로드 보장)\n */\nexport type { DatabaseForeignKeys };\n\ntype InheritableColumns<TTable extends TableName<DatabaseSchemaExtend>> =\n  TTable extends keyof DatabaseSchemaExtend ? ColumnKeys<DatabaseSchemaExtend[TTable]> : never;\n\n// 테이블 데이터 타입\ntype TableData = {\n  references: Set<string>;\n  rows: Record<string, unknown>[];\n  uniqueIndexes: EntityIndex[];\n  uniquesMap: Map<string, string>;\n  jsonColumns: string[];\n};\n\n// 참조 필드 타입\nexport type UBRef = {\n  uuid: string;\n  of: string;\n  use?: string;\n};\n\n// upsert 옵션\nexport type UpsertOptions<TTable extends TableName<DatabaseSchemaExtend>> = {\n  chunkSize?: number;\n  cleanOrphans?: ForeignKeyColumns<TTable> | ForeignKeyColumns<TTable>[];\n  inherit?: InheritableColumns<TTable>[];\n};\n\n// insertOnly 옵션\nexport type InsertOnlyOptions = {\n  chunkSize?: number;\n};\n\nexport function isRefField(field: unknown): field is UBRef {\n  return (\n    field !== undefined &&\n    field !== null &&\n    (field as UBRef)?.of !== undefined &&\n    (field as UBRef)?.uuid !== undefined\n  );\n}\n\nexport class UpsertBuilder {\n  tables: Map<string, TableData>;\n  constructor() {\n    this.tables = new Map();\n  }\n\n  getTable(tableName: string): TableData {\n    const table = this.tables.get(tableName);\n    if (table) {\n      return table;\n    }\n\n    const tableSpec = (() => {\n      try {\n        return EntityManager.getTableSpec(tableName);\n      } catch {\n        return null;\n      }\n    })();\n\n    const tableData = {\n      references: new Set<string>(),\n      rows: [],\n      uniqueIndexes: tableSpec?.uniqueIndexes ?? [],\n      uniquesMap: new Map<string, string>(),\n      jsonColumns: tableSpec?.jsonColumns ?? [],\n    };\n    this.tables.set(tableName, tableData);\n    return tableData;\n  }\n\n  hasTable(tableName: string): boolean {\n    return this.tables.has(tableName);\n  }\n\n  register<T extends string>(\n    tableName: string,\n    row: {\n      [key in T]?: UBRef | string | number | boolean | bigint | null | object | unknown;\n    },\n  ): UBRef {\n    const table = this.getTable(tableName);\n\n    // 해당 테이블의 unique 인덱스를 순회하며 키 생성\n    const uniqueKeys = table.uniqueIndexes\n      .map((unqIndex) => {\n        const uniqueKeyArray = unqIndex.columns.map((unqCol) => {\n          const val = row[unqCol.name as keyof typeof row];\n          if (isRefField(val)) {\n            return val.uuid;\n          } else {\n            return row[unqCol.name as keyof typeof row] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입\n          }\n        });\n\n        // 값이 모두 null인 경우 키 생성 패스\n        if (uniqueKeyArray.length === 0) {\n          return null;\n        }\n        return uniqueKeyArray.join(\"---delimiter--\");\n      })\n      .filter(nonNullable);\n\n    // uuid 생성 로직\n    const { uuid, isReused } = (() => {\n      // 키를 순회하여 이미 존재하는 키가 있는지 확인\n      if (uniqueKeys.length > 0) {\n        for (const uniqueKey of uniqueKeys) {\n          if (table.uniquesMap.has(uniqueKey)) {\n            return {\n              uuid: assertDefined(table.uniquesMap.get(uniqueKey), \"Unique key not found\"),\n              isReused: true,\n            };\n          }\n        }\n      }\n\n      // 찾을 수 없는 경우 생성\n      return { uuid: randomUUID(), isReused: false };\n    })();\n\n    // 모든 유니크키에 대해 유니크맵에 uuid 저장\n    if (uniqueKeys.length > 0) {\n      for (const uniqueKey of uniqueKeys) {\n        table.uniquesMap.set(uniqueKey, uuid);\n      }\n    }\n\n    // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가\n    // 이 정보를 나중에 치환할 때 사용\n    row = Object.fromEntries(\n      Object.entries(row).map(([rowKey, rowValue]) => {\n        if (isRefField(rowValue)) {\n          rowValue.use ??= \"id\";\n          table.references.add(`${rowValue.of}.${rowValue.use}`);\n          return [rowKey, rowValue];\n        } else if (table.jsonColumns.includes(rowKey) && rowValue !== null) {\n          // JSON 컬럼인 경우 JSON.stringify 처리 (Knex는 JSON 타입을 지원하지 않음)\n          return [rowKey, JSON.stringify(rowValue)];\n        } else if (isArray(rowValue)) {\n          // 배열은 그대로 저장\n          return [rowKey, rowValue];\n        } else {\n          return [rowKey, rowValue];\n        }\n      }),\n    ) as { [key in T]?: unknown };\n\n    table.rows.push({\n      uuid,\n      ...row,\n    });\n\n    const result: UBRef = {\n      of: tableName,\n      uuid: (row as { uuid?: string }).uuid ?? uuid,\n    };\n\n    Naite.t(\"puri:ub-register\", {\n      tableName,\n      uuid: result.uuid,\n      isUuidReused: isReused,\n      row,\n    });\n\n    return result;\n  }\n\n  async upsert<TTable extends TableName<DatabaseSchemaExtend>>(\n    wdb: Knex,\n    tableName: TTable,\n    options?: UpsertOptions<TTable>,\n  ): Promise<IdType<DatabaseSchemaExtend, TTable>[]> {\n    return this.upsertOrInsert(wdb, tableName, \"upsert\", options);\n  }\n\n  async insertOnly<TTable extends TableName<DatabaseSchemaExtend>>(\n    wdb: Knex,\n    tableName: TTable,\n    options?: InsertOnlyOptions,\n  ): Promise<IdType<DatabaseSchemaExtend, TTable>[]> {\n    return this.upsertOrInsert(wdb, tableName, \"insert\", options);\n  }\n\n  async upsertOrInsert<TTable extends TableName<DatabaseSchemaExtend>>(\n    wdb: Knex,\n    tableName: TTable,\n    mode: \"upsert\" | \"insert\",\n    options?: UpsertOptions<TTable>,\n  ): Promise<IdType<DatabaseSchemaExtend, TTable>[]> {\n    if (!this.hasTable(tableName)) {\n      return [];\n    }\n\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);\n    } else if (table.rows.length === 0) {\n      throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);\n    }\n\n    if (\n      table.rows.some((row) =>\n        Object.entries(row).some(([, value]) => isRefField(value) && value.of !== tableName),\n      )\n    ) {\n      throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);\n    }\n\n    // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출\n    const { references, refTables } = Array.from(this.tables).reduce(\n      (r, [, table]) => {\n        const reference = Array.from(table.references.values()).find((ref) =>\n          ref.includes(`${tableName}.`),\n        );\n        if (reference) {\n          r.references.push(reference);\n          r.refTables.push(table);\n        }\n\n        return r;\n      },\n      {\n        references: [] as string[],\n        refTables: [] as TableData[],\n      },\n    );\n    const extractFields = unique(references)\n      .map((reference) => reference.split(\".\")[1])\n      .filter((field): field is string => field !== undefined);\n\n    // 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)\n    const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);\n\n    if (hasCircular) {\n      throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);\n    }\n\n    const uuidMap = new Map<string, unknown>();\n    const allIds: (number | string)[] = [];\n\n    // 레벨별로 순차 처리\n    for (let levelIdx = 0; levelIdx < levels.length; levelIdx++) {\n      const levelRows = levels[levelIdx];\n      logger.debug(\"Processing Query Level: {current} / {total}\", {\n        current: levelIdx + 1,\n        total: levels.length,\n      });\n\n      // 이전 레벨에서 얻은 ID로 자기 참조 해결\n      const resolvedRows = levelRows.map((row) => {\n        const resolved = { ...row };\n        for (const [key, value] of Object.entries(row)) {\n          if (isRefField(value) && value.of === tableName) {\n            const parent = uuidMap.get(value.uuid);\n\n            if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);\n\n            resolved[key] = (parent as Record<string, unknown>)[value.use ?? \"id\"];\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: value.of, uuid: value.uuid, use: value.use ?? \"id\" },\n              to: resolved[key],\n            });\n          }\n        }\n        return resolved;\n      });\n\n      // 현재 레벨 upsert\n      const chunkSize = options?.chunkSize;\n      const levelChunks = chunkSize ? cluster(resolvedRows, chunkSize) : [resolvedRows];\n      const selectFields = unique([\"id\", ...extractFields]);\n\n      for (let index = 0; index < levelChunks.length; index++) {\n        const dataChunk = levelChunks[index];\n        if (dataChunk.length === 0) continue;\n        logger.debug(\"Processing Chunk: {current} / {total}\", {\n          current: index + 1,\n          total: levelChunks.length,\n        });\n\n        // uuid를 별도로 보관하고, DB에 저장할 데이터에서 제거\n        const originalUuids = dataChunk.map((r) => r.uuid as string);\n        const dataForDb = dataChunk.map(({ uuid, ...rest }) => rest);\n\n        let resultRows: { id: number | string; [key: string]: unknown }[];\n\n        if (mode === \"insert\") {\n          // INSERT 모드 - RETURNING 사용\n          resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);\n        } else {\n          // UPSERT 모드 - id 없는 row들의 id를 사전 조회로 채우기\n          const rowsWithoutId = dataForDb.filter((row) => !row.id);\n\n          if (rowsWithoutId.length > 0 && table.uniqueIndexes.length > 0) {\n            // 모든 uniqueIndexes로 기존 레코드 조회\n            for (const uniqueIndex of table.uniqueIndexes) {\n              const columns = uniqueIndex.columns.map((c) => c.name);\n\n              // 조회할 조건들 추출 (각 row의 unique 컬럼 값들)\n              const conditions: unknown[][] = [];\n              for (const row of rowsWithoutId) {\n                const values = columns.map((col) => row[col]);\n                // null이 포함된 조건은 제외 (PostgreSQL UNIQUE는 NULL 무시)\n                if (!values.some((v) => v == null)) {\n                  conditions.push(values);\n                }\n              }\n\n              if (conditions.length === 0) continue;\n\n              // 배치 SELECT\n              const existingRows = (await wdb(tableName)\n                .whereIn(columns, conditions as Record<string, unknown>[][])\n                .select(\"id\", ...columns)) as Record<string, unknown>[];\n\n              // Map 생성: unique 컬럼 조합 → id\n              const existingMap = new Map<string, number | string>();\n              for (const existing of existingRows) {\n                const key = columns\n                  .map((col) => String(existing[col] ?? \"\"))\n                  .join(\"---delimiter---\");\n                const id = existing.id;\n                if (typeof id === \"number\" || typeof id === \"string\") {\n                  existingMap.set(key, id);\n                }\n              }\n\n              // id 없는 row들에 매칭되는 id 채우기\n              for (const row of rowsWithoutId) {\n                if (row.id) continue; // 이미 다른 uniqueIndex에서 채워진 경우 스킵\n\n                const key = columns.map((col) => String(row[col] ?? \"\")).join(\"---delimiter---\");\n                const existingId = existingMap.get(key);\n\n                if (existingId) {\n                  row.id = existingId;\n                }\n              }\n            }\n          }\n\n          // onConflict는 id만 사용 (모든 uniqueIndexes는 이미 사전 조회로 처리됨)\n          const conflictColumns = [\"id\"];\n\n          const allColumns = Object.keys(dataForDb[0]);\n          let updateColumns = allColumns.filter((c) => c !== \"id\");\n\n          // inherit 옵션 처리 - inherit 컬럼은 update 대상에서 제외\n          if (options?.inherit?.length) {\n            const inheritColumns = options.inherit as string[];\n\n            const excludedFromUpdate = updateColumns.filter((c) => inheritColumns.includes(c));\n            updateColumns = updateColumns.filter((c) => !inheritColumns.includes(c));\n\n            // 실제로 제외된 컬럼 로깅\n            if (excludedFromUpdate.length) {\n              Naite.t(\"puri:ub-inherit\", {\n                tableName,\n                inheritColumns,\n                excludedFromUpdate,\n              });\n            }\n          }\n\n          // updateColumns가 비어있어도 merge()를 사용하여 모든 행이 RETURNING되도록 보장\n          const mergeColumns = updateColumns.length ? updateColumns : conflictColumns;\n\n          resultRows = await wdb\n            .insert(dataForDb)\n            .into(tableName)\n            .onConflict(conflictColumns)\n            .merge(mergeColumns)\n            .returning(selectFields);\n        }\n\n        if (originalUuids.length !== resultRows.length) {\n          throw new Error(`${tableName}: register/returning 불일치`);\n        }\n\n        for (let i = 0; i < resultRows.length; i++) {\n          uuidMap.set(originalUuids[i], resultRows[i]);\n          allIds.push(resultRows[i].id);\n        }\n      }\n    }\n\n    // 해당 테이블 참조를 실제 밸류로 변경\n    for (const table of refTables) {\n      table.rows = table.rows.map((row) => {\n        for (const key of Object.keys(row)) {\n          const prop = row[key];\n          if (isRefField(prop) && prop.of === tableName) {\n            const parent = uuidMap.get(prop.uuid);\n            if (!parent) {\n              console.error(prop);\n              throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);\n            }\n            const resolvedValue = (parent as Record<string, unknown>)[prop.use ?? \"id\"];\n            row[key] = resolvedValue;\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: prop.of, uuid: prop.uuid, use: prop.use ?? \"id\" },\n              to: resolvedValue,\n            });\n          }\n        }\n        return row;\n      });\n    }\n\n    if (options?.cleanOrphans) {\n      const cleanOrphans = options.cleanOrphans;\n      const fkColumns = isArray(cleanOrphans)\n        ? (cleanOrphans as ForeignKeyColumns<TTable>[])\n        : [cleanOrphans as ForeignKeyColumns<TTable>];\n\n      // 현재 register된 레코드들의 FK 값들 추출\n      const fkConditions = fkColumns.map((fkCol) => {\n        const fkValues = [...new Set(table.rows.map((row) => row[fkCol]).filter((v) => v != null))];\n        return { column: fkCol, values: fkValues };\n      });\n\n      // 모든 FK 컬럼에 값이 있는 경우에만 삭제 실행\n      if (fkConditions.every((fc) => fc.values.length > 0)) {\n        let deleteQuery = wdb(tableName);\n\n        // 각 FK 컬럼에 대한 WHERE IN 조건 추가\n        for (const { column, values } of fkConditions) {\n          deleteQuery = deleteQuery.whereIn(column, values as Knex.Value[]);\n        }\n\n        // 방금 upsert한 ID는 제외\n        deleteQuery = deleteQuery.whereNotIn(\"id\", allIds);\n\n        const deletedCount = await deleteQuery.delete();\n\n        Naite.t(\"puri:ub-clean-orphans\", {\n          tableName,\n          cleanOrphans: fkColumns,\n          deletedCount,\n        });\n      }\n    }\n\n    // 해당 테이블의 데이터 초기화\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n\n    Naite.t(\"puri:ub-upserted\", {\n      tableName,\n      mode,\n      rowCount: allIds.length,\n      returnedIds: allIds,\n    });\n\n    return allIds as IdType<DatabaseSchemaExtend, TTable>[];\n  }\n\n  async updateBatch(\n    wdb: Knex,\n    tableName: string,\n    options?: {\n      chunkSize?: number;\n      where?: string | string[];\n    },\n  ): Promise<void> {\n    options = {\n      ...options,\n      chunkSize: options?.chunkSize ?? 500,\n      where: options?.where ?? \"id\",\n    };\n\n    if (!this.hasTable(tableName)) {\n      return;\n    }\n    const table = this.tables.get(tableName);\n    if (!table) {\n      throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);\n    } else if (table.rows.length === 0) {\n      return;\n    }\n\n    const whereColumns = Array.isArray(options.where) ? options.where : [options.where ?? \"id\"];\n    const rows = table.rows.map((_row) => {\n      const { uuid: _, ...row } = _row; // uuid 제외\n      return row as RowWithId<string>;\n    });\n\n    await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);\n\n    Naite.t(\"puri:ub-batch-updated\", {\n      tableName,\n      rowCount: rows.length,\n      whereColumns,\n    });\n\n    // updateBatch 완료 후 처리된 데이터 제거\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n  }\n\n  // ============================================================================\n  // Private Helper Methods\n  // ============================================================================\n\n  /**\n   * rows를 의존성 순서에 따라 레벨별로 그룹화\n   * - 자기 참조 없는 경우 : 모든 rows가 Level 0\n   * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화\n   */\n  private buildInsertLevels(\n    rows: Record<string, unknown>[],\n    tableName: string,\n  ): { levels: Record<string, unknown>[][]; hasCircular: boolean } {\n    // 1. 자기 참조가 없으면 한 레벨로 처리\n    const hasSelfRef = rows\n      .flatMap((row) => Object.values(row))\n      .some((value) => isRefField(value) && value.of === tableName);\n    if (!hasSelfRef) return { levels: [rows], hasCircular: false };\n\n    // 2. uuid → row 매핑 (중복 uuid 방지)\n    const rowByUuid = new Map<string, Record<string, unknown>>();\n    for (const row of rows) {\n      const uuid = row.uuid as string | undefined;\n      if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);\n      rowByUuid.set(uuid, row);\n    }\n\n    let pending = Array.from(rowByUuid.values());\n    const levels: Record<string, unknown>[][] = [];\n    const inserted = new Set<string>();\n\n    // 3. 레벨별 분류\n    while (pending.length > 0) {\n      const currentLevel: Record<string, unknown>[] = [];\n      const nextPending: Record<string, unknown>[] = [];\n\n      for (const row of pending) {\n        // 이 row가 참조하는 자기 참조들\n        const selfRefs = Object.values(row).filter(\n          (value) => isRefField(value) && value.of === tableName,\n        ) as UBRef[];\n\n        // 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함\n        const canInsert = selfRefs.every((ref) => {\n          if (!rowByUuid.has(ref.uuid)) {\n            throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);\n          }\n          return inserted.has(ref.uuid);\n        });\n\n        if (canInsert) {\n          currentLevel.push(row);\n        } else {\n          nextPending.push(row);\n        }\n      }\n\n      // 순환 참조 감지\n      if (currentLevel.length === 0) return { levels: [], hasCircular: true };\n\n      // 레벨 확정 + inserted 갱신\n      levels.push(currentLevel);\n      for (const row of currentLevel) {\n        inserted.add(row.uuid as string);\n      }\n\n      pending = nextPending;\n    }\n\n    return { levels, hasCircular: false };\n  }\n}\n"],"mappings":"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"}
|
|
409
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"upsert-builder.js","names":["result: UBRef","table","allIds: (number | string)[]","resultRows: { id: number | string; [key: string]: unknown }[]","conditions: unknown[][]","levels: Record<string, unknown>[][]","currentLevel: Record<string, unknown>[]","nextPending: Record<string, unknown>[]"],"sources":["../../src/database/upsert-builder.ts"],"sourcesContent":["import { randomUUID } from \"crypto\";\n\nimport { getLogger } from \"@logtape/logtape\";\nimport { type Knex } from \"knex\";\nimport { cluster, isArray, unique } from \"radashi\";\n\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport {\n  type DatabaseForeignKeys,\n  type DatabaseSchemaExtend,\n  type EntityIndex,\n} from \"../types/types\";\nimport { assertDefined, nonNullable } from \"../utils/utils\";\nimport { batchUpdate } from \"./_batch_update\";\nimport { type RowWithId } from \"./_batch_update\";\nimport { type ColumnKeys, type ForeignKeyColumns, type IdType, type TableName } from \"./puri.types\";\n\nconst logger = getLogger([\"sonamu\", \"internal\", \"upsert-builder\"]);\n\n/**\n * FK 타입 추론을 위해 DatabaseForeignKeys export\n * (module augmentation 자동 로드 보장)\n */\nexport type { DatabaseForeignKeys };\n\ntype InheritableColumns<TTable extends TableName<DatabaseSchemaExtend>> =\n  TTable extends keyof DatabaseSchemaExtend ? ColumnKeys<DatabaseSchemaExtend[TTable]> : never;\n\n// 테이블 데이터 타입\ntype TableData = {\n  references: Set<string>;\n  rows: Record<string, unknown>[];\n  uniqueIndexes: EntityIndex[];\n  uniquesMap: Map<string, string>;\n  jsonColumns: string[];\n};\n\n// 참조 필드 타입\nexport type UBRef = {\n  uuid: string;\n  of: string;\n  use?: string;\n};\n\n// upsert 옵션\nexport type UpsertOptions<TTable extends TableName<DatabaseSchemaExtend>> = {\n  chunkSize?: number;\n  cleanOrphans?: ForeignKeyColumns<TTable> | ForeignKeyColumns<TTable>[];\n  inherit?: InheritableColumns<TTable>[];\n};\n\n// insertOnly 옵션\nexport type InsertOnlyOptions = {\n  chunkSize?: number;\n};\n\nexport function isRefField(field: unknown): field is UBRef {\n  return (\n    field !== undefined &&\n    field !== null &&\n    (field as UBRef)?.of !== undefined &&\n    (field as UBRef)?.uuid !== undefined\n  );\n}\n\nexport class UpsertBuilder {\n  tables: Map<string, TableData>;\n  constructor() {\n    this.tables = new Map();\n  }\n\n  getTable(tableName: string): TableData {\n    const table = this.tables.get(tableName);\n    if (table) {\n      return table;\n    }\n\n    const tableSpec = (() => {\n      try {\n        return EntityManager.getTableSpec(tableName);\n      } catch {\n        return null;\n      }\n    })();\n\n    const tableData = {\n      references: new Set<string>(),\n      rows: [],\n      uniqueIndexes: tableSpec?.uniqueIndexes ?? [],\n      uniquesMap: new Map<string, string>(),\n      jsonColumns: tableSpec?.jsonColumns ?? [],\n    };\n    this.tables.set(tableName, tableData);\n    return tableData;\n  }\n\n  hasTable(tableName: string): boolean {\n    return this.tables.has(tableName);\n  }\n\n  register<T extends string>(\n    tableName: string,\n    row: {\n      [key in T]?: UBRef | string | number | boolean | bigint | null | object | unknown;\n    },\n  ): UBRef {\n    const table = this.getTable(tableName);\n\n    // 해당 테이블의 unique 인덱스를 순회하며 키 생성\n    const uniqueKeys = table.uniqueIndexes\n      .map((unqIndex) => {\n        const uniqueKeyArray = unqIndex.columns.map((unqCol) => {\n          const val = row[unqCol.name as keyof typeof row];\n          if (isRefField(val)) {\n            return val.uuid;\n          } else {\n            return row[unqCol.name as keyof typeof row] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입\n          }\n        });\n\n        // 값이 모두 null인 경우 키 생성 패스\n        if (uniqueKeyArray.length === 0) {\n          return null;\n        }\n        return uniqueKeyArray.join(\"---delimiter--\");\n      })\n      .filter(nonNullable);\n\n    // uuid 생성 로직\n    const { uuid, isReused } = (() => {\n      // 키를 순회하여 이미 존재하는 키가 있는지 확인\n      if (uniqueKeys.length > 0) {\n        for (const uniqueKey of uniqueKeys) {\n          if (table.uniquesMap.has(uniqueKey)) {\n            return {\n              uuid: assertDefined(table.uniquesMap.get(uniqueKey), \"Unique key not found\"),\n              isReused: true,\n            };\n          }\n        }\n      }\n\n      // 찾을 수 없는 경우 생성\n      return { uuid: randomUUID(), isReused: false };\n    })();\n\n    // 모든 유니크키에 대해 유니크맵에 uuid 저장\n    if (uniqueKeys.length > 0) {\n      for (const uniqueKey of uniqueKeys) {\n        table.uniquesMap.set(uniqueKey, uuid);\n      }\n    }\n\n    // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가\n    // 이 정보를 나중에 치환할 때 사용\n    row = Object.fromEntries(\n      Object.entries(row).map(([rowKey, rowValue]) => {\n        if (isRefField(rowValue)) {\n          rowValue.use ??= \"id\";\n          table.references.add(`${rowValue.of}.${rowValue.use}`);\n          return [rowKey, rowValue];\n        } else if (table.jsonColumns.includes(rowKey) && rowValue !== null) {\n          // JSON 컬럼인 경우 JSON.stringify 처리 (Knex는 JSON 타입을 지원하지 않음)\n          return [rowKey, JSON.stringify(rowValue)];\n        } else if (isArray(rowValue)) {\n          // 배열은 그대로 저장\n          return [rowKey, rowValue];\n        } else {\n          return [rowKey, rowValue];\n        }\n      }),\n    ) as { [key in T]?: unknown };\n\n    table.rows.push({\n      uuid,\n      ...row,\n    });\n\n    const result: UBRef = {\n      of: tableName,\n      uuid: (row as { uuid?: string }).uuid ?? uuid,\n    };\n\n    Naite.t(\"puri:ub-register\", {\n      tableName,\n      uuid: result.uuid,\n      isUuidReused: isReused,\n      row,\n    });\n\n    return result;\n  }\n\n  async upsert<TTable extends TableName<DatabaseSchemaExtend>>(\n    wdb: Knex,\n    tableName: TTable,\n    options?: UpsertOptions<TTable>,\n  ): Promise<IdType<DatabaseSchemaExtend, TTable>[]> {\n    return this.upsertOrInsert(wdb, tableName, \"upsert\", options);\n  }\n\n  async insertOnly<TTable extends TableName<DatabaseSchemaExtend>>(\n    wdb: Knex,\n    tableName: TTable,\n    options?: InsertOnlyOptions,\n  ): Promise<IdType<DatabaseSchemaExtend, TTable>[]> {\n    return this.upsertOrInsert(wdb, tableName, \"insert\", options);\n  }\n\n  async upsertOrInsert<TTable extends TableName<DatabaseSchemaExtend>>(\n    wdb: Knex,\n    tableName: TTable,\n    mode: \"upsert\" | \"insert\",\n    options?: UpsertOptions<TTable>,\n  ): Promise<IdType<DatabaseSchemaExtend, TTable>[]> {\n    if (!this.hasTable(tableName)) {\n      return [];\n    }\n\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);\n    } else if (table.rows.length === 0) {\n      throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);\n    }\n\n    if (\n      table.rows.some((row) =>\n        Object.entries(row).some(([, value]) => isRefField(value) && value.of !== tableName),\n      )\n    ) {\n      throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);\n    }\n\n    // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출\n    const { references, refTables } = Array.from(this.tables).reduce(\n      (r, [, table]) => {\n        const reference = Array.from(table.references.values()).find((ref) =>\n          ref.includes(`${tableName}.`),\n        );\n        if (reference) {\n          r.references.push(reference);\n          r.refTables.push(table);\n        }\n\n        return r;\n      },\n      {\n        references: [] as string[],\n        refTables: [] as TableData[],\n      },\n    );\n    const extractFields = unique(references)\n      .map((reference) => reference.split(\".\")[1])\n      .filter((field): field is string => field !== undefined);\n\n    // 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)\n    const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);\n\n    if (hasCircular) {\n      throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);\n    }\n\n    const uuidMap = new Map<string, unknown>();\n    const allIds: (number | string)[] = [];\n\n    // 레벨별로 순차 처리\n    for (let levelIdx = 0; levelIdx < levels.length; levelIdx++) {\n      const levelRows = levels[levelIdx];\n      logger.debug(\"Processing Query Level: {current} / {total}\", {\n        current: levelIdx + 1,\n        total: levels.length,\n      });\n\n      // 이전 레벨에서 얻은 ID로 자기 참조 해결\n      const resolvedRows = levelRows.map((row) => {\n        const resolved = { ...row };\n        for (const [key, value] of Object.entries(row)) {\n          if (isRefField(value) && value.of === tableName) {\n            const parent = uuidMap.get(value.uuid);\n\n            if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);\n\n            resolved[key] = (parent as Record<string, unknown>)[value.use ?? \"id\"];\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: value.of, uuid: value.uuid, use: value.use ?? \"id\" },\n              to: resolved[key],\n            });\n          }\n        }\n        return resolved;\n      });\n\n      // 현재 레벨 upsert\n      const chunkSize = options?.chunkSize;\n      const levelChunks = chunkSize ? cluster(resolvedRows, chunkSize) : [resolvedRows];\n      const selectFields = unique([\"id\", ...extractFields]);\n\n      for (let index = 0; index < levelChunks.length; index++) {\n        const dataChunk = levelChunks[index];\n        if (dataChunk.length === 0) continue;\n        logger.debug(\"Processing Chunk: {current} / {total}\", {\n          current: index + 1,\n          total: levelChunks.length,\n        });\n\n        // uuid를 별도로 보관하고, DB에 저장할 데이터에서 제거\n        const originalUuids = dataChunk.map((r) => r.uuid as string);\n        const dataForDb = dataChunk.map(({ uuid: _, ...rest }) => rest);\n\n        let resultRows: { id: number | string; [key: string]: unknown }[];\n\n        if (mode === \"insert\") {\n          // INSERT 모드 - RETURNING 사용\n          resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);\n        } else {\n          // UPSERT 모드 - id 없는 row들의 id를 사전 조회로 채우기\n          const rowsWithoutId = dataForDb.filter((row) => !row.id);\n\n          if (rowsWithoutId.length > 0 && table.uniqueIndexes.length > 0) {\n            // 모든 uniqueIndexes로 기존 레코드 조회\n            for (const uniqueIndex of table.uniqueIndexes) {\n              const columns = uniqueIndex.columns.map((c) => c.name);\n\n              // 조회할 조건들 추출 (각 row의 unique 컬럼 값들)\n              const conditions: unknown[][] = [];\n              for (const row of rowsWithoutId) {\n                const values = columns.map((col) => row[col]);\n                // null이 포함된 조건은 제외 (PostgreSQL UNIQUE는 NULL 무시)\n                if (!values.some((v) => v === null || v === undefined)) {\n                  conditions.push(values);\n                }\n              }\n\n              if (conditions.length === 0) continue;\n\n              // 배치 SELECT\n              const existingRows = (await wdb(tableName)\n                .whereIn(columns, conditions as Record<string, unknown>[][])\n                .select(\"id\", ...columns)) as Record<string, unknown>[];\n\n              // Map 생성: unique 컬럼 조합 → id\n              const existingMap = new Map<string, number | string>();\n              for (const existing of existingRows) {\n                const key = columns\n                  .map((col) => String(existing[col] ?? \"\"))\n                  .join(\"---delimiter---\");\n                const id = existing.id;\n                if (typeof id === \"number\" || typeof id === \"string\") {\n                  existingMap.set(key, id);\n                }\n              }\n\n              // id 없는 row들에 매칭되는 id 채우기\n              for (const row of rowsWithoutId) {\n                if (row.id) continue; // 이미 다른 uniqueIndex에서 채워진 경우 스킵\n\n                const key = columns.map((col) => String(row[col] ?? \"\")).join(\"---delimiter---\");\n                const existingId = existingMap.get(key);\n\n                if (existingId) {\n                  row.id = existingId;\n                }\n              }\n            }\n          }\n\n          // onConflict는 id만 사용 (모든 uniqueIndexes는 이미 사전 조회로 처리됨)\n          const conflictColumns = [\"id\"];\n\n          const allColumns = Object.keys(dataForDb[0]);\n          let updateColumns = allColumns.filter((c) => c !== \"id\");\n\n          // inherit 옵션 처리 - inherit 컬럼은 update 대상에서 제외\n          if (options?.inherit?.length) {\n            const inheritColumns = options.inherit as string[];\n\n            const excludedFromUpdate = updateColumns.filter((c) => inheritColumns.includes(c));\n            updateColumns = updateColumns.filter((c) => !inheritColumns.includes(c));\n\n            // 실제로 제외된 컬럼 로깅\n            if (excludedFromUpdate.length) {\n              Naite.t(\"puri:ub-inherit\", {\n                tableName,\n                inheritColumns,\n                excludedFromUpdate,\n              });\n            }\n          }\n\n          // updateColumns가 비어있어도 merge()를 사용하여 모든 행이 RETURNING되도록 보장\n          const mergeColumns = updateColumns.length ? updateColumns : conflictColumns;\n\n          resultRows = await wdb\n            .insert(dataForDb)\n            .into(tableName)\n            .onConflict(conflictColumns)\n            .merge(mergeColumns)\n            .returning(selectFields);\n        }\n\n        if (originalUuids.length !== resultRows.length) {\n          throw new Error(`${tableName}: register/returning 불일치`);\n        }\n\n        for (let i = 0; i < resultRows.length; i++) {\n          uuidMap.set(originalUuids[i], resultRows[i]);\n          allIds.push(resultRows[i].id);\n        }\n      }\n    }\n\n    // 해당 테이블 참조를 실제 밸류로 변경\n    for (const table of refTables) {\n      table.rows = table.rows.map((row) => {\n        for (const key of Object.keys(row)) {\n          const prop = row[key];\n          if (isRefField(prop) && prop.of === tableName) {\n            const parent = uuidMap.get(prop.uuid);\n            if (!parent) {\n              console.error(prop);\n              throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);\n            }\n            const resolvedValue = (parent as Record<string, unknown>)[prop.use ?? \"id\"];\n            row[key] = resolvedValue;\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: prop.of, uuid: prop.uuid, use: prop.use ?? \"id\" },\n              to: resolvedValue,\n            });\n          }\n        }\n        return row;\n      });\n    }\n\n    if (options?.cleanOrphans) {\n      const cleanOrphans = options.cleanOrphans;\n      const fkColumns = isArray(cleanOrphans)\n        ? (cleanOrphans as ForeignKeyColumns<TTable>[])\n        : [cleanOrphans as ForeignKeyColumns<TTable>];\n\n      // 현재 register된 레코드들의 FK 값들 추출\n      const fkConditions = fkColumns.map((fkCol) => {\n        const fkValues = [...new Set(table.rows.map((row) => row[fkCol]).filter(Boolean))];\n        return { column: fkCol, values: fkValues };\n      });\n\n      // 모든 FK 컬럼에 값이 있는 경우에만 삭제 실행\n      if (fkConditions.every((fc) => fc.values.length > 0)) {\n        let deleteQuery = wdb(tableName);\n\n        // 각 FK 컬럼에 대한 WHERE IN 조건 추가\n        for (const { column, values } of fkConditions) {\n          deleteQuery = deleteQuery.whereIn(column, values as Knex.Value[]);\n        }\n\n        // 방금 upsert한 ID는 제외\n        deleteQuery = deleteQuery.whereNotIn(\"id\", allIds);\n\n        const deletedCount = await deleteQuery.delete();\n\n        Naite.t(\"puri:ub-clean-orphans\", {\n          tableName,\n          cleanOrphans: fkColumns,\n          deletedCount,\n        });\n      }\n    }\n\n    // 해당 테이블의 데이터 초기화\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n\n    Naite.t(\"puri:ub-upserted\", {\n      tableName,\n      mode,\n      rowCount: allIds.length,\n      returnedIds: allIds,\n    });\n\n    return allIds as IdType<DatabaseSchemaExtend, TTable>[];\n  }\n\n  async updateBatch(\n    wdb: Knex,\n    tableName: string,\n    options?: {\n      chunkSize?: number;\n      where?: string | string[];\n    },\n  ): Promise<void> {\n    options = {\n      ...options,\n      chunkSize: options?.chunkSize ?? 500,\n      where: options?.where ?? \"id\",\n    };\n\n    if (!this.hasTable(tableName)) {\n      return;\n    }\n    const table = this.tables.get(tableName);\n    if (!table) {\n      throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);\n    } else if (table.rows.length === 0) {\n      return;\n    }\n\n    const whereColumns = Array.isArray(options.where) ? options.where : [options.where ?? \"id\"];\n    const rows = table.rows.map((_row) => {\n      const { uuid: _, ...row } = _row; // uuid 제외\n      return row as RowWithId<string>;\n    });\n\n    await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);\n\n    Naite.t(\"puri:ub-batch-updated\", {\n      tableName,\n      rowCount: rows.length,\n      whereColumns,\n    });\n\n    // updateBatch 완료 후 처리된 데이터 제거\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n  }\n\n  // ============================================================================\n  // Private Helper Methods\n  // ============================================================================\n\n  /**\n   * rows를 의존성 순서에 따라 레벨별로 그룹화\n   * - 자기 참조 없는 경우 : 모든 rows가 Level 0\n   * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화\n   */\n  private buildInsertLevels(\n    rows: Record<string, unknown>[],\n    tableName: string,\n  ): { levels: Record<string, unknown>[][]; hasCircular: boolean } {\n    // 1. 자기 참조가 없으면 한 레벨로 처리\n    const hasSelfRef = rows\n      .flatMap((row) => Object.values(row))\n      .some((value) => isRefField(value) && value.of === tableName);\n    if (!hasSelfRef) return { levels: [rows], hasCircular: false };\n\n    // 2. uuid → row 매핑 (중복 uuid 방지)\n    const rowByUuid = new Map<string, Record<string, unknown>>();\n    for (const row of rows) {\n      const uuid = row.uuid as string | undefined;\n      if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);\n      rowByUuid.set(uuid, row);\n    }\n\n    let pending = Array.from(rowByUuid.values());\n    const levels: Record<string, unknown>[][] = [];\n    const inserted = new Set<string>();\n\n    // 3. 레벨별 분류\n    while (pending.length > 0) {\n      const currentLevel: Record<string, unknown>[] = [];\n      const nextPending: Record<string, unknown>[] = [];\n\n      for (const row of pending) {\n        // 이 row가 참조하는 자기 참조들\n        const selfRefs = Object.values(row).filter(\n          (value) => isRefField(value) && value.of === tableName,\n        ) as UBRef[];\n\n        // 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함\n        const canInsert = selfRefs.every((ref) => {\n          if (!rowByUuid.has(ref.uuid)) {\n            throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);\n          }\n          return inserted.has(ref.uuid);\n        });\n\n        if (canInsert) {\n          currentLevel.push(row);\n        } else {\n          nextPending.push(row);\n        }\n      }\n\n      // 순환 참조 감지\n      if (currentLevel.length === 0) return { levels: [], hasCircular: true };\n\n      // 레벨 확정 + inserted 갱신\n      levels.push(currentLevel);\n      for (const row of currentLevel) {\n        inserted.add(row.uuid as string);\n      }\n\n      pending = nextPending;\n    }\n\n    return { levels, hasCircular: false };\n  }\n}\n"],"mappings":"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"}
|
|
@@ -504,7 +504,7 @@ var init_sonamu_dictionary = __esmMin((() => {
|
|
|
504
504
|
const stats = {};
|
|
505
505
|
const total = rows.length;
|
|
506
506
|
for (const locale of locales) {
|
|
507
|
-
const filled = rows.filter((row) => row[locale]
|
|
507
|
+
const filled = rows.filter((row) => !!row[locale]).length;
|
|
508
508
|
const percent = total > 0 ? Math.round(filled / total * 100) : 0;
|
|
509
509
|
stats[locale] = {
|
|
510
510
|
total,
|
|
@@ -889,4 +889,4 @@ var init_sonamu_dictionary = __esmMin((() => {
|
|
|
889
889
|
//#endregion
|
|
890
890
|
init_sonamu_dictionary();
|
|
891
891
|
export { SonamuDictionary, init_sonamu_dictionary, sonamuDictionary };
|
|
892
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"sonamu-dictionary.js","names":["entries: DictEntry[]","helpers: string[]","lines: string[]","rows: DictionaryRow[]","row: DictionaryRow","stats: Record<string, { total: number; filled: number; percent: number }>","columnWidths: number[]","projectDictEntries: Record<string, DictEntry[]>","colToHeader: Map<string, string>","rowValues: Record<string, string>","newEntry: DictEntry","sgPath: string | null","searchPaths: string[]"],"sources":["../../src/dict/sonamu-dictionary.ts"],"sourcesContent":["import { execSync } from \"child_process\";\nimport fs from \"fs\";\nimport path from \"path\";\n\nimport { Workbook } from \"@sheetkit/node\";\nimport ts from \"typescript\";\n\nimport { Sonamu } from \"../api/sonamu\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { BadRequestException } from \"../exceptions/so-exceptions\";\nimport { formatCode } from \"../utils/formatter\";\nimport { SD } from \"./sd\";\nimport {\n  type DictEntry,\n  type DictionaryResult,\n  type DictionaryRow,\n  type EntityKeyInfo,\n  type I18nConfig,\n  type ImportResult,\n  type UsageResult,\n} from \"./types\";\n\n/**\n * 0-based 컬럼 인덱스를 엑셀 컬럼 문자로 변환 (0 -> \"A\", 25 -> \"Z\", 26 -> \"AA\")\n */\nfunction colLetter(index: number): string {\n  let result = \"\";\n  let n = index;\n  while (n >= 0) {\n    result = String.fromCharCode(65 + (n % 26)) + result;\n    n = Math.floor(n / 26) - 1;\n  }\n  return result;\n}\n\n/**\n * Sonamu Dictionary 관리 클래스\n * i18n 딕셔너리의 CRUD 및 Excel import/export를 담당합니다.\n */\nexport class SonamuDictionary {\n  /**\n   * TypeScript Compiler API를 사용하여 dict 파일 파싱\n   *\n   * 지원 패턴:\n   * - export default { ... } as const;\n   * - export default defineLocale({ ... });\n   * - 문자열 값: \"key\": \"value\" 또는 key: `value`\n   * - 함수 값: \"key\": (param: Type) => `template`\n   */\n  parseDictFile(filePath: string): DictEntry[] {\n    const content = fs.readFileSync(filePath, \"utf-8\");\n    const sourceFile = ts.createSourceFile(filePath, content, ts.ScriptTarget.Latest, true);\n\n    const entries: DictEntry[] = [];\n\n    ts.forEachChild(sourceFile, (node) => {\n      if (ts.isExportAssignment(node)) {\n        const objectLiteral = this.unwrapToObjectLiteral(node.expression);\n        if (objectLiteral) {\n          this.extractEntriesFromObject(objectLiteral, sourceFile, entries);\n        }\n      }\n    });\n\n    return entries;\n  }\n\n  /**\n   * 파일에서 특정 이름의 const 선언을 찾아 ObjectLiteral 파싱\n   * 예: const entityLabels = { ... } as const;\n   */\n  parseConstObjectDeclaration(filePath: string, varName: string): DictEntry[] {\n    const content = fs.readFileSync(filePath, \"utf-8\");\n    const sourceFile = ts.createSourceFile(filePath, content, ts.ScriptTarget.Latest, true);\n\n    const entries: DictEntry[] = [];\n\n    ts.forEachChild(sourceFile, (node) => {\n      if (ts.isVariableStatement(node)) {\n        for (const decl of node.declarationList.declarations) {\n          if (ts.isIdentifier(decl.name) && decl.name.text === varName && decl.initializer) {\n            const objectLiteral = this.unwrapToObjectLiteral(decl.initializer);\n            if (objectLiteral) {\n              this.extractEntriesFromObject(objectLiteral, sourceFile, entries);\n            }\n          }\n        }\n      }\n    });\n\n    return entries;\n  }\n\n  /**\n   * 문자열이 화살표 함수 또는 함수 표현식인지 판별\n   */\n  isExpressionFunction(code: string): boolean {\n    // 빈 문자열이나 공백만 있는 경우\n    if (!code.trim()) {\n      return false;\n    }\n\n    const ARROW_FUNCTION_PATTERN = /^\\s*\\([^)]*\\)\\s*=>/;\n\n    return ARROW_FUNCTION_PATTERN.test(code);\n  }\n\n  /**\n   * export default 표현식에서 ObjectLiteralExpression 추출\n   * - as const 처리\n   * - defineLocale({ ... }) 호출 처리\n   */\n  private unwrapToObjectLiteral(expr: ts.Expression): ts.ObjectLiteralExpression | null {\n    // as const 처리\n    if (ts.isAsExpression(expr)) {\n      return this.unwrapToObjectLiteral(expr.expression);\n    }\n    // 직접 객체 리터럴\n    if (ts.isObjectLiteralExpression(expr)) {\n      return expr;\n    }\n    // defineLocale({ ... }) 호출\n    if (ts.isCallExpression(expr)) {\n      const firstArg = expr.arguments[0];\n      if (firstArg && ts.isObjectLiteralExpression(firstArg)) {\n        return firstArg;\n      }\n    }\n    return null;\n  }\n\n  /**\n   * ObjectLiteralExpression에서 DictEntry 추출\n   */\n  private extractEntriesFromObject(\n    objectLiteral: ts.ObjectLiteralExpression,\n    sourceFile: ts.SourceFile,\n    entries: DictEntry[],\n  ): void {\n    for (const prop of objectLiteral.properties) {\n      const entry = this.extractDictEntry(prop, sourceFile);\n      if (entry) {\n        entries.push(entry);\n      }\n    }\n  }\n\n  /**\n   * PropertyName에서 키 문자열 추출\n   * - 문자열 리터럴: \"key\"\n   * - 식별자: key (unquoted)\n   */\n  private getPropertyKey(name: ts.PropertyName): string | null {\n    if (ts.isStringLiteral(name)) {\n      return name.text;\n    }\n    if (ts.isIdentifier(name)) {\n      return name.text;\n    }\n    return null;\n  }\n\n  /**\n   * 프로퍼티에서 DictEntry 추출\n   * - 문자열: 실제 문자열 값\n   * - 함수: 원본 소스 (여러 줄은 한 줄로 정규화)\n   */\n  private extractDictEntry(\n    prop: ts.ObjectLiteralElementLike,\n    sourceFile: ts.SourceFile,\n  ): DictEntry | null {\n    if (!ts.isPropertyAssignment(prop)) {\n      return null;\n    }\n\n    const key = this.getPropertyKey(prop.name);\n    if (!key) return null;\n\n    const init = prop.initializer;\n\n    // 화살표 함수\n    if (ts.isArrowFunction(init)) {\n      const funcText = init.getText(sourceFile);\n      const normalized = funcText.replace(/\\s*\\n\\s*/g, \" \").trim();\n      return { key, value: normalized, isFunction: true };\n    }\n\n    // 문자열 리터럴\n    if (ts.isStringLiteral(init)) {\n      return { key, value: init.text, isFunction: false };\n    }\n\n    // 템플릿 리터럴 (변수 없음)\n    if (ts.isNoSubstitutionTemplateLiteral(init)) {\n      return { key, value: init.text, isFunction: false };\n    }\n\n    // 기타 (예: 함수 표현식)\n    return {\n      key,\n      value: init.getText(sourceFile),\n      isFunction: ts.isFunctionExpression(init),\n    };\n  }\n\n  /**\n   * 프로젝트의 i18n dict 파일 경로를 반환합니다.\n   * @param locale - 로케일 (ko, en 등)\n   * @param target - 타겟 디렉토리 (api, web, app)\n   */\n  getProjectDictPath(locale: string, target: \"api\" | \"web\" | \"app\" = \"api\"): string {\n    const dir = target === \"api\" ? Sonamu.config.api.dir : target;\n    return path.join(Sonamu.appRootPath, dir, \"src\", \"i18n\", `${locale}.ts`);\n  }\n\n  /**\n   * sonamu 내장 dict 파일 경로를 반환합니다.\n   */\n  private getSonamuDictPath(locale: string): string {\n    const packageRoot = path.resolve(import.meta.dirname, \"..\", \"..\");\n    return path.join(packageRoot, \"src\", \"dict\", `${locale}.ts`);\n  }\n\n  /**\n   * 필요한 dict 키가 프로젝트에 존재하는지 확인하고, 없으면 추가합니다.\n   * defaultLocale에만 추가하며, 다른 locale은 사용자가 직접 번역해야 합니다.\n   *\n   * @param requiredKeys - 필요한 키 목록\n   * @param target - 타겟 디렉토리 (api, web, app)\n   * @returns 추가된 키 목록\n   */\n  async ensureDictKeys(\n    requiredKeys: string[],\n    target: \"api\" | \"web\" | \"app\" = \"api\",\n  ): Promise<string[]> {\n    const { defaultLocale } = Sonamu.config.i18n;\n    const projectDictPath = this.getProjectDictPath(defaultLocale, target);\n\n    // 프로젝트 dict 파일이 없으면 아무것도 하지 않음\n    if (!fs.existsSync(projectDictPath)) {\n      return [];\n    }\n\n    // 프로젝트 dict에서 기존 키 파싱\n    const projectEntries = this.parseDictFile(projectDictPath);\n    const existingKeys = new Set(projectEntries.map((e) => e.key));\n\n    // 누락된 키 찾기\n    const missingKeys = requiredKeys.filter((key) => !existingKeys.has(key));\n\n    if (missingKeys.length === 0) {\n      return [];\n    }\n\n    // sonamu dict에서 기본값 가져오기\n    const sonamuDictPath = this.getSonamuDictPath(defaultLocale);\n    if (!fs.existsSync(sonamuDictPath)) {\n      return [];\n    }\n\n    const sonamuEntries = this.parseDictFile(sonamuDictPath);\n    const sonamuDict = new Map(sonamuEntries.map((e) => [e.key, e]));\n\n    // 추가할 엔트리 생성\n    const entriesToAdd = missingKeys\n      .map((key) => sonamuDict.get(key))\n      .filter((entry): entry is NonNullable<typeof entry> => entry !== undefined);\n\n    if (entriesToAdd.length === 0) {\n      return [];\n    }\n\n    // 프로젝트 dict 파일에 추가\n    await this.appendEntriesToDictFile(projectDictPath, entriesToAdd, defaultLocale, true);\n\n    return entriesToAdd.map((e) => e.key);\n  }\n\n  /**\n   * dict 파일에 엔트리를 추가합니다.\n   * 기존 파일을 파싱하고, 새 엔트리를 추가한 뒤, 전체 파일을 재생성합니다.\n   */\n  private async appendEntriesToDictFile(\n    filePath: string,\n    entries: DictEntry[],\n    locale: string,\n    isDefaultLocale: boolean,\n  ): Promise<void> {\n    // 기존 entries 파싱\n    const existingEntries = this.parseDictFile(filePath);\n\n    // 새 entries 추가\n    for (const entry of entries) {\n      existingEntries.push(entry);\n    }\n\n    // 파일 재생성\n    const content = this.generateProjectDict(locale, existingEntries, isDefaultLocale);\n    const formatted = await formatCode(content, \"typescript\", filePath);\n    fs.writeFileSync(filePath, formatted, \"utf-8\");\n  }\n\n  /**\n   * 함수 값들에서 사용되는 헬퍼 함수를 감지합니다.\n   */\n  private detectUsedHelpers(entries: DictEntry[]): {\n    helpers: string[];\n    usesFormat: boolean;\n  } {\n    const functionEntries = entries.filter((e) => e.isFunction);\n    const helpers: string[] = [];\n\n    for (const helper of [\"plural\", \"josa\"]) {\n      // 함수명이 단어 경계로 사용되는지 확인 (예: plural( 또는 plural,)\n      const pattern = new RegExp(`\\\\b${helper}\\\\s*\\\\(`);\n      if (functionEntries.some((e) => pattern.test(e.value))) {\n        helpers.push(helper);\n      }\n    }\n\n    // format 사용 여부 별도 감지 (format.number(...), format.date(...) 등)\n    const formatPattern = /\\bformat\\.\\w+\\s*\\(/;\n    const usesFormat = functionEntries.some((e) => formatPattern.test(e.value));\n\n    return { helpers, usesFormat };\n  }\n\n  /**\n   * Project dict 파일 생성\n   */\n  generateProjectDict(locale: string, entries: DictEntry[], isDefaultLocale: boolean): string {\n    // key 알파벳 순 정렬\n    const sorted = [...entries].toSorted((a, b) => a.key.localeCompare(b.key));\n\n    const lines: string[] = [];\n\n    // 함수 값에서 사용되는 헬퍼 함수 감지\n    const { helpers, usesFormat } = this.detectUsedHelpers(entries);\n\n    // 헬퍼 함수 import 추가\n    const imports = [...helpers];\n    if (usesFormat) {\n      imports.push(\"createFormat\");\n    }\n    if (imports.length > 0) {\n      lines.push(`import { ${imports.join(\", \")} } from \"sonamu/dict\";`);\n    }\n\n    if (!isDefaultLocale) {\n      lines.push('import { defineLocale } from \"./sd.generated\";');\n    }\n\n    if (imports.length > 0 || !isDefaultLocale) {\n      lines.push(\"\");\n    }\n\n    // format 사용 시 createFormat 호출 추가\n    if (usesFormat) {\n      lines.push(`const format = createFormat(\"${locale}\");`);\n      lines.push(\"\");\n    }\n\n    lines.push(\"/**\");\n    lines.push(` * Project ${locale.toUpperCase()} Dictionary`);\n    lines.push(\" */\");\n\n    if (isDefaultLocale) {\n      lines.push(\"export default {\");\n    } else {\n      lines.push(\"export default defineLocale({\");\n    }\n\n    for (const entry of sorted) {\n      if (entry.isFunction) {\n        // 함수인 경우: 원형 그대로 출력\n        lines.push(`  \"${entry.key}\": ${entry.value},`);\n      } else {\n        lines.push(`  \"${entry.key}\": ${JSON.stringify(entry.value)},`);\n      }\n    }\n\n    if (isDefaultLocale) {\n      lines.push(\"} as const;\");\n    } else {\n      lines.push(\"});\");\n    }\n    lines.push(\"\");\n\n    return lines.join(\"\\n\");\n  }\n\n  /**\n   * i18n 설정을 가져옵니다.\n   */\n  private getI18nConfig(): I18nConfig {\n    return Sonamu.config.i18n;\n  }\n\n  /**\n   * i18n 디렉토리 경로를 반환하고, 없으면 생성합니다.\n   */\n  private ensureI18nDir(): string {\n    const i18nDir = path.join(Sonamu.apiRootPath, \"src\", \"i18n\");\n    if (!fs.existsSync(i18nDir)) {\n      fs.mkdirSync(i18nDir, { recursive: true });\n    }\n    return i18nDir;\n  }\n\n  /**\n   * dict 파일을 저장합니다.\n   */\n  private async saveDictFile(\n    locale: string,\n    entries: DictEntry[],\n    isDefaultLocale: boolean,\n  ): Promise<void> {\n    const i18nDir = this.ensureI18nDir();\n    const dictPath = path.join(i18nDir, `${locale}.ts`);\n    const content = this.generateProjectDict(locale, entries, isDefaultLocale);\n    const formatted = await formatCode(content, \"typescript\", dictPath);\n    fs.writeFileSync(dictPath, formatted, \"utf-8\");\n  }\n\n  /**\n   * i18n key를 파싱하여 entity 관련 정보 추출\n   */\n  parseEntityKey(key: string): EntityKeyInfo {\n    // entity.{EntityId} (list, create, edit 제외)\n    const entityTitleMatch = key.match(/^entity\\.([A-Z][a-zA-Z0-9]*)$/);\n    if (\n      entityTitleMatch &&\n      !key.includes(\".list\") &&\n      !key.includes(\".create\") &&\n      !key.includes(\".edit\")\n    ) {\n      return { type: \"entityTitle\", entityId: entityTitleMatch[1] };\n    }\n\n    // entity.{EntityId}.{propName}\n    const propDescMatch = key.match(/^entity\\.([A-Z][a-zA-Z0-9]*)\\.([a-z_][a-z0-9_]*)$/);\n    if (propDescMatch) {\n      return { type: \"propDesc\", entityId: propDescMatch[1], propName: propDescMatch[2] };\n    }\n\n    // enum.{EnumId}.{value}\n    const enumLabelMatch = key.match(/^enum\\.([A-Z][a-zA-Z0-9]*)\\.(.+)$/);\n    if (enumLabelMatch) {\n      return { type: \"enumLabel\", enumId: enumLabelMatch[1], enumValue: enumLabelMatch[2] };\n    }\n\n    return { type: \"other\" };\n  }\n\n  /**\n   * entity key에 대해 entity.json 업데이트 수행\n   * @returns 업데이트 여부\n   */\n  async updateEntityByKey(key: string, value: string): Promise<boolean> {\n    const keyInfo = this.parseEntityKey(key);\n\n    switch (keyInfo.type) {\n      case \"entityTitle\": {\n        try {\n          const entity = EntityManager.get(keyInfo.entityId);\n          if (entity.title !== value) {\n            entity.title = value;\n            await entity.save();\n            return true;\n          }\n        } catch {\n          // entity not found\n        }\n        return false;\n      }\n\n      case \"propDesc\": {\n        try {\n          const entity = EntityManager.get(keyInfo.entityId);\n          const propIndex = entity.props.findIndex((p) => p.name === keyInfo.propName);\n          if (propIndex !== -1 && entity.props[propIndex].desc !== value) {\n            entity.props[propIndex].desc = value;\n            await entity.save();\n            return true;\n          }\n        } catch {\n          // entity not found\n        }\n        return false;\n      }\n\n      case \"enumLabel\": {\n        for (const entityId of EntityManager.getAllIds()) {\n          const entity = EntityManager.get(entityId);\n          if (entity.enumLabels[keyInfo.enumId]) {\n            if (entity.enumLabels[keyInfo.enumId][keyInfo.enumValue] !== value) {\n              entity.enumLabels[keyInfo.enumId][keyInfo.enumValue] = value;\n              await entity.save();\n              return true;\n            }\n            break;\n          }\n        }\n        return false;\n      }\n\n      default:\n        return false;\n    }\n  }\n\n  /**\n   * sd.generated.ts에서 entity labels 추출\n   * entity.json에서 관리되는 값만 포함 (.list, .create, .edit 제외)\n   */\n  extractEntityLabels(): DictEntry[] {\n    const sdPath = path.join(Sonamu.apiRootPath, \"src\", \"i18n\", \"sd.generated.ts\");\n    if (!fs.existsSync(sdPath)) {\n      return [];\n    }\n\n    return this.parseConstObjectDeclaration(sdPath, \"entityLabels\");\n  }\n\n  /**\n   * sd.generated.ts에서 rc-keys 추출\n   * react-components에서 관리되는 i18n 키들\n   * @param locale - 로케일 (ko, en 등)\n   */\n  extractRCKeys(locale: string): DictEntry[] {\n    const sdPath = path.join(Sonamu.apiRootPath, \"src\", \"i18n\", \"sd.generated.ts\");\n    if (!fs.existsSync(sdPath)) {\n      return [];\n    }\n\n    // locale별 변수명 매핑 (sd.template.ts의 getRCKeysVarName과 동일)\n    const varName = (() => {\n      if (locale === \"ko\") return \"rcKeysKo\";\n      if (locale === \"en\") return \"rcKeysEn\";\n      // 다른 locale은 en을 fallback으로 사용\n      return \"rcKeysEn\";\n    })();\n\n    return this.parseConstObjectDeclaration(sdPath, varName);\n  }\n\n  /**\n   * Project dict 파일([locale].ts)에서 딕셔너리 로드\n   */\n  loadProjectDict(locale: string): { entries: DictEntry[] } {\n    const dictPath = path.join(Sonamu.apiRootPath, \"src\", \"i18n\", `${locale}.ts`);\n    if (!fs.existsSync(dictPath)) {\n      return { entries: [] };\n    }\n    return { entries: this.parseDictFile(dictPath) };\n  }\n\n  /**\n   * 딕셔너리 데이터 수집 (sonamu + entity + project)\n   */\n  async collectDictionary(): Promise<DictionaryResult> {\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    const rows: DictionaryRow[] = [];\n    const rowMap = new Map<string, DictionaryRow>();\n\n    // 1. RC Keys (sonamu source, 각 locale별)\n    for (const locale of locales) {\n      const rcKeys = this.extractRCKeys(locale);\n      for (const rcKey of rcKeys) {\n        let row = rowMap.get(rcKey.key);\n        if (!row) {\n          row = {\n            key: rcKey.key,\n            source: \"sonamu\",\n            isFunction: rcKey.isFunction ?? false,\n          };\n          rowMap.set(rcKey.key, row);\n        }\n        row[locale] = rcKey.value;\n        if (rcKey.isFunction) {\n          row.isFunction = true;\n        }\n      }\n    }\n\n    // 2. Entity labels (default locale 기준)\n    const entityLabels = this.extractEntityLabels();\n    for (const label of entityLabels) {\n      const row: DictionaryRow = {\n        key: label.key,\n        source: \"entity\",\n        isFunction: label.isFunction ?? false,\n        [defaultLocale]: label.value,\n      };\n      rowMap.set(label.key, row);\n    }\n\n    // 3. Project dict (각 locale별)\n    for (const locale of locales) {\n      const { entries } = this.loadProjectDict(locale);\n      for (const entry of entries) {\n        const existing = rowMap.get(entry.key);\n        if (existing) {\n          // sonamu, entity source가 있으면 해당 locale 값만 추가\n          existing[locale] = entry.value;\n          if (entry.isFunction) {\n            existing.isFunction = true;\n          }\n        } else {\n          // project source로 새로 추가\n          let row = rowMap.get(entry.key);\n          if (!row) {\n            row = {\n              key: entry.key,\n              source: \"project\",\n              isFunction: entry.isFunction,\n            };\n            rowMap.set(entry.key, row);\n          }\n          row[locale] = entry.value;\n        }\n      }\n    }\n\n    rows.push(...rowMap.values());\n    rows.sort((a, b) => a.key.localeCompare(b.key));\n\n    // 통계 계산: locale별 (채워진 값 / 전체 키 수)\n    const stats: Record<string, { total: number; filled: number; percent: number }> = {};\n    const total = rows.length;\n    for (const locale of locales) {\n      const filled = rows.filter((row) => row[locale] != null && row[locale] !== \"\").length;\n      const percent = total > 0 ? Math.round((filled / total) * 100) : 0;\n      stats[locale] = { total, filled, percent };\n    }\n\n    return { rows, locales, defaultLocale, stats };\n  }\n\n  /**\n   * 딕셔너리 조회\n   */\n  async getDictionary(): Promise<DictionaryResult> {\n    return this.collectDictionary();\n  }\n\n  /**\n   * Excel로 내보내기\n   */\n  async exportToExcel(): Promise<{ filename: string; buffer: Buffer }> {\n    const { rows, locales } = await this.collectDictionary();\n\n    const wb = new Workbook();\n    const sheet = \"i18n\";\n    wb.setSheetName(\"Sheet1\", sheet);\n\n    const projectName = `${Sonamu.config.projectName ?? \"Sonamu\"} Dictionary`;\n    const headers = [\"key\", \"source\", ...locales];\n\n    // 스타일 정의\n    const titleStyleId = wb.addStyle({\n      font: { size: 23 },\n      alignment: { vertical: \"center\", horizontal: \"left\" },\n    });\n    const headerStyleId = wb.addStyle({\n      font: { bold: true, size: 11 },\n      alignment: { horizontal: \"center\", vertical: \"center\" },\n      fill: { pattern: \"solid\", fgColor: \"F1F1F1\" },\n      border: {\n        top: { style: \"thin\", color: \"D0D0D0\" },\n        left: { style: \"thin\", color: \"D0D0D0\" },\n        bottom: { style: \"thin\", color: \"D0D0D0\" },\n        right: { style: \"thin\", color: \"D0D0D0\" },\n      },\n    });\n    const dataStyleId = wb.addStyle({\n      font: { size: 11 },\n      alignment: { vertical: \"center\", horizontal: \"left\" },\n    });\n\n    // 행 1: 프로젝트명\n    wb.setCellValue(sheet, \"A1\", projectName);\n    wb.setCellStyle(sheet, \"A1\", titleStyleId);\n    wb.setRowHeight(sheet, 1, 28);\n\n    // 행 2: 빈 행 (기본값)\n\n    // 행 3: 헤더\n    wb.setRowValues(sheet, 3, \"A\", headers);\n    wb.setRowHeight(sheet, 3, 26);\n    for (let col = 0; col < headers.length; col++) {\n      wb.setCellStyle(sheet, `${colLetter(col)}3`, headerStyleId);\n    }\n\n    // 행 4 이후: 데이터\n    for (let i = 0; i < rows.length; i++) {\n      const row = rows[i];\n      const values = [row.key, row.source, ...locales.map((locale) => row[locale] ?? \"\")];\n      const rowNum = i + 4;\n      wb.setRowValues(sheet, rowNum, \"A\", values);\n      wb.setRowHeight(sheet, rowNum, 24);\n      for (let col = 0; col < values.length; col++) {\n        wb.setCellStyle(sheet, `${colLetter(col)}${rowNum}`, dataStyleId);\n      }\n    }\n\n    // 컬럼 너비 계산\n    const MAX_WIDTH = 50;\n    const MIN_WIDTH = 10;\n    const columnWidths: number[] = headers.map((header) => Math.max(header.length, MIN_WIDTH));\n\n    for (const row of rows) {\n      const values = [row.key, row.source, ...locales.map((locale) => row[locale] ?? \"\")];\n      values.forEach((value, idx) => {\n        const textLength = String(value).length;\n        columnWidths[idx] = Math.min(Math.max(columnWidths[idx], textLength), MAX_WIDTH);\n      });\n    }\n\n    // 컬럼 너비 적용\n    for (let col = 0; col < columnWidths.length; col++) {\n      wb.setColWidth(sheet, colLetter(col), columnWidths[col] + 2);\n    }\n\n    return {\n      filename: `${projectName}-${new Date().toISOString().split(\"T\")[0]}.xlsx`,\n      buffer: wb.writeBufferSync(),\n    };\n  }\n\n  /**\n   * Excel에서 가져오기\n   *\n   * 형식:\n   * - 1행: 프로젝트명\n   * - 2행: 빈 행\n   * - 3행: 헤더 (key, source, locales...)\n   * - 4행 이후: 데이터\n   */\n  async importFromExcel(buffer: Buffer): Promise<ImportResult> {\n    const wb = Workbook.openBufferSync(buffer);\n    const sheet = wb.sheetNames[0];\n    const allRows = wb.getRows(sheet);\n\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    let updatedEntities = 0;\n    let updatedLocales = 0;\n\n    // locale별 project dict entries\n    const projectDictEntries: Record<string, DictEntry[]> = {};\n    for (const locale of locales) {\n      projectDictEntries[locale] = [];\n    }\n\n    // 헤더 행 찾기: 첫 번째 컬럼(A)이 \"key\"인 행\n    let headerRowNum = 0;\n    for (const rowData of allRows) {\n      const firstCell = rowData.cells.find((c) => c.column === \"A\");\n      const firstCellValue = String(firstCell?.value ?? \"\")\n        .trim()\n        .toLowerCase();\n      if (firstCellValue === \"key\") {\n        headerRowNum = rowData.row;\n        break;\n      }\n    }\n\n    if (headerRowNum === 0) {\n      throw new BadRequestException(SD(\"sonamu.error.headerRowNotFound\"));\n    }\n\n    // 헤더 행에서 컬럼 매핑 구성\n    const headerRowData = allRows.find((r) => r.row === headerRowNum);\n    const colToHeader: Map<string, string> = new Map();\n    if (headerRowData) {\n      for (const cell of headerRowData.cells) {\n        colToHeader.set(cell.column, String(cell.value ?? \"\"));\n      }\n    }\n\n    // 데이터 행 읽기 (헤더 다음 행부터)\n    for (const rowData of allRows) {\n      if (rowData.row <= headerRowNum) continue;\n\n      const rowValues: Record<string, string> = {};\n      for (const cell of rowData.cells) {\n        const headerName = colToHeader.get(cell.column);\n        if (headerName) {\n          rowValues[headerName] = String(cell.value ?? \"\");\n        }\n      }\n\n      const key = rowValues.key;\n      const source = rowValues.source as \"entity\" | \"project\";\n\n      if (!key || !source) continue;\n\n      if (source === \"entity\") {\n        // entity source: default locale만 entity.json에 저장\n        const defaultValue = rowValues[defaultLocale];\n        if (defaultValue) {\n          const updated = await this.updateEntityByKey(key, defaultValue);\n          if (updated) {\n            updatedEntities++;\n          }\n        }\n\n        // non-default locale은 project dict에 저장\n        for (const locale of locales) {\n          if (locale === defaultLocale) continue;\n          const cellValue = rowValues[locale]?.trim();\n          if (cellValue) {\n            projectDictEntries[locale].push({\n              key,\n              value: cellValue,\n              isFunction: this.isExpressionFunction(cellValue),\n            });\n          }\n        }\n      } else if (source === \"project\") {\n        // project source: 모든 locale을 project dict에 저장\n        for (const locale of locales) {\n          const cellValue = rowValues[locale]?.trim();\n          if (cellValue) {\n            projectDictEntries[locale].push({\n              key,\n              value: cellValue,\n              isFunction: this.isExpressionFunction(cellValue),\n            });\n          }\n        }\n      }\n    }\n\n    // Project dict 파일 생성\n    for (const locale of locales) {\n      const entries = projectDictEntries[locale];\n      if (entries.length > 0) {\n        await this.saveDictFile(locale, entries, locale === defaultLocale);\n        updatedLocales++;\n      }\n    }\n\n    return {\n      success: true,\n      updatedEntities,\n      updatedLocales,\n    };\n  }\n\n  /**\n   * 딕셔너리 항목 수정\n   */\n  async updateEntry(params: {\n    oldKey: string;\n    newKey: string;\n    source: \"entity\" | \"project\" | \"sonamu\";\n    values: Record<string, string>;\n  }): Promise<void> {\n    const { oldKey, newKey, source, values } = params;\n\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    // entity source의 default locale 처리\n    if (source === \"entity\" && values[defaultLocale]) {\n      await this.updateEntityByKey(newKey, values[defaultLocale]);\n    }\n\n    // project dict 업데이트\n    // - entity의 non-default locale\n    // - project source의 모든 locale\n    // - sonamu source의 모든 locale (override)\n    for (const locale of locales) {\n      // entity source의 default locale은 entity.json에서 처리했으므로 스킵\n      if (source === \"entity\" && locale === defaultLocale) continue;\n\n      const cellValue = values[locale]?.trim();\n      if (!cellValue) continue;\n\n      // 기존 dict 로드\n      const { entries } = this.loadProjectDict(locale);\n\n      // key 변경 시 기존 key 제거\n      if (oldKey !== newKey) {\n        const oldIndex = entries.findIndex((e) => e.key === oldKey);\n        if (oldIndex !== -1) {\n          entries.splice(oldIndex, 1);\n        }\n      }\n\n      // 새 값 업데이트 또는 추가\n      const existingIndex = entries.findIndex((e) => e.key === newKey);\n      const newEntry: DictEntry = {\n        key: newKey,\n        value: cellValue,\n        isFunction: this.isExpressionFunction(cellValue),\n      };\n\n      if (existingIndex !== -1) {\n        entries[existingIndex] = newEntry;\n      } else {\n        entries.push(newEntry);\n      }\n\n      // dict 파일 저장\n      await this.saveDictFile(locale, entries, locale === defaultLocale);\n    }\n  }\n\n  /**\n   * 딕셔너리 항목 추가\n   */\n  async createEntry(params: { key: string; values: Record<string, string> }): Promise<void> {\n    const { key, values } = params;\n\n    if (!key?.trim()) {\n      throw new BadRequestException(SD(\"sonamu.error.keyRequired\"));\n    }\n\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    // 중복 키 체크\n    for (const locale of locales) {\n      const { entries } = this.loadProjectDict(locale);\n      if (entries.some((e) => e.key === key)) {\n        throw new BadRequestException(SD(\"sonamu.error.keyAlreadyExists\")(key));\n      }\n    }\n\n    // 각 locale에 새 키 추가\n    for (const locale of locales) {\n      const cellValue = values[locale]?.trim();\n      if (!cellValue) continue;\n\n      const { entries } = this.loadProjectDict(locale);\n      entries.push({\n        key,\n        value: cellValue,\n        isFunction: this.isExpressionFunction(cellValue),\n      });\n\n      await this.saveDictFile(locale, entries, locale === defaultLocale);\n    }\n  }\n\n  /**\n   * 딕셔너리 항목 삭제\n   */\n  async deleteEntry(key: string): Promise<void> {\n    if (!key) {\n      throw new BadRequestException(SD(\"sonamu.error.keyRequired\"));\n    }\n\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    let deleted = false;\n    for (const locale of locales) {\n      const { entries } = this.loadProjectDict(locale);\n      const index = entries.findIndex((e) => e.key === key);\n      if (index !== -1) {\n        entries.splice(index, 1);\n        deleted = true;\n\n        await this.saveDictFile(locale, entries, locale === defaultLocale);\n      }\n    }\n\n    if (!deleted) {\n      throw new BadRequestException(SD(\"sonamu.error.keyNotFound\")(key));\n    }\n  }\n\n  /**\n   * 미사용 키 검사 (ast-grep 사용)\n   */\n  async checkUsage(keys: string[]): Promise<UsageResult> {\n    // ast-grep 설치 확인\n    let sgPath: string | null = null;\n    try {\n      sgPath = execSync(\"which sg\", { encoding: \"utf-8\" }).trim();\n    } catch {\n      try {\n        sgPath = execSync(\"which ast-grep\", { encoding: \"utf-8\" }).trim();\n      } catch {\n        // ast-grep not installed\n      }\n    }\n\n    if (!sgPath) {\n      return {\n        error:\n          \"ast-grep이 설치되어 있지 않습니다. brew install ast-grep 또는 npm install -g @ast-grep/cli로 설치해주세요.\",\n        unusedKeys: [],\n      };\n    }\n\n    const searchPaths: string[] = [];\n    for (const entry of [\"api\", \"web\", \"app\"]) {\n      const srcPath = path.join(Sonamu.appRootPath, entry, \"src\");\n      if (fs.existsSync(srcPath)) {\n        searchPaths.push(srcPath);\n      }\n    }\n\n    if (searchPaths.length === 0) {\n      return {\n        error: \"검색할 src 디렉토리를 찾을 수 없습니다.\",\n        unusedKeys: [],\n      };\n    }\n\n    const usedKeys = new Set<string>();\n\n    try {\n      // ast-grep으로 SD(\"...\") 패턴 검색\n      // 패턴: SD(\"KEY\") 또는 SD('KEY') 형태\n      const patterns = ['SD(\"$KEY\")', \"SD('$KEY')\"];\n\n      for (const searchPath of searchPaths) {\n        for (const pattern of patterns) {\n          try {\n            const result = execSync(`${sgPath} --pattern '${pattern}' --json ${searchPath}`, {\n              encoding: \"utf-8\",\n              maxBuffer: 50 * 1024 * 1024, // 50MB\n            });\n\n            if (result.trim()) {\n              const matches = JSON.parse(result);\n              for (const match of matches) {\n                // metaVariables.single.KEY.text에서 키 추출\n                const keyText = match.metaVariables?.single?.KEY?.text;\n                if (keyText) {\n                  // 따옴표 제거\n                  const cleanKey = keyText.replace(/^[\"']|[\"']$/g, \"\");\n                  usedKeys.add(cleanKey);\n                }\n              }\n            }\n          } catch {\n            // 패턴 매치 없으면 에러 (무시)\n          }\n        }\n      }\n\n      // keys 중에서 usedKeys에 없는 것들이 미사용 키\n      const unusedKeys = keys.filter((k) => !usedKeys.has(k));\n\n      return { unusedKeys, usedKeysCount: usedKeys.size };\n    } catch (e) {\n      return {\n        error: `검색 중 오류 발생: ${e instanceof Error ? e.message : String(e)}`,\n        unusedKeys: [],\n      };\n    }\n  }\n}\n\nexport const sonamuDictionary = new SonamuDictionary();\n"],"mappings":"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"}
|
|
892
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"sonamu-dictionary.js","names":["entries: DictEntry[]","helpers: string[]","lines: string[]","rows: DictionaryRow[]","row: DictionaryRow","stats: Record<string, { total: number; filled: number; percent: number }>","columnWidths: number[]","projectDictEntries: Record<string, DictEntry[]>","colToHeader: Map<string, string>","rowValues: Record<string, string>","newEntry: DictEntry","sgPath: string | null","searchPaths: string[]"],"sources":["../../src/dict/sonamu-dictionary.ts"],"sourcesContent":["import { execSync } from \"child_process\";\nimport fs from \"fs\";\nimport path from \"path\";\n\nimport { Workbook } from \"@sheetkit/node\";\nimport ts from \"typescript\";\n\nimport { Sonamu } from \"../api/sonamu\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { BadRequestException } from \"../exceptions/so-exceptions\";\nimport { formatCode } from \"../utils/formatter\";\nimport { SD } from \"./sd\";\nimport {\n  type DictEntry,\n  type DictionaryResult,\n  type DictionaryRow,\n  type EntityKeyInfo,\n  type I18nConfig,\n  type ImportResult,\n  type UsageResult,\n} from \"./types\";\n\n/**\n * 0-based 컬럼 인덱스를 엑셀 컬럼 문자로 변환 (0 -> \"A\", 25 -> \"Z\", 26 -> \"AA\")\n */\nfunction colLetter(index: number): string {\n  let result = \"\";\n  let n = index;\n  while (n >= 0) {\n    result = String.fromCharCode(65 + (n % 26)) + result;\n    n = Math.floor(n / 26) - 1;\n  }\n  return result;\n}\n\n/**\n * Sonamu Dictionary 관리 클래스\n * i18n 딕셔너리의 CRUD 및 Excel import/export를 담당합니다.\n */\nexport class SonamuDictionary {\n  /**\n   * TypeScript Compiler API를 사용하여 dict 파일 파싱\n   *\n   * 지원 패턴:\n   * - export default { ... } as const;\n   * - export default defineLocale({ ... });\n   * - 문자열 값: \"key\": \"value\" 또는 key: `value`\n   * - 함수 값: \"key\": (param: Type) => `template`\n   */\n  parseDictFile(filePath: string): DictEntry[] {\n    const content = fs.readFileSync(filePath, \"utf-8\");\n    const sourceFile = ts.createSourceFile(filePath, content, ts.ScriptTarget.Latest, true);\n\n    const entries: DictEntry[] = [];\n\n    ts.forEachChild(sourceFile, (node) => {\n      if (ts.isExportAssignment(node)) {\n        const objectLiteral = this.unwrapToObjectLiteral(node.expression);\n        if (objectLiteral) {\n          this.extractEntriesFromObject(objectLiteral, sourceFile, entries);\n        }\n      }\n    });\n\n    return entries;\n  }\n\n  /**\n   * 파일에서 특정 이름의 const 선언을 찾아 ObjectLiteral 파싱\n   * 예: const entityLabels = { ... } as const;\n   */\n  parseConstObjectDeclaration(filePath: string, varName: string): DictEntry[] {\n    const content = fs.readFileSync(filePath, \"utf-8\");\n    const sourceFile = ts.createSourceFile(filePath, content, ts.ScriptTarget.Latest, true);\n\n    const entries: DictEntry[] = [];\n\n    ts.forEachChild(sourceFile, (node) => {\n      if (ts.isVariableStatement(node)) {\n        for (const decl of node.declarationList.declarations) {\n          if (ts.isIdentifier(decl.name) && decl.name.text === varName && decl.initializer) {\n            const objectLiteral = this.unwrapToObjectLiteral(decl.initializer);\n            if (objectLiteral) {\n              this.extractEntriesFromObject(objectLiteral, sourceFile, entries);\n            }\n          }\n        }\n      }\n    });\n\n    return entries;\n  }\n\n  /**\n   * 문자열이 화살표 함수 또는 함수 표현식인지 판별\n   */\n  isExpressionFunction(code: string): boolean {\n    // 빈 문자열이나 공백만 있는 경우\n    if (!code.trim()) {\n      return false;\n    }\n\n    const ARROW_FUNCTION_PATTERN = /^\\s*\\([^)]*\\)\\s*=>/;\n\n    return ARROW_FUNCTION_PATTERN.test(code);\n  }\n\n  /**\n   * export default 표현식에서 ObjectLiteralExpression 추출\n   * - as const 처리\n   * - defineLocale({ ... }) 호출 처리\n   */\n  private unwrapToObjectLiteral(expr: ts.Expression): ts.ObjectLiteralExpression | null {\n    // as const 처리\n    if (ts.isAsExpression(expr)) {\n      return this.unwrapToObjectLiteral(expr.expression);\n    }\n    // 직접 객체 리터럴\n    if (ts.isObjectLiteralExpression(expr)) {\n      return expr;\n    }\n    // defineLocale({ ... }) 호출\n    if (ts.isCallExpression(expr)) {\n      const firstArg = expr.arguments[0];\n      if (firstArg && ts.isObjectLiteralExpression(firstArg)) {\n        return firstArg;\n      }\n    }\n    return null;\n  }\n\n  /**\n   * ObjectLiteralExpression에서 DictEntry 추출\n   */\n  private extractEntriesFromObject(\n    objectLiteral: ts.ObjectLiteralExpression,\n    sourceFile: ts.SourceFile,\n    entries: DictEntry[],\n  ): void {\n    for (const prop of objectLiteral.properties) {\n      const entry = this.extractDictEntry(prop, sourceFile);\n      if (entry) {\n        entries.push(entry);\n      }\n    }\n  }\n\n  /**\n   * PropertyName에서 키 문자열 추출\n   * - 문자열 리터럴: \"key\"\n   * - 식별자: key (unquoted)\n   */\n  private getPropertyKey(name: ts.PropertyName): string | null {\n    if (ts.isStringLiteral(name)) {\n      return name.text;\n    }\n    if (ts.isIdentifier(name)) {\n      return name.text;\n    }\n    return null;\n  }\n\n  /**\n   * 프로퍼티에서 DictEntry 추출\n   * - 문자열: 실제 문자열 값\n   * - 함수: 원본 소스 (여러 줄은 한 줄로 정규화)\n   */\n  private extractDictEntry(\n    prop: ts.ObjectLiteralElementLike,\n    sourceFile: ts.SourceFile,\n  ): DictEntry | null {\n    if (!ts.isPropertyAssignment(prop)) {\n      return null;\n    }\n\n    const key = this.getPropertyKey(prop.name);\n    if (!key) return null;\n\n    const init = prop.initializer;\n\n    // 화살표 함수\n    if (ts.isArrowFunction(init)) {\n      const funcText = init.getText(sourceFile);\n      const normalized = funcText.replace(/\\s*\\n\\s*/g, \" \").trim();\n      return { key, value: normalized, isFunction: true };\n    }\n\n    // 문자열 리터럴\n    if (ts.isStringLiteral(init)) {\n      return { key, value: init.text, isFunction: false };\n    }\n\n    // 템플릿 리터럴 (변수 없음)\n    if (ts.isNoSubstitutionTemplateLiteral(init)) {\n      return { key, value: init.text, isFunction: false };\n    }\n\n    // 기타 (예: 함수 표현식)\n    return {\n      key,\n      value: init.getText(sourceFile),\n      isFunction: ts.isFunctionExpression(init),\n    };\n  }\n\n  /**\n   * 프로젝트의 i18n dict 파일 경로를 반환합니다.\n   * @param locale - 로케일 (ko, en 등)\n   * @param target - 타겟 디렉토리 (api, web, app)\n   */\n  getProjectDictPath(locale: string, target: \"api\" | \"web\" | \"app\" = \"api\"): string {\n    const dir = target === \"api\" ? Sonamu.config.api.dir : target;\n    return path.join(Sonamu.appRootPath, dir, \"src\", \"i18n\", `${locale}.ts`);\n  }\n\n  /**\n   * sonamu 내장 dict 파일 경로를 반환합니다.\n   */\n  private getSonamuDictPath(locale: string): string {\n    const packageRoot = path.resolve(import.meta.dirname, \"..\", \"..\");\n    return path.join(packageRoot, \"src\", \"dict\", `${locale}.ts`);\n  }\n\n  /**\n   * 필요한 dict 키가 프로젝트에 존재하는지 확인하고, 없으면 추가합니다.\n   * defaultLocale에만 추가하며, 다른 locale은 사용자가 직접 번역해야 합니다.\n   *\n   * @param requiredKeys - 필요한 키 목록\n   * @param target - 타겟 디렉토리 (api, web, app)\n   * @returns 추가된 키 목록\n   */\n  async ensureDictKeys(\n    requiredKeys: string[],\n    target: \"api\" | \"web\" | \"app\" = \"api\",\n  ): Promise<string[]> {\n    const { defaultLocale } = Sonamu.config.i18n;\n    const projectDictPath = this.getProjectDictPath(defaultLocale, target);\n\n    // 프로젝트 dict 파일이 없으면 아무것도 하지 않음\n    if (!fs.existsSync(projectDictPath)) {\n      return [];\n    }\n\n    // 프로젝트 dict에서 기존 키 파싱\n    const projectEntries = this.parseDictFile(projectDictPath);\n    const existingKeys = new Set(projectEntries.map((e) => e.key));\n\n    // 누락된 키 찾기\n    const missingKeys = requiredKeys.filter((key) => !existingKeys.has(key));\n\n    if (missingKeys.length === 0) {\n      return [];\n    }\n\n    // sonamu dict에서 기본값 가져오기\n    const sonamuDictPath = this.getSonamuDictPath(defaultLocale);\n    if (!fs.existsSync(sonamuDictPath)) {\n      return [];\n    }\n\n    const sonamuEntries = this.parseDictFile(sonamuDictPath);\n    const sonamuDict = new Map(sonamuEntries.map((e) => [e.key, e]));\n\n    // 추가할 엔트리 생성\n    const entriesToAdd = missingKeys\n      .map((key) => sonamuDict.get(key))\n      .filter((entry): entry is NonNullable<typeof entry> => entry !== undefined);\n\n    if (entriesToAdd.length === 0) {\n      return [];\n    }\n\n    // 프로젝트 dict 파일에 추가\n    await this.appendEntriesToDictFile(projectDictPath, entriesToAdd, defaultLocale, true);\n\n    return entriesToAdd.map((e) => e.key);\n  }\n\n  /**\n   * dict 파일에 엔트리를 추가합니다.\n   * 기존 파일을 파싱하고, 새 엔트리를 추가한 뒤, 전체 파일을 재생성합니다.\n   */\n  private async appendEntriesToDictFile(\n    filePath: string,\n    entries: DictEntry[],\n    locale: string,\n    isDefaultLocale: boolean,\n  ): Promise<void> {\n    // 기존 entries 파싱\n    const existingEntries = this.parseDictFile(filePath);\n\n    // 새 entries 추가\n    for (const entry of entries) {\n      existingEntries.push(entry);\n    }\n\n    // 파일 재생성\n    const content = this.generateProjectDict(locale, existingEntries, isDefaultLocale);\n    const formatted = await formatCode(content, \"typescript\", filePath);\n    fs.writeFileSync(filePath, formatted, \"utf-8\");\n  }\n\n  /**\n   * 함수 값들에서 사용되는 헬퍼 함수를 감지합니다.\n   */\n  private detectUsedHelpers(entries: DictEntry[]): {\n    helpers: string[];\n    usesFormat: boolean;\n  } {\n    const functionEntries = entries.filter((e) => e.isFunction);\n    const helpers: string[] = [];\n\n    for (const helper of [\"plural\", \"josa\"]) {\n      // 함수명이 단어 경계로 사용되는지 확인 (예: plural( 또는 plural,)\n      const pattern = new RegExp(`\\\\b${helper}\\\\s*\\\\(`);\n      if (functionEntries.some((e) => pattern.test(e.value))) {\n        helpers.push(helper);\n      }\n    }\n\n    // format 사용 여부 별도 감지 (format.number(...), format.date(...) 등)\n    const formatPattern = /\\bformat\\.\\w+\\s*\\(/;\n    const usesFormat = functionEntries.some((e) => formatPattern.test(e.value));\n\n    return { helpers, usesFormat };\n  }\n\n  /**\n   * Project dict 파일 생성\n   */\n  generateProjectDict(locale: string, entries: DictEntry[], isDefaultLocale: boolean): string {\n    // key 알파벳 순 정렬\n    const sorted = [...entries].toSorted((a, b) => a.key.localeCompare(b.key));\n\n    const lines: string[] = [];\n\n    // 함수 값에서 사용되는 헬퍼 함수 감지\n    const { helpers, usesFormat } = this.detectUsedHelpers(entries);\n\n    // 헬퍼 함수 import 추가\n    const imports = [...helpers];\n    if (usesFormat) {\n      imports.push(\"createFormat\");\n    }\n    if (imports.length > 0) {\n      lines.push(`import { ${imports.join(\", \")} } from \"sonamu/dict\";`);\n    }\n\n    if (!isDefaultLocale) {\n      lines.push('import { defineLocale } from \"./sd.generated\";');\n    }\n\n    if (imports.length > 0 || !isDefaultLocale) {\n      lines.push(\"\");\n    }\n\n    // format 사용 시 createFormat 호출 추가\n    if (usesFormat) {\n      lines.push(`const format = createFormat(\"${locale}\");`);\n      lines.push(\"\");\n    }\n\n    lines.push(\"/**\");\n    lines.push(` * Project ${locale.toUpperCase()} Dictionary`);\n    lines.push(\" */\");\n\n    if (isDefaultLocale) {\n      lines.push(\"export default {\");\n    } else {\n      lines.push(\"export default defineLocale({\");\n    }\n\n    for (const entry of sorted) {\n      if (entry.isFunction) {\n        // 함수인 경우: 원형 그대로 출력\n        lines.push(`  \"${entry.key}\": ${entry.value},`);\n      } else {\n        lines.push(`  \"${entry.key}\": ${JSON.stringify(entry.value)},`);\n      }\n    }\n\n    if (isDefaultLocale) {\n      lines.push(\"} as const;\");\n    } else {\n      lines.push(\"});\");\n    }\n    lines.push(\"\");\n\n    return lines.join(\"\\n\");\n  }\n\n  /**\n   * i18n 설정을 가져옵니다.\n   */\n  private getI18nConfig(): I18nConfig {\n    return Sonamu.config.i18n;\n  }\n\n  /**\n   * i18n 디렉토리 경로를 반환하고, 없으면 생성합니다.\n   */\n  private ensureI18nDir(): string {\n    const i18nDir = path.join(Sonamu.apiRootPath, \"src\", \"i18n\");\n    if (!fs.existsSync(i18nDir)) {\n      fs.mkdirSync(i18nDir, { recursive: true });\n    }\n    return i18nDir;\n  }\n\n  /**\n   * dict 파일을 저장합니다.\n   */\n  private async saveDictFile(\n    locale: string,\n    entries: DictEntry[],\n    isDefaultLocale: boolean,\n  ): Promise<void> {\n    const i18nDir = this.ensureI18nDir();\n    const dictPath = path.join(i18nDir, `${locale}.ts`);\n    const content = this.generateProjectDict(locale, entries, isDefaultLocale);\n    const formatted = await formatCode(content, \"typescript\", dictPath);\n    fs.writeFileSync(dictPath, formatted, \"utf-8\");\n  }\n\n  /**\n   * i18n key를 파싱하여 entity 관련 정보 추출\n   */\n  parseEntityKey(key: string): EntityKeyInfo {\n    // entity.{EntityId} (list, create, edit 제외)\n    const entityTitleMatch = key.match(/^entity\\.([A-Z][a-zA-Z0-9]*)$/);\n    if (\n      entityTitleMatch &&\n      !key.includes(\".list\") &&\n      !key.includes(\".create\") &&\n      !key.includes(\".edit\")\n    ) {\n      return { type: \"entityTitle\", entityId: entityTitleMatch[1] };\n    }\n\n    // entity.{EntityId}.{propName}\n    const propDescMatch = key.match(/^entity\\.([A-Z][a-zA-Z0-9]*)\\.([a-z_][a-z0-9_]*)$/);\n    if (propDescMatch) {\n      return { type: \"propDesc\", entityId: propDescMatch[1], propName: propDescMatch[2] };\n    }\n\n    // enum.{EnumId}.{value}\n    const enumLabelMatch = key.match(/^enum\\.([A-Z][a-zA-Z0-9]*)\\.(.+)$/);\n    if (enumLabelMatch) {\n      return { type: \"enumLabel\", enumId: enumLabelMatch[1], enumValue: enumLabelMatch[2] };\n    }\n\n    return { type: \"other\" };\n  }\n\n  /**\n   * entity key에 대해 entity.json 업데이트 수행\n   * @returns 업데이트 여부\n   */\n  async updateEntityByKey(key: string, value: string): Promise<boolean> {\n    const keyInfo = this.parseEntityKey(key);\n\n    switch (keyInfo.type) {\n      case \"entityTitle\": {\n        try {\n          const entity = EntityManager.get(keyInfo.entityId);\n          if (entity.title !== value) {\n            entity.title = value;\n            await entity.save();\n            return true;\n          }\n        } catch {\n          // entity not found\n        }\n        return false;\n      }\n\n      case \"propDesc\": {\n        try {\n          const entity = EntityManager.get(keyInfo.entityId);\n          const propIndex = entity.props.findIndex((p) => p.name === keyInfo.propName);\n          if (propIndex !== -1 && entity.props[propIndex].desc !== value) {\n            entity.props[propIndex].desc = value;\n            await entity.save();\n            return true;\n          }\n        } catch {\n          // entity not found\n        }\n        return false;\n      }\n\n      case \"enumLabel\": {\n        for (const entityId of EntityManager.getAllIds()) {\n          const entity = EntityManager.get(entityId);\n          if (entity.enumLabels[keyInfo.enumId]) {\n            if (entity.enumLabels[keyInfo.enumId][keyInfo.enumValue] !== value) {\n              entity.enumLabels[keyInfo.enumId][keyInfo.enumValue] = value;\n              await entity.save();\n              return true;\n            }\n            break;\n          }\n        }\n        return false;\n      }\n\n      default:\n        return false;\n    }\n  }\n\n  /**\n   * sd.generated.ts에서 entity labels 추출\n   * entity.json에서 관리되는 값만 포함 (.list, .create, .edit 제외)\n   */\n  extractEntityLabels(): DictEntry[] {\n    const sdPath = path.join(Sonamu.apiRootPath, \"src\", \"i18n\", \"sd.generated.ts\");\n    if (!fs.existsSync(sdPath)) {\n      return [];\n    }\n\n    return this.parseConstObjectDeclaration(sdPath, \"entityLabels\");\n  }\n\n  /**\n   * sd.generated.ts에서 rc-keys 추출\n   * react-components에서 관리되는 i18n 키들\n   * @param locale - 로케일 (ko, en 등)\n   */\n  extractRCKeys(locale: string): DictEntry[] {\n    const sdPath = path.join(Sonamu.apiRootPath, \"src\", \"i18n\", \"sd.generated.ts\");\n    if (!fs.existsSync(sdPath)) {\n      return [];\n    }\n\n    // locale별 변수명 매핑 (sd.template.ts의 getRCKeysVarName과 동일)\n    const varName = (() => {\n      if (locale === \"ko\") return \"rcKeysKo\";\n      if (locale === \"en\") return \"rcKeysEn\";\n      // 다른 locale은 en을 fallback으로 사용\n      return \"rcKeysEn\";\n    })();\n\n    return this.parseConstObjectDeclaration(sdPath, varName);\n  }\n\n  /**\n   * Project dict 파일([locale].ts)에서 딕셔너리 로드\n   */\n  loadProjectDict(locale: string): { entries: DictEntry[] } {\n    const dictPath = path.join(Sonamu.apiRootPath, \"src\", \"i18n\", `${locale}.ts`);\n    if (!fs.existsSync(dictPath)) {\n      return { entries: [] };\n    }\n    return { entries: this.parseDictFile(dictPath) };\n  }\n\n  /**\n   * 딕셔너리 데이터 수집 (sonamu + entity + project)\n   */\n  async collectDictionary(): Promise<DictionaryResult> {\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    const rows: DictionaryRow[] = [];\n    const rowMap = new Map<string, DictionaryRow>();\n\n    // 1. RC Keys (sonamu source, 각 locale별)\n    for (const locale of locales) {\n      const rcKeys = this.extractRCKeys(locale);\n      for (const rcKey of rcKeys) {\n        let row = rowMap.get(rcKey.key);\n        if (!row) {\n          row = {\n            key: rcKey.key,\n            source: \"sonamu\",\n            isFunction: rcKey.isFunction ?? false,\n          };\n          rowMap.set(rcKey.key, row);\n        }\n        row[locale] = rcKey.value;\n        if (rcKey.isFunction) {\n          row.isFunction = true;\n        }\n      }\n    }\n\n    // 2. Entity labels (default locale 기준)\n    const entityLabels = this.extractEntityLabels();\n    for (const label of entityLabels) {\n      const row: DictionaryRow = {\n        key: label.key,\n        source: \"entity\",\n        isFunction: label.isFunction ?? false,\n        [defaultLocale]: label.value,\n      };\n      rowMap.set(label.key, row);\n    }\n\n    // 3. Project dict (각 locale별)\n    for (const locale of locales) {\n      const { entries } = this.loadProjectDict(locale);\n      for (const entry of entries) {\n        const existing = rowMap.get(entry.key);\n        if (existing) {\n          // sonamu, entity source가 있으면 해당 locale 값만 추가\n          existing[locale] = entry.value;\n          if (entry.isFunction) {\n            existing.isFunction = true;\n          }\n        } else {\n          // project source로 새로 추가\n          let row = rowMap.get(entry.key);\n          if (!row) {\n            row = {\n              key: entry.key,\n              source: \"project\",\n              isFunction: entry.isFunction,\n            };\n            rowMap.set(entry.key, row);\n          }\n          row[locale] = entry.value;\n        }\n      }\n    }\n\n    rows.push(...rowMap.values());\n    rows.sort((a, b) => a.key.localeCompare(b.key));\n\n    // 통계 계산: locale별 (채워진 값 / 전체 키 수)\n    const stats: Record<string, { total: number; filled: number; percent: number }> = {};\n    const total = rows.length;\n    for (const locale of locales) {\n      const filled = rows.filter((row) => !!row[locale]).length;\n      const percent = total > 0 ? Math.round((filled / total) * 100) : 0;\n      stats[locale] = { total, filled, percent };\n    }\n\n    return { rows, locales, defaultLocale, stats };\n  }\n\n  /**\n   * 딕셔너리 조회\n   */\n  async getDictionary(): Promise<DictionaryResult> {\n    return this.collectDictionary();\n  }\n\n  /**\n   * Excel로 내보내기\n   */\n  async exportToExcel(): Promise<{ filename: string; buffer: Buffer }> {\n    const { rows, locales } = await this.collectDictionary();\n\n    const wb = new Workbook();\n    const sheet = \"i18n\";\n    wb.setSheetName(\"Sheet1\", sheet);\n\n    const projectName = `${Sonamu.config.projectName ?? \"Sonamu\"} Dictionary`;\n    const headers = [\"key\", \"source\", ...locales];\n\n    // 스타일 정의\n    const titleStyleId = wb.addStyle({\n      font: { size: 23 },\n      alignment: { vertical: \"center\", horizontal: \"left\" },\n    });\n    const headerStyleId = wb.addStyle({\n      font: { bold: true, size: 11 },\n      alignment: { horizontal: \"center\", vertical: \"center\" },\n      fill: { pattern: \"solid\", fgColor: \"F1F1F1\" },\n      border: {\n        top: { style: \"thin\", color: \"D0D0D0\" },\n        left: { style: \"thin\", color: \"D0D0D0\" },\n        bottom: { style: \"thin\", color: \"D0D0D0\" },\n        right: { style: \"thin\", color: \"D0D0D0\" },\n      },\n    });\n    const dataStyleId = wb.addStyle({\n      font: { size: 11 },\n      alignment: { vertical: \"center\", horizontal: \"left\" },\n    });\n\n    // 행 1: 프로젝트명\n    wb.setCellValue(sheet, \"A1\", projectName);\n    wb.setCellStyle(sheet, \"A1\", titleStyleId);\n    wb.setRowHeight(sheet, 1, 28);\n\n    // 행 2: 빈 행 (기본값)\n\n    // 행 3: 헤더\n    wb.setRowValues(sheet, 3, \"A\", headers);\n    wb.setRowHeight(sheet, 3, 26);\n    for (let col = 0; col < headers.length; col++) {\n      wb.setCellStyle(sheet, `${colLetter(col)}3`, headerStyleId);\n    }\n\n    // 행 4 이후: 데이터\n    for (let i = 0; i < rows.length; i++) {\n      const row = rows[i];\n      const values = [row.key, row.source, ...locales.map((locale) => row[locale] ?? \"\")];\n      const rowNum = i + 4;\n      wb.setRowValues(sheet, rowNum, \"A\", values);\n      wb.setRowHeight(sheet, rowNum, 24);\n      for (let col = 0; col < values.length; col++) {\n        wb.setCellStyle(sheet, `${colLetter(col)}${rowNum}`, dataStyleId);\n      }\n    }\n\n    // 컬럼 너비 계산\n    const MAX_WIDTH = 50;\n    const MIN_WIDTH = 10;\n    const columnWidths: number[] = headers.map((header) => Math.max(header.length, MIN_WIDTH));\n\n    for (const row of rows) {\n      const values = [row.key, row.source, ...locales.map((locale) => row[locale] ?? \"\")];\n      values.forEach((value, idx) => {\n        const textLength = String(value).length;\n        columnWidths[idx] = Math.min(Math.max(columnWidths[idx], textLength), MAX_WIDTH);\n      });\n    }\n\n    // 컬럼 너비 적용\n    for (let col = 0; col < columnWidths.length; col++) {\n      wb.setColWidth(sheet, colLetter(col), columnWidths[col] + 2);\n    }\n\n    return {\n      filename: `${projectName}-${new Date().toISOString().split(\"T\")[0]}.xlsx`,\n      buffer: wb.writeBufferSync(),\n    };\n  }\n\n  /**\n   * Excel에서 가져오기\n   *\n   * 형식:\n   * - 1행: 프로젝트명\n   * - 2행: 빈 행\n   * - 3행: 헤더 (key, source, locales...)\n   * - 4행 이후: 데이터\n   */\n  async importFromExcel(buffer: Buffer): Promise<ImportResult> {\n    const wb = Workbook.openBufferSync(buffer);\n    const sheet = wb.sheetNames[0];\n    const allRows = wb.getRows(sheet);\n\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    let updatedEntities = 0;\n    let updatedLocales = 0;\n\n    // locale별 project dict entries\n    const projectDictEntries: Record<string, DictEntry[]> = {};\n    for (const locale of locales) {\n      projectDictEntries[locale] = [];\n    }\n\n    // 헤더 행 찾기: 첫 번째 컬럼(A)이 \"key\"인 행\n    let headerRowNum = 0;\n    for (const rowData of allRows) {\n      const firstCell = rowData.cells.find((c) => c.column === \"A\");\n      const firstCellValue = String(firstCell?.value ?? \"\")\n        .trim()\n        .toLowerCase();\n      if (firstCellValue === \"key\") {\n        headerRowNum = rowData.row;\n        break;\n      }\n    }\n\n    if (headerRowNum === 0) {\n      throw new BadRequestException(SD(\"sonamu.error.headerRowNotFound\"));\n    }\n\n    // 헤더 행에서 컬럼 매핑 구성\n    const headerRowData = allRows.find((r) => r.row === headerRowNum);\n    const colToHeader: Map<string, string> = new Map();\n    if (headerRowData) {\n      for (const cell of headerRowData.cells) {\n        colToHeader.set(cell.column, String(cell.value ?? \"\"));\n      }\n    }\n\n    // 데이터 행 읽기 (헤더 다음 행부터)\n    for (const rowData of allRows) {\n      if (rowData.row <= headerRowNum) continue;\n\n      const rowValues: Record<string, string> = {};\n      for (const cell of rowData.cells) {\n        const headerName = colToHeader.get(cell.column);\n        if (headerName) {\n          rowValues[headerName] = String(cell.value ?? \"\");\n        }\n      }\n\n      const key = rowValues.key;\n      const source = rowValues.source as \"entity\" | \"project\";\n\n      if (!key || !source) continue;\n\n      if (source === \"entity\") {\n        // entity source: default locale만 entity.json에 저장\n        const defaultValue = rowValues[defaultLocale];\n        if (defaultValue) {\n          const updated = await this.updateEntityByKey(key, defaultValue);\n          if (updated) {\n            updatedEntities++;\n          }\n        }\n\n        // non-default locale은 project dict에 저장\n        for (const locale of locales) {\n          if (locale === defaultLocale) continue;\n          const cellValue = rowValues[locale]?.trim();\n          if (cellValue) {\n            projectDictEntries[locale].push({\n              key,\n              value: cellValue,\n              isFunction: this.isExpressionFunction(cellValue),\n            });\n          }\n        }\n      } else if (source === \"project\") {\n        // project source: 모든 locale을 project dict에 저장\n        for (const locale of locales) {\n          const cellValue = rowValues[locale]?.trim();\n          if (cellValue) {\n            projectDictEntries[locale].push({\n              key,\n              value: cellValue,\n              isFunction: this.isExpressionFunction(cellValue),\n            });\n          }\n        }\n      }\n    }\n\n    // Project dict 파일 생성\n    for (const locale of locales) {\n      const entries = projectDictEntries[locale];\n      if (entries.length > 0) {\n        await this.saveDictFile(locale, entries, locale === defaultLocale);\n        updatedLocales++;\n      }\n    }\n\n    return {\n      success: true,\n      updatedEntities,\n      updatedLocales,\n    };\n  }\n\n  /**\n   * 딕셔너리 항목 수정\n   */\n  async updateEntry(params: {\n    oldKey: string;\n    newKey: string;\n    source: \"entity\" | \"project\" | \"sonamu\";\n    values: Record<string, string>;\n  }): Promise<void> {\n    const { oldKey, newKey, source, values } = params;\n\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    // entity source의 default locale 처리\n    if (source === \"entity\" && values[defaultLocale]) {\n      await this.updateEntityByKey(newKey, values[defaultLocale]);\n    }\n\n    // project dict 업데이트\n    // - entity의 non-default locale\n    // - project source의 모든 locale\n    // - sonamu source의 모든 locale (override)\n    for (const locale of locales) {\n      // entity source의 default locale은 entity.json에서 처리했으므로 스킵\n      if (source === \"entity\" && locale === defaultLocale) continue;\n\n      const cellValue = values[locale]?.trim();\n      if (!cellValue) continue;\n\n      // 기존 dict 로드\n      const { entries } = this.loadProjectDict(locale);\n\n      // key 변경 시 기존 key 제거\n      if (oldKey !== newKey) {\n        const oldIndex = entries.findIndex((e) => e.key === oldKey);\n        if (oldIndex !== -1) {\n          entries.splice(oldIndex, 1);\n        }\n      }\n\n      // 새 값 업데이트 또는 추가\n      const existingIndex = entries.findIndex((e) => e.key === newKey);\n      const newEntry: DictEntry = {\n        key: newKey,\n        value: cellValue,\n        isFunction: this.isExpressionFunction(cellValue),\n      };\n\n      if (existingIndex !== -1) {\n        entries[existingIndex] = newEntry;\n      } else {\n        entries.push(newEntry);\n      }\n\n      // dict 파일 저장\n      await this.saveDictFile(locale, entries, locale === defaultLocale);\n    }\n  }\n\n  /**\n   * 딕셔너리 항목 추가\n   */\n  async createEntry(params: { key: string; values: Record<string, string> }): Promise<void> {\n    const { key, values } = params;\n\n    if (!key?.trim()) {\n      throw new BadRequestException(SD(\"sonamu.error.keyRequired\"));\n    }\n\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    // 중복 키 체크\n    for (const locale of locales) {\n      const { entries } = this.loadProjectDict(locale);\n      if (entries.some((e) => e.key === key)) {\n        throw new BadRequestException(SD(\"sonamu.error.keyAlreadyExists\")(key));\n      }\n    }\n\n    // 각 locale에 새 키 추가\n    for (const locale of locales) {\n      const cellValue = values[locale]?.trim();\n      if (!cellValue) continue;\n\n      const { entries } = this.loadProjectDict(locale);\n      entries.push({\n        key,\n        value: cellValue,\n        isFunction: this.isExpressionFunction(cellValue),\n      });\n\n      await this.saveDictFile(locale, entries, locale === defaultLocale);\n    }\n  }\n\n  /**\n   * 딕셔너리 항목 삭제\n   */\n  async deleteEntry(key: string): Promise<void> {\n    if (!key) {\n      throw new BadRequestException(SD(\"sonamu.error.keyRequired\"));\n    }\n\n    const { defaultLocale, supportedLocales } = this.getI18nConfig();\n    const locales = supportedLocales;\n\n    let deleted = false;\n    for (const locale of locales) {\n      const { entries } = this.loadProjectDict(locale);\n      const index = entries.findIndex((e) => e.key === key);\n      if (index !== -1) {\n        entries.splice(index, 1);\n        deleted = true;\n\n        await this.saveDictFile(locale, entries, locale === defaultLocale);\n      }\n    }\n\n    if (!deleted) {\n      throw new BadRequestException(SD(\"sonamu.error.keyNotFound\")(key));\n    }\n  }\n\n  /**\n   * 미사용 키 검사 (ast-grep 사용)\n   */\n  async checkUsage(keys: string[]): Promise<UsageResult> {\n    // ast-grep 설치 확인\n    let sgPath: string | null = null;\n    try {\n      sgPath = execSync(\"which sg\", { encoding: \"utf-8\" }).trim();\n    } catch {\n      try {\n        sgPath = execSync(\"which ast-grep\", { encoding: \"utf-8\" }).trim();\n      } catch {\n        // ast-grep not installed\n      }\n    }\n\n    if (!sgPath) {\n      return {\n        error:\n          \"ast-grep이 설치되어 있지 않습니다. brew install ast-grep 또는 npm install -g @ast-grep/cli로 설치해주세요.\",\n        unusedKeys: [],\n      };\n    }\n\n    const searchPaths: string[] = [];\n    for (const entry of [\"api\", \"web\", \"app\"]) {\n      const srcPath = path.join(Sonamu.appRootPath, entry, \"src\");\n      if (fs.existsSync(srcPath)) {\n        searchPaths.push(srcPath);\n      }\n    }\n\n    if (searchPaths.length === 0) {\n      return {\n        error: \"검색할 src 디렉토리를 찾을 수 없습니다.\",\n        unusedKeys: [],\n      };\n    }\n\n    const usedKeys = new Set<string>();\n\n    try {\n      // ast-grep으로 SD(\"...\") 패턴 검색\n      // 패턴: SD(\"KEY\") 또는 SD('KEY') 형태\n      const patterns = ['SD(\"$KEY\")', \"SD('$KEY')\"];\n\n      for (const searchPath of searchPaths) {\n        for (const pattern of patterns) {\n          try {\n            const result = execSync(`${sgPath} --pattern '${pattern}' --json ${searchPath}`, {\n              encoding: \"utf-8\",\n              maxBuffer: 50 * 1024 * 1024, // 50MB\n            });\n\n            if (result.trim()) {\n              const matches = JSON.parse(result);\n              for (const match of matches) {\n                // metaVariables.single.KEY.text에서 키 추출\n                const keyText = match.metaVariables?.single?.KEY?.text;\n                if (keyText) {\n                  // 따옴표 제거\n                  const cleanKey = keyText.replace(/^[\"']|[\"']$/g, \"\");\n                  usedKeys.add(cleanKey);\n                }\n              }\n            }\n          } catch {\n            // 패턴 매치 없으면 에러 (무시)\n          }\n        }\n      }\n\n      // keys 중에서 usedKeys에 없는 것들이 미사용 키\n      const unusedKeys = keys.filter((k) => !usedKeys.has(k));\n\n      return { unusedKeys, usedKeysCount: usedKeys.size };\n    } catch (e) {\n      return {\n        error: `검색 중 오류 발생: ${e instanceof Error ? e.message : String(e)}`,\n        unusedKeys: [],\n      };\n    }\n  }\n}\n\nexport const sonamuDictionary = new SonamuDictionary();\n"],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"code-generation.d.ts","sourceRoot":"","sources":["../../src/migration/code-generation.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,KAAK,IAAI,EAAE,MAAM,MAAM,CAAC;AAKjC,OAAO,EAEL,KAAK,gBAAgB,EAGrB,KAAK,cAAc,EACnB,KAAK,YAAY,EAClB,MAAM,gBAAgB,CAAC;
|
|
1
|
+
{"version":3,"file":"code-generation.d.ts","sourceRoot":"","sources":["../../src/migration/code-generation.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,KAAK,IAAI,EAAE,MAAM,MAAM,CAAC;AAKjC,OAAO,EAEL,KAAK,gBAAgB,EAGrB,KAAK,cAAc,EACnB,KAAK,YAAY,EAClB,MAAM,gBAAgB,CAAC;AAqhDxB;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,aAAa,EAAE,cAAc,EAAE,EAAE,SAAS,EAAE,cAAc,EAAE;SAG/E,cAAc,EAAE;UACf,cAAc,EAAE;EAuC/B;AAWD;;GAEG;AACH,wBAAgB,yBAAyB,CAAC,KAAK,EAAE,cAAc,GAAG,cAAc,CAoB/E;AAuHD;;;;GAIG;AACH,wBAAsB,kBAAkB,CAAC,SAAS,EAAE,YAAY,GAAG,OAAO,CAAC,gBAAgB,EAAE,CAAC,CAS7F;AAED;;;;;;GAMG;AACH,wBAAsB,iBAAiB,CACrC,SAAS,EAAE,YAAY,EACvB,KAAK,EAAE,YAAY,EACnB,SAAS,CAAC,EAAE,IAAI,GACf,OAAO,CAAC,gBAAgB,EAAE,CAAC,CA0G7B"}
|