clean_translations.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. const fs = require('fs');
  2. const path = require('path');
  3. const filePath = process.argv[2];
  4. if (!filePath) {
  5. console.error('Please provide a file path');
  6. process.exit(1);
  7. }
  8. const content = fs.readFileSync(filePath, 'utf8');
  9. // These are missing keys that we want to ensure exist in each language block
  10. const missingKeysData = {
  11. kbSettingsSaved: { zh: "检索与对话配置已保存", en: "Knowledge base settings saved", ja: "設定を保存しました" },
  12. failedToSaveSettings: { zh: "保存设置失败", en: "Failed to save settings", ja: "設定の保存に失敗しました" },
  13. actionFailed: { zh: "操作失败", en: "Action failed", ja: "操作に失敗しました" },
  14. userAddedToOrganization: { zh: "用户已添加到组织", en: "User added to organization", ja: "ユーザーが組織に追加されました" },
  15. featureUpdated: { zh: "功能已更新", en: "Feature updated", ja: "機能が更新されました" },
  16. roleTenantAdmin: { zh: "租户管理员", en: "Tenant Administrator", ja: "テナント管理者" },
  17. roleRegularUser: { zh: "普通用户", en: "Regular User", ja: "一般ユーザー" },
  18. creatingRegularUser: { zh: "正在创建普通用户", en: "Creating regular user", ja: "一般ユーザーを作成中" },
  19. editUserRole: { zh: "修改用户角色", en: "Edit user role", ja: "ユーザーロールを編集" },
  20. targetRole: { zh: "目标角色", en: "Target Role", ja: "対象のロール" },
  21. editCategory: { zh: "编辑分类", en: "Edit category", ja: "カテゴリを編集" },
  22. totalTenants: { zh: "总租户数", en: "Total Tenants", ja: "総テナント数" },
  23. systemUsers: { zh: "系统用户", en: "System Users", ja: "システムユーザー" },
  24. systemHealth: { zh: "系统健康", en: "System Health", ja: "システムヘルス" },
  25. operational: { zh: "运行正常", en: "Operational", ja: "正常稼働中" },
  26. orgManagement: { zh: "组织管理", en: "Organization Management", ja: "組織管理" },
  27. globalTenantControl: { zh: "全局租户控制", en: "Global Tenant Control", ja: "グローバルテナントコントロール" },
  28. newTenant: { zh: "新租户", en: "New Tenant", ja: "新規テナント" },
  29. domainOptional: { zh: "域名 (可选)", en: "Domain (Optional)", ja: "ドメイン (任意)" },
  30. saveChanges: { zh: "保存修改", en: "Save changes", ja: "変更を保存" },
  31. modelConfiguration: { zh: "模型配置", en: "Model Configuration", ja: "モデル設定" },
  32. defaultLLMModel: { zh: "默认推理模型", en: "Default LLM Model", ja: "デフォルト推論モデル" },
  33. selectLLM: { zh: "选择 LLM", en: "Select LLM", ja: "LLMを選択" },
  34. selectEmbedding: { zh: "选择 Embedding", en: "Select Embedding", ja: "埋め込みを選択" },
  35. rerankModel: { zh: "Rerank 模型", en: "Rerank Model", ja: "リランクモデル" },
  36. none: { zh: "无", en: "None", ja: "なし" },
  37. indexingChunkingConfig: { zh: "索引与切片配置", en: "Indexing & Chunking Config", ja: "インデックスとチャンク設定" },
  38. chatHyperparameters: { zh: "聊天超参数", en: "Chat Hyperparameters", ja: "チャットハイパーパラメータ" },
  39. temperature: { zh: "随机性 (Temperature)", en: "Temperature", ja: "温度" },
  40. precise: { zh: "精确", en: "Precise", ja: "精密" },
  41. creative: { zh: "创意", en: "Creative", ja: "クリエイティブ" },
  42. maxResponseTokens: { zh: "最大响应标识 (Max Tokens)", en: "Max Response Tokens", ja: "最大応答トークン数" },
  43. retrievalSearchSettings: { zh: "检索与搜索设置", en: "Retrieval & Search Settings", ja: "検索設定" },
  44. topK: { zh: "召回数量 (Top K)", en: "Top K", ja: "Top K" },
  45. similarityThreshold: { zh: "相似度阈值", en: "Similarity Threshold", ja: "類似度しきい値" },
  46. enableHybridSearch: { zh: "启用混合检索", en: "Enable Hybrid Search", ja: "ハイブリッド検索を有効にする" },
  47. hybridSearchDesc: { zh: "同时使用向量和全文检索以提高召回率", en: "Use both vector and full-text search to improve recall", ja: "ベクトル検索と全文検索を併用して検索精度を向上させます" },
  48. hybridWeight: { zh: "混合权重 (0.0=全文, 1.0=向量)", en: "Hybrid Weight (0.0=Fulltext, 1.0=Vector)", ja: "ハイブリッド重み (0.0=全文, 1.0=ベクトル)" },
  49. pureText: { zh: "纯文本", en: "Pure Text", ja: "純粋なテキスト" },
  50. pureVector: { zh: "纯向量", en: "Pure Vector", ja: "純粋なベクトル" },
  51. enableQueryExpansion: { zh: "启用查询扩展", en: "Enable Query Expansion", ja: "クエリ拡張を有効にする" },
  52. queryExpansionDesc: { zh: "生成多个查询变体以提高覆盖率", en: "Generate multiple query variations for better coverage", ja: "複数のクエリバリアントを生成してカバレッジを向上させます" },
  53. enableHyDE: { zh: "启用 HyDE", en: "Enable HyDE", ja: "HyDEを有効にする" },
  54. hydeDesc: { zh: "生成假设回答以改善语义搜索", en: "Generate hypothetical answers to improve semantic search", ja: "仮想的な回答を生成してセマンティック検索を改善します" },
  55. enableReranking: { zh: "启用重排序 (Rerank)", en: "Enable Reranking", ja: "リランクを有効にする" },
  56. rerankingDesc: { zh: "使用 Rerank 模型对结果进行二次排序", en: "Use Rerank model to re-sort results", ja: "リランクモデルを使用して結果を再ソートします" },
  57. broad: { zh: "宽泛", en: "Broad", ja: "広範" },
  58. strict: { zh: "严格", en: "Strict", ja: "厳格" },
  59. maxInput: { zh: "最大输入", en: "Max Input", ja: "最大入力" },
  60. dimensions: { zh: "维度", en: "Dimensions", ja: "次元" },
  61. defaultBadge: { zh: "默认", en: "Default", ja: "デフォルト" },
  62. dims: { zh: "维度: $1", en: "Dims: $1", ja: "次元: $1" },
  63. ctx: { zh: "上下文: $1", en: "Ctx: $1", ja: "コンテキスト: $1" },
  64. baseApi: { zh: "Base API: $1", en: "Base API: $1", ja: "Base API: $1" },
  65. configured: { zh: "已配置", en: "Configured", ja: "設定済み" },
  66. groupUpdated: { zh: "分组已更新", en: "Group updated", ja: "グループが更新されました" },
  67. groupDeleted: { zh: "分组已删除", en: "Group deleted", ja: "グループが削除されました" },
  68. groupCreated: { zh: "分组已创建", en: "Group created", ja: "グループが作成されました" },
  69. navCatalog: { zh: "目录", en: "Catalog", ja: "カタログ" },
  70. allDocuments: { zh: "所有文档", en: "All Documents", ja: "すべてのドキュメント" },
  71. categories: { zh: "分类", en: "Categories", ja: "カテゴリ" },
  72. uncategorizedFiles: { zh: "未分类文件", en: "Uncategorized Files", ja: "未分類ファイル" },
  73. category: { zh: "分类", en: "Category", ja: "カテゴリ" },
  74. statusReadyDesc: { zh: "已索引可查询", en: "Indexed and searchable", ja: "インデックス済みで検索可能" },
  75. statusIndexingDesc: { zh: "正在建立词向量索引", en: "Building vector index", ja: "ベクトルインデックスを作成中" },
  76. selectCategory: { zh: "选择分类", en: "Select Category", ja: "カテゴリを選択" },
  77. noneUncategorized: { zh: "无未分类文件", en: "No uncategorized files", ja: "未分類ファイルなし" },
  78. previous: { zh: "上一页", en: "Previous", ja: "前へ" },
  79. next: { zh: "下一页", en: "Next", ja: "次へ" },
  80. createCategory: { zh: "创建分类", en: "Create Category", ja: "カテゴリを作成" },
  81. categoryDesc: { zh: "描述您的知识分类", en: "Describe your knowledge category", ja: "ナレッジカテゴリを説明します" },
  82. categoryName: { zh: "分类名称", en: "Category Name", ja: "カテゴリ名" },
  83. createCategoryBtn: { zh: "立即创建", en: "Create Now", ja: "今すぐ作成" },
  84. newGroup: { zh: "新建分组", en: "New Group", ja: "新規グループ" },
  85. noKnowledgeGroups: { zh: "暂无知识库分组", en: "No knowledge groups yet", ja: "ナレッジグループがまだありません" },
  86. createGroupDesc: { zh: "开始创建您的第一个知识库分组并上传相关文档。", en: "Start by creating your first knowledge group and uploading documents.", ja: "最初のナレッジグループを作成してドキュメントをアップロードしてください。" },
  87. noDescriptionProvided: { zh: "未提供描述", en: "No description provided", ja: "説明なし" },
  88. browseManageFiles: { zh: "浏览并管理该分组下的文件和笔记。", en: "Browse and manage files and notes in this group.", ja: "このグループ内のファイルとメモを閲覧・管理します。" },
  89. filterGroupFiles: { zh: "根据名称搜索分组内文件...", en: "Search files in group by name...", ja: "名前でグループ内のファイルを検索..." },
  90. generalSettingsSubtitle: { zh: "管理您的应用程序首选项。", en: "Manage your application preferences.", ja: "アプリケーションの設定を管理します。" },
  91. userManagementSubtitle: { zh: "管理访问权限和帐户。", en: "Manage access and accounts.", ja: "アクセス権限とアカウントを管理します。" },
  92. modelManagementSubtitle: { zh: "配置全局 AI 模型。", en: "Configure global AI models.", ja: "グローバルなAIモデルを設定します。" },
  93. kbSettingsSubtitle: { zh: "索引和聊天参数的技术配置。", en: "Technical configuration for indexing and chat parameters.", ja: "インデックス作成とチャットパラメータの技術設定。" },
  94. tenantsSubtitle: { zh: "全局系统概览。", en: "Global system overview.", ja: "グローバルシステムの概要。" },
  95. allNotes: { zh: "所有笔记", en: "All Notes", ja: "すべてのノート" },
  96. filterNotesPlaceholder: { zh: "筛选笔记...", en: "Filter notes...", ja: "ノートをフィルタリング..." },
  97. noteTitlePlaceholder: { zh: "标题...", en: "Title...", ja: "タイトル..." },
  98. startWritingPlaceholder: { zh: "开始写作...", en: "Start writing...", ja: "書き始める..." },
  99. previewHeader: { zh: "预览", en: "Preview", ja: "プレビュー" },
  100. noContentToPreview: { zh: "没有可预览的内容", en: "No content to preview", ja: "プレビューするコンテンツがありません" },
  101. hidePreview: { zh: "隐藏预览", en: "Hide Preview", ja: "プレビューを非表示" },
  102. showPreview: { zh: "显示预览", en: "Show Preview", ja: "プレビューを表示" },
  103. directoryLabel: { zh: "目录", en: "Directory", ja: "ディレクトリ" },
  104. uncategorized: { zh: "未分类", en: "Uncategorized", ja: "未分類" },
  105. enterNamePlaceholder: { zh: "输入名称...", en: "Enter name...", ja: "名前を入力..." },
  106. subFolderPlaceholder: { zh: "子文件夹...", en: "Sub-folder...", ja: "サブフォルダ..." },
  107. categoryCreated: { zh: "分类已创建", en: "Category created", ja: "カテゴリが作成されました" },
  108. failedToCreateCategory: { zh: "创建分类失败", en: "Failed to create category", ja: "カテゴリの作成に失敗しました" },
  109. failedToDeleteCategory: { zh: "删除分类失败", en: "Failed to delete category", ja: "カテゴリの削除に失敗しました" },
  110. confirmDeleteCategory: { zh: "您确定要删除此分类吗?", en: "Are you sure you want to delete this category?", ja: "このカテゴリを削除してもよろしいですか?" }
  111. };
  112. const lines = content.split('\n');
  113. let currentLang = null;
  114. let resultLines = [];
  115. let keysSeen = new Set();
  116. const langStartRegex = /^\s+(\w+): \{/;
  117. const keyRegex = /^\s+([a-zA-Z0-9_-]+):/;
  118. for (let i = 0; i < lines.length; i++) {
  119. const line = lines[i];
  120. const langMatch = line.match(langStartRegex);
  121. if (langMatch) {
  122. // If we were in a language block, append missing keys before finishing it
  123. if (currentLang) {
  124. addMissingKeys(currentLang, resultLines, keysSeen);
  125. }
  126. currentLang = langMatch[1];
  127. keysSeen = new Set();
  128. resultLines.push(line);
  129. continue;
  130. }
  131. if (currentLang) {
  132. const keyMatch = line.match(keyRegex);
  133. if (keyMatch) {
  134. const key = keyMatch[1];
  135. if (keysSeen.has(key)) {
  136. // Duplicate key, skip it
  137. continue;
  138. }
  139. keysSeen.add(key);
  140. }
  141. // If the line ends the block
  142. if (line.trim() === '},') {
  143. addMissingKeys(currentLang, resultLines, keysSeen);
  144. currentLang = null;
  145. resultLines.push(line);
  146. continue;
  147. }
  148. // Also handle the very last block which might not have a comma
  149. if (line.trim() === '}' && i > lines.length - 5) {
  150. addMissingKeys(currentLang, resultLines, keysSeen);
  151. currentLang = null;
  152. resultLines.push(line);
  153. continue;
  154. }
  155. }
  156. resultLines.push(line);
  157. }
  158. function addMissingKeys(lang, targetLines, seen) {
  159. for (const [key, translations] of Object.entries(missingKeysData)) {
  160. if (!seen.has(key)) {
  161. const val = translations[lang] || translations['en'] || key;
  162. const escapedVal = JSON.stringify(val);
  163. targetLines.push(` ${key}: ${escapedVal},`);
  164. seen.add(key);
  165. }
  166. }
  167. }
  168. fs.writeFileSync(filePath, resultLines.join('\n'), 'utf8');
  169. console.log('Translations file cleaned and updated successfully!');