sonamu 0.7.30 → 0.7.31

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.
@@ -1 +1 @@
1
- {"version":3,"file":"sd.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/sd.template.ts"],"names":[],"mappings":"AAGA,OAAO,EAAiB,KAAK,iBAAiB,EAAE,MAAM,6BAA6B,CAAC;AACpF,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAEzD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC;;;GAGG;AACH,qBAAa,YAAa,SAAQ,QAAQ;;IAKxC,gBAAgB,CAAC,MAAM,CAAC,EAAE,iBAAiB,EAAE,QAAQ,CAAC,EAAE,KAAK,GAAG,KAAK,GAAG,KAAK;;;;IAW7E,MAAM,CAAC,OAAO,EAAE,eAAe,CAAC,IAAI,CAAC;;;;;;;IAkLrC,OAAO,CAAC,UAAU;IAIlB;;;;;;OAMG;IACH,OAAO,CAAC,mBAAmB;IAiC3B;;OAEG;IACH,OAAO,CAAC,wBAAwB;IAYhC,OAAO,CAAC,YAAY;IAIpB;;OAEG;IACH,OAAO,CAAC,0BAA0B;CAyBnC"}
1
+ {"version":3,"file":"sd.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/sd.template.ts"],"names":[],"mappings":"AAGA,OAAO,EAAiB,KAAK,iBAAiB,EAAE,MAAM,6BAA6B,CAAC;AACpF,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAEzD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC;;;GAGG;AACH,qBAAa,YAAa,SAAQ,QAAQ;;IAKxC,gBAAgB,CAAC,MAAM,CAAC,EAAE,iBAAiB,EAAE,QAAQ,CAAC,EAAE,KAAK,GAAG,KAAK,GAAG,KAAK;;;;IAW7E,MAAM,CAAC,OAAO,EAAE,eAAe,CAAC,IAAI,CAAC;;;;;;;IAsLrC,OAAO,CAAC,UAAU;IAIlB;;;;;;OAMG;IACH,OAAO,CAAC,mBAAmB;IAiC3B;;OAEG;IACH,OAAO,CAAC,wBAAwB;IAYhC,OAAO,CAAC,YAAY;IAIpB;;OAEG;IACH,OAAO,CAAC,0BAA0B;CAyBnC"}
@@ -94,7 +94,10 @@ export function defineLocale(dict: Partial<MergedDictionary>) {
94
94
  // 각 locale별로 entity labels + Sonamu 내장 dict + 프로젝트 dict 합침
95
95
  const dictionaries: Record<string, Partial<MergedDictionary>> = {
96
96
  ${defaultLocale}: { ...sonamuDict${this.capitalize(defaultLocale)}, ...entityLabels, ...${defaultLocale} },
97
- ${supportedLocales.filter((locale)=>locale !== defaultLocale).map((locale)=>` ${locale}: { ...sonamuDict${this.capitalize(locale)}, ...${locale} },`).join("\n")}
97
+ ${supportedLocales.filter((locale)=>locale !== defaultLocale).map((locale)=>[
98
+ "en",
99
+ "ko"
100
+ ].includes(locale) ? ` ${locale}: { ...sonamuDict${this.capitalize(locale)}, ...${locale} },` : ` ${locale}: { ...${locale} },`).join("\n")}
98
101
  };
99
102
 
100
103
  type SDReturnType<K extends DictKey> = MergedDictionary[K] extends (...args: infer P) => string
@@ -264,4 +267,4 @@ ${entryLines.join("\n")}
264
267
  }
265
268
  }
266
269
 
267
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/sd.template.ts"],"sourcesContent":["import fs from \"fs\";\nimport path from \"path\";\nimport { Sonamu } from \"../../api/sonamu\";\nimport { EntityManager, type EntityNamesRecord } from \"../../entity/entity-manager\";\nimport type { TemplateOptions } from \"../../types/types\";\nimport { parseDictFile } from \"../../utils/dict-parser\";\nimport { Template } from \"../template\";\n\n/**\n * Sonamu Dictionary (SD) 템플릿\n * i18n을 위한 sd.generated.ts 파일을 생성합니다.\n */\nexport class Template__sd extends Template {\n  constructor() {\n    super(\"sd\");\n  }\n\n  getTargetAndPath(_names?: EntityNamesRecord, sdTarget?: \"api\" | \"web\" | \"app\") {\n    const target = sdTarget ?? \"api\";\n    // api.dir은 상대 경로(\"api\")이므로, web/app도 상대 경로로 맞춤\n    const dir = target === \"api\" ? Sonamu.config.api.dir : target;\n\n    return {\n      target: `${dir}/src/i18n`,\n      path: \"sd.generated.ts\",\n    };\n  }\n\n  render(options: TemplateOptions[\"sd\"]) {\n    const { target } = options;\n    const i18nConfig = Sonamu.config.i18n ?? {\n      defaultLocale: \"ko\",\n      supportedLocales: [\"ko\"],\n    };\n\n    const { defaultLocale, supportedLocales } = i18nConfig;\n\n    // entity.json에서 entity labels 추출\n    const entityLabels = this.extractEntityLabels();\n\n    // 플랫폼별 locale 관리 코드\n    const localeManagementCode =\n      target === \"api\"\n        ? `\nimport { Sonamu } from \"sonamu\";\n\nconst DEFAULT_LOCALE = \"${defaultLocale}\";\nconst SUPPORTED_LOCALES = ${JSON.stringify(supportedLocales)};\nfunction getCurrentLocale(): string {\n  const ctx = Sonamu.getContext();\n  return ctx?.locale ?? DEFAULT_LOCALE;\n}\n`.trim()\n        : `\nconst DEFAULT_LOCALE = \"${defaultLocale}\";\nconst SUPPORTED_LOCALES = ${JSON.stringify(supportedLocales)};\nlet _currentLocale = DEFAULT_LOCALE;\n\nexport function setLocale(locale: string) {\n  _currentLocale = locale;\n}\n\nexport function getCurrentLocale(): string {\n  return _currentLocale;\n}\n`.trim();\n\n    // sonamuDict를 소스 파일에서 파싱하여 코드로 변환 (타입 정보 보존)\n    const sonamuDictKoCode = this.generateDictCodeFromSource(\"sonamuDictKo\", \"ko\");\n    const sonamuDictEnCode = this.generateDictCodeFromSource(\"sonamuDictEn\", \"en\");\n\n    // locale import\n    const localeImports = supportedLocales\n      .map((locale) => `import ${locale} from \"./${locale}\";`)\n      .join(\"\\n\");\n\n    // entityLabels를 코드로 변환\n    const entityLabelsCode = this.generateEntityLabelsCode(entityLabels);\n\n    const body = `\n${localeManagementCode}\n\n${localeImports}\n\n// entity.json에서 추출한 entity labels (defaultLocale 전용)\n${entityLabelsCode}\n\n${sonamuDictKoCode}\n${sonamuDictEnCode}\n\n// defaultLocale의 dictionary를 기준으로 키 추출\ntype ProjectDictionary = typeof ${defaultLocale};\ntype SonamuDictionary = typeof sonamuDict${this.capitalize(defaultLocale)};\ntype EntityLabels = typeof entityLabels;\ntype RawMergedDictionary = EntityLabels & SonamuDictionary & ProjectDictionary;\n\n// 키는 유지하되, 값 타입은 string 또는 함수로 일반화 (다른 locale의 리터럴 타입 충돌 방지)\ntype MergedDictionary = {\n  [K in keyof RawMergedDictionary]: RawMergedDictionary[K] extends (...args: infer P) => string\n    ? (...args: P) => string\n    : string;\n};\ntype DictKey = keyof MergedDictionary;\nexport type LocalizedString = string & { __brand: \"LocalizedString\" };\n\nexport function defineLocale(dict: Partial<MergedDictionary>) {\n  return dict;\n}\n\n// 각 locale별로 entity labels + Sonamu 내장 dict + 프로젝트 dict 합침\nconst dictionaries: Record<string, Partial<MergedDictionary>> = {\n  ${defaultLocale}: { ...sonamuDict${this.capitalize(defaultLocale)}, ...entityLabels, ...${defaultLocale} },\n  ${supportedLocales\n    .filter((locale) => locale !== defaultLocale)\n    .map((locale) => `  ${locale}: { ...sonamuDict${this.capitalize(locale)}, ...${locale} },`)\n    .join(\"\\n\")}\n};\n\ntype SDReturnType<K extends DictKey> = MergedDictionary[K] extends (...args: infer P) => string\n  ? (...args: P) => LocalizedString\n  : LocalizedString;\n\nfunction getDictValue<K extends DictKey>(key: K, locale: string): SDReturnType<K> {\n  const dict = dictionaries[locale];\n  const value = dict?.[key] ?? dictionaries[DEFAULT_LOCALE]?.[key] ?? key;\n  return value as unknown as SDReturnType<K>;\n}\n\n/**\n * Sonamu Dictionary 함수\n * locale에 맞는 번역 텍스트를 반환합니다.\n *\n * @example\n * SD(\"common.save\")  // → \"저장\" (LocalizedString)\n * SD(\"user.notFound\")(1)  // → \"존재하지 않는 User ID 1\" (LocalizedString)\n */\nexport function SD<K extends DictKey>(key: K): SDReturnType<K> {\n  const locale = getCurrentLocale();\n  return getDictValue(key, locale);\n}\n\n/**\n * 특정 locale의 번역 텍스트를 반환하는 함수를 생성합니다.\n *\n * @example\n * const EN = SD.locale(\"en\");\n * EN(\"common.save\")  // → \"Save\"\n */\nSD.locale = (locale: string) => <K extends DictKey>(key: K): SDReturnType<K> => {\n  return getDictValue(key, locale);\n};\n\n/**\n * locale에 따라 적절한 컬럼 값을 반환합니다.\n * DB에 name, name_ko, name_en처럼 localized column이 있을 때 사용합니다.\n *\n * 우선순위 (ko locale): column_ko → column → column_en\n * 우선순위 (en locale): column_en → column → column_ko\n *\n * @example\n * localizedColumn(tag, \"name\")\n */\nexport function localizedColumn<T extends Record<string, unknown>, K extends keyof T & string>(\n  row: T,\n  column: K,\n): string | undefined {\n  const locale = getCurrentLocale();\n  const otherLocales = SUPPORTED_LOCALES.filter((l: string) => l !== locale);\n  const localizedKey = (column: K, locale: string) => \\`\\${String(column)}_\\${locale}\\`;\n  const keys = [localizedKey(column, locale), column, ...otherLocales.map((l) => localizedKey(column, l))];\n\n  for (const key of keys) {\n    const value = row[key];\n    if (value != null && value !== \"\") {\n      return String(value);\n    }\n  }\n\n  return undefined;\n}\n\n/**\n * Enum의 localized labels를 Proxy로 반환합니다.\n * Select 컴포넌트 등에서 EnumLabel[key] 대신 사용합니다.\n *\n * @example\n * SD.enumLabels(\"TagOrderBy\")[key]  // → 현재 locale의 라벨\n */\nSD.enumLabels = (enumName: string): Record<string, LocalizedString> => {\n  return new Proxy({} as Record<string, LocalizedString>, {\n    get(_, key: string) {\n      const dictKey = \\`enum.\\${enumName}.\\${key}\\` as DictKey;\n      return getDictValue(dictKey, getCurrentLocale());\n    }\n  });\n};\n`.trim();\n\n    return {\n      ...this.getTargetAndPath(undefined, target),\n      body,\n      importKeys: [],\n      customHeaders: [],\n    };\n  }\n\n  private capitalize(str: string): string {\n    return str.charAt(0).toUpperCase() + str.slice(1);\n  }\n\n  /**\n   * 모든 entity.json에서 entity labels 추출\n   * entity.json에서 직접 관리되는 값만 포함 (자동 생성 값 제외)\n   * - entity.{entityId}: entity title\n   * - entity.{entityId}.{propName}: prop desc\n   * - enum.{EnumId}.{value}: enum label\n   */\n  private extractEntityLabels(): { key: string; value: string }[] {\n    const labels: { key: string; value: string }[] = [];\n\n    if (!EntityManager.isAutoloaded) {\n      return labels;\n    }\n\n    const entityIds = EntityManager.getAllIds();\n\n    for (const entityId of entityIds) {\n      const entity = EntityManager.get(entityId);\n\n      // entity title (entity.json에서 관리)\n      labels.push({ key: `entity.${entityId}`, value: entity.title });\n\n      // prop labels (entity.json에서 관리)\n      for (const prop of entity.props) {\n        if (prop.desc) {\n          labels.push({ key: `entity.${entityId}.${prop.name}`, value: prop.desc });\n        }\n      }\n\n      // enum labels (entity.json에서 관리)\n      for (const [enumId, enumLabelsMap] of Object.entries(entity.enumLabels)) {\n        for (const [value, label] of Object.entries(enumLabelsMap)) {\n          labels.push({ key: `enum.${enumId}.${value}`, value: label });\n        }\n      }\n    }\n\n    return labels;\n  }\n\n  /**\n   * entityLabels를 TypeScript 코드로 변환\n   */\n  private generateEntityLabelsCode(labels: { key: string; value: string }[]): string {\n    if (labels.length === 0) {\n      return \"const entityLabels = {} as const;\";\n    }\n\n    const entries = labels.map(({ key, value }) => `  \"${key}\": \"${this.escapeString(value)}\",`);\n\n    return `const entityLabels = {\n${entries.join(\"\\n\")}\n} as const;`;\n  }\n\n  private escapeString(str: string): string {\n    return str.replace(/\\\\/g, \"\\\\\\\\\").replace(/\"/g, '\\\\\"').replace(/\\n/g, \"\\\\n\");\n  }\n\n  /**\n   * sonamu dict 소스 파일을 읽어 파싱하고 코드로 변환\n   */\n  private generateDictCodeFromSource(varName: string, locale: string): string {\n    // sonamu 패키지 루트에서 src/dict 경로 찾기\n    // __dirname이 dist/template/implementations일 수 있으므로 패키지 루트 기준으로 접근\n    const packageRoot = path.resolve(import.meta.dirname, \"..\", \"..\", \"..\");\n    const dictPath = path.join(packageRoot, \"src\", \"dict\", `${locale}.ts`);\n\n    if (!fs.existsSync(dictPath)) {\n      return `const ${varName} = {};`;\n    }\n\n    const entries = parseDictFile(dictPath);\n\n    if (entries.length === 0) {\n      return `const ${varName} = {};`;\n    }\n\n    const entryLines = entries.map(({ key, value, isFunction }) => {\n      const codeValue = isFunction ? value : `\"${this.escapeString(value)}\"`;\n      return `  \"${key}\": ${codeValue},`;\n    });\n\n    return `const ${varName} = {\n${entryLines.join(\"\\n\")}\n};`;\n  }\n}\n"],"names":["fs","path","Sonamu","EntityManager","parseDictFile","Template","Template__sd","getTargetAndPath","_names","sdTarget","target","dir","config","api","render","options","i18nConfig","i18n","defaultLocale","supportedLocales","entityLabels","extractEntityLabels","localeManagementCode","JSON","stringify","trim","sonamuDictKoCode","generateDictCodeFromSource","sonamuDictEnCode","localeImports","map","locale","join","entityLabelsCode","generateEntityLabelsCode","body","capitalize","filter","undefined","importKeys","customHeaders","str","charAt","toUpperCase","slice","labels","isAutoloaded","entityIds","getAllIds","entityId","entity","get","push","key","value","title","prop","props","desc","name","enumId","enumLabelsMap","Object","entries","enumLabels","label","length","escapeString","replace","varName","packageRoot","resolve","dirname","dictPath","existsSync","entryLines","isFunction","codeValue"],"mappings":"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"}
270
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/sd.template.ts"],"sourcesContent":["import fs from \"fs\";\nimport path from \"path\";\nimport { Sonamu } from \"../../api/sonamu\";\nimport { EntityManager, type EntityNamesRecord } from \"../../entity/entity-manager\";\nimport type { TemplateOptions } from \"../../types/types\";\nimport { parseDictFile } from \"../../utils/dict-parser\";\nimport { Template } from \"../template\";\n\n/**\n * Sonamu Dictionary (SD) 템플릿\n * i18n을 위한 sd.generated.ts 파일을 생성합니다.\n */\nexport class Template__sd extends Template {\n  constructor() {\n    super(\"sd\");\n  }\n\n  getTargetAndPath(_names?: EntityNamesRecord, sdTarget?: \"api\" | \"web\" | \"app\") {\n    const target = sdTarget ?? \"api\";\n    // api.dir은 상대 경로(\"api\")이므로, web/app도 상대 경로로 맞춤\n    const dir = target === \"api\" ? Sonamu.config.api.dir : target;\n\n    return {\n      target: `${dir}/src/i18n`,\n      path: \"sd.generated.ts\",\n    };\n  }\n\n  render(options: TemplateOptions[\"sd\"]) {\n    const { target } = options;\n    const i18nConfig = Sonamu.config.i18n ?? {\n      defaultLocale: \"ko\",\n      supportedLocales: [\"ko\"],\n    };\n\n    const { defaultLocale, supportedLocales } = i18nConfig;\n\n    // entity.json에서 entity labels 추출\n    const entityLabels = this.extractEntityLabels();\n\n    // 플랫폼별 locale 관리 코드\n    const localeManagementCode =\n      target === \"api\"\n        ? `\nimport { Sonamu } from \"sonamu\";\n\nconst DEFAULT_LOCALE = \"${defaultLocale}\";\nconst SUPPORTED_LOCALES = ${JSON.stringify(supportedLocales)};\nfunction getCurrentLocale(): string {\n  const ctx = Sonamu.getContext();\n  return ctx?.locale ?? DEFAULT_LOCALE;\n}\n`.trim()\n        : `\nconst DEFAULT_LOCALE = \"${defaultLocale}\";\nconst SUPPORTED_LOCALES = ${JSON.stringify(supportedLocales)};\nlet _currentLocale = DEFAULT_LOCALE;\n\nexport function setLocale(locale: string) {\n  _currentLocale = locale;\n}\n\nexport function getCurrentLocale(): string {\n  return _currentLocale;\n}\n`.trim();\n\n    // sonamuDict를 소스 파일에서 파싱하여 코드로 변환 (타입 정보 보존)\n    const sonamuDictKoCode = this.generateDictCodeFromSource(\"sonamuDictKo\", \"ko\");\n    const sonamuDictEnCode = this.generateDictCodeFromSource(\"sonamuDictEn\", \"en\");\n\n    // locale import\n    const localeImports = supportedLocales\n      .map((locale) => `import ${locale} from \"./${locale}\";`)\n      .join(\"\\n\");\n\n    // entityLabels를 코드로 변환\n    const entityLabelsCode = this.generateEntityLabelsCode(entityLabels);\n\n    const body = `\n${localeManagementCode}\n\n${localeImports}\n\n// entity.json에서 추출한 entity labels (defaultLocale 전용)\n${entityLabelsCode}\n\n${sonamuDictKoCode}\n${sonamuDictEnCode}\n\n// defaultLocale의 dictionary를 기준으로 키 추출\ntype ProjectDictionary = typeof ${defaultLocale};\ntype SonamuDictionary = typeof sonamuDict${this.capitalize(defaultLocale)};\ntype EntityLabels = typeof entityLabels;\ntype RawMergedDictionary = EntityLabels & SonamuDictionary & ProjectDictionary;\n\n// 키는 유지하되, 값 타입은 string 또는 함수로 일반화 (다른 locale의 리터럴 타입 충돌 방지)\ntype MergedDictionary = {\n  [K in keyof RawMergedDictionary]: RawMergedDictionary[K] extends (...args: infer P) => string\n    ? (...args: P) => string\n    : string;\n};\ntype DictKey = keyof MergedDictionary;\nexport type LocalizedString = string & { __brand: \"LocalizedString\" };\n\nexport function defineLocale(dict: Partial<MergedDictionary>) {\n  return dict;\n}\n\n// 각 locale별로 entity labels + Sonamu 내장 dict + 프로젝트 dict 합침\nconst dictionaries: Record<string, Partial<MergedDictionary>> = {\n  ${defaultLocale}: { ...sonamuDict${this.capitalize(defaultLocale)}, ...entityLabels, ...${defaultLocale} },\n  ${supportedLocales\n    .filter((locale) => locale !== defaultLocale)\n    .map((locale) =>\n      [\"en\", \"ko\"].includes(locale)\n        ? `  ${locale}: { ...sonamuDict${this.capitalize(locale)}, ...${locale} },`\n        : `  ${locale}: { ...${locale} },`,\n    )\n    .join(\"\\n\")}\n};\n\ntype SDReturnType<K extends DictKey> = MergedDictionary[K] extends (...args: infer P) => string\n  ? (...args: P) => LocalizedString\n  : LocalizedString;\n\nfunction getDictValue<K extends DictKey>(key: K, locale: string): SDReturnType<K> {\n  const dict = dictionaries[locale];\n  const value = dict?.[key] ?? dictionaries[DEFAULT_LOCALE]?.[key] ?? key;\n  return value as unknown as SDReturnType<K>;\n}\n\n/**\n * Sonamu Dictionary 함수\n * locale에 맞는 번역 텍스트를 반환합니다.\n *\n * @example\n * SD(\"common.save\")  // → \"저장\" (LocalizedString)\n * SD(\"user.notFound\")(1)  // → \"존재하지 않는 User ID 1\" (LocalizedString)\n */\nexport function SD<K extends DictKey>(key: K): SDReturnType<K> {\n  const locale = getCurrentLocale();\n  return getDictValue(key, locale);\n}\n\n/**\n * 특정 locale의 번역 텍스트를 반환하는 함수를 생성합니다.\n *\n * @example\n * const EN = SD.locale(\"en\");\n * EN(\"common.save\")  // → \"Save\"\n */\nSD.locale = (locale: string) => <K extends DictKey>(key: K): SDReturnType<K> => {\n  return getDictValue(key, locale);\n};\n\n/**\n * locale에 따라 적절한 컬럼 값을 반환합니다.\n * DB에 name, name_ko, name_en처럼 localized column이 있을 때 사용합니다.\n *\n * 우선순위 (ko locale): column_ko → column → column_en\n * 우선순위 (en locale): column_en → column → column_ko\n *\n * @example\n * localizedColumn(tag, \"name\")\n */\nexport function localizedColumn<T extends Record<string, unknown>, K extends keyof T & string>(\n  row: T,\n  column: K,\n): string | undefined {\n  const locale = getCurrentLocale();\n  const otherLocales = SUPPORTED_LOCALES.filter((l: string) => l !== locale);\n  const localizedKey = (column: K, locale: string) => \\`\\${String(column)}_\\${locale}\\`;\n  const keys = [localizedKey(column, locale), column, ...otherLocales.map((l) => localizedKey(column, l))];\n\n  for (const key of keys) {\n    const value = row[key];\n    if (value != null && value !== \"\") {\n      return String(value);\n    }\n  }\n\n  return undefined;\n}\n\n/**\n * Enum의 localized labels를 Proxy로 반환합니다.\n * Select 컴포넌트 등에서 EnumLabel[key] 대신 사용합니다.\n *\n * @example\n * SD.enumLabels(\"TagOrderBy\")[key]  // → 현재 locale의 라벨\n */\nSD.enumLabels = (enumName: string): Record<string, LocalizedString> => {\n  return new Proxy({} as Record<string, LocalizedString>, {\n    get(_, key: string) {\n      const dictKey = \\`enum.\\${enumName}.\\${key}\\` as DictKey;\n      return getDictValue(dictKey, getCurrentLocale());\n    }\n  });\n};\n`.trim();\n\n    return {\n      ...this.getTargetAndPath(undefined, target),\n      body,\n      importKeys: [],\n      customHeaders: [],\n    };\n  }\n\n  private capitalize(str: string): string {\n    return str.charAt(0).toUpperCase() + str.slice(1);\n  }\n\n  /**\n   * 모든 entity.json에서 entity labels 추출\n   * entity.json에서 직접 관리되는 값만 포함 (자동 생성 값 제외)\n   * - entity.{entityId}: entity title\n   * - entity.{entityId}.{propName}: prop desc\n   * - enum.{EnumId}.{value}: enum label\n   */\n  private extractEntityLabels(): { key: string; value: string }[] {\n    const labels: { key: string; value: string }[] = [];\n\n    if (!EntityManager.isAutoloaded) {\n      return labels;\n    }\n\n    const entityIds = EntityManager.getAllIds();\n\n    for (const entityId of entityIds) {\n      const entity = EntityManager.get(entityId);\n\n      // entity title (entity.json에서 관리)\n      labels.push({ key: `entity.${entityId}`, value: entity.title });\n\n      // prop labels (entity.json에서 관리)\n      for (const prop of entity.props) {\n        if (prop.desc) {\n          labels.push({ key: `entity.${entityId}.${prop.name}`, value: prop.desc });\n        }\n      }\n\n      // enum labels (entity.json에서 관리)\n      for (const [enumId, enumLabelsMap] of Object.entries(entity.enumLabels)) {\n        for (const [value, label] of Object.entries(enumLabelsMap)) {\n          labels.push({ key: `enum.${enumId}.${value}`, value: label });\n        }\n      }\n    }\n\n    return labels;\n  }\n\n  /**\n   * entityLabels를 TypeScript 코드로 변환\n   */\n  private generateEntityLabelsCode(labels: { key: string; value: string }[]): string {\n    if (labels.length === 0) {\n      return \"const entityLabels = {} as const;\";\n    }\n\n    const entries = labels.map(({ key, value }) => `  \"${key}\": \"${this.escapeString(value)}\",`);\n\n    return `const entityLabels = {\n${entries.join(\"\\n\")}\n} as const;`;\n  }\n\n  private escapeString(str: string): string {\n    return str.replace(/\\\\/g, \"\\\\\\\\\").replace(/\"/g, '\\\\\"').replace(/\\n/g, \"\\\\n\");\n  }\n\n  /**\n   * sonamu dict 소스 파일을 읽어 파싱하고 코드로 변환\n   */\n  private generateDictCodeFromSource(varName: string, locale: string): string {\n    // sonamu 패키지 루트에서 src/dict 경로 찾기\n    // __dirname이 dist/template/implementations일 수 있으므로 패키지 루트 기준으로 접근\n    const packageRoot = path.resolve(import.meta.dirname, \"..\", \"..\", \"..\");\n    const dictPath = path.join(packageRoot, \"src\", \"dict\", `${locale}.ts`);\n\n    if (!fs.existsSync(dictPath)) {\n      return `const ${varName} = {};`;\n    }\n\n    const entries = parseDictFile(dictPath);\n\n    if (entries.length === 0) {\n      return `const ${varName} = {};`;\n    }\n\n    const entryLines = entries.map(({ key, value, isFunction }) => {\n      const codeValue = isFunction ? value : `\"${this.escapeString(value)}\"`;\n      return `  \"${key}\": ${codeValue},`;\n    });\n\n    return `const ${varName} = {\n${entryLines.join(\"\\n\")}\n};`;\n  }\n}\n"],"names":["fs","path","Sonamu","EntityManager","parseDictFile","Template","Template__sd","getTargetAndPath","_names","sdTarget","target","dir","config","api","render","options","i18nConfig","i18n","defaultLocale","supportedLocales","entityLabels","extractEntityLabels","localeManagementCode","JSON","stringify","trim","sonamuDictKoCode","generateDictCodeFromSource","sonamuDictEnCode","localeImports","map","locale","join","entityLabelsCode","generateEntityLabelsCode","body","capitalize","filter","includes","undefined","importKeys","customHeaders","str","charAt","toUpperCase","slice","labels","isAutoloaded","entityIds","getAllIds","entityId","entity","get","push","key","value","title","prop","props","desc","name","enumId","enumLabelsMap","Object","entries","enumLabels","label","length","escapeString","replace","varName","packageRoot","resolve","dirname","dictPath","existsSync","entryLines","isFunction","codeValue"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "sonamu",
3
- "version": "0.7.30",
3
+ "version": "0.7.31",
4
4
  "description": "Sonamu — TypeScript Fullstack API Framework",
5
5
  "keywords": [
6
6
  "typescript",
@@ -111,10 +111,10 @@
111
111
  "vitest": "^4.0.10",
112
112
  "xlsx": "^0.18.5",
113
113
  "zod": "^4.1.12",
114
- "@sonamu-kit/hmr-hook": "^0.4.1",
115
114
  "@sonamu-kit/tasks": "^0.1.3",
116
- "@sonamu-kit/ts-loader": "^2.1.3",
117
- "@sonamu-kit/hmr-runner": "^0.1.1"
115
+ "@sonamu-kit/hmr-runner": "^0.1.1",
116
+ "@sonamu-kit/hmr-hook": "^0.4.1",
117
+ "@sonamu-kit/ts-loader": "^2.1.3"
118
118
  },
119
119
  "devDependencies": {
120
120
  "@biomejs/biome": "^2.3.10",
@@ -112,7 +112,11 @@ const dictionaries: Record<string, Partial<MergedDictionary>> = {
112
112
  ${defaultLocale}: { ...sonamuDict${this.capitalize(defaultLocale)}, ...entityLabels, ...${defaultLocale} },
113
113
  ${supportedLocales
114
114
  .filter((locale) => locale !== defaultLocale)
115
- .map((locale) => ` ${locale}: { ...sonamuDict${this.capitalize(locale)}, ...${locale} },`)
115
+ .map((locale) =>
116
+ ["en", "ko"].includes(locale)
117
+ ? ` ${locale}: { ...sonamuDict${this.capitalize(locale)}, ...${locale} },`
118
+ : ` ${locale}: { ...${locale} },`,
119
+ )
116
120
  .join("\n")}
117
121
  };
118
122