mirror of
https://github.com/CherryHQ/cherry-studio.git
synced 2025-12-19 06:30:10 +08:00
* refactor: rename i18n commands for better consistency - Rename `check:i18n` to `i18n:check` - Rename `sync:i18n` to `i18n:sync` - Rename `update:i18n` to `i18n:translate` (clearer purpose) - Rename `auto:i18n` to `i18n:all` (runs check, sync, and translate) - Update lint script to use new `i18n:check` command name This follows the common naming convention of grouping related commands under a namespace prefix (e.g., `test:main`, `test:renderer`). 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * refactor: update i18n command names and improve documentation - Renamed i18n commands for consistency: `sync:i18n` to `i18n:sync`, `check:i18n` to `i18n:check`, and `auto:i18n` to `i18n:translate`. - Updated relevant documentation and scripts to reflect new command names. - Improved formatting and clarity in i18n-related guides and scripts. This change enhances the clarity and usability of i18n commands across the project. --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
153 lines
4.8 KiB
TypeScript
153 lines
4.8 KiB
TypeScript
import * as fs from 'fs'
|
|
import * as path from 'path'
|
|
|
|
import { sortedObjectByKeys } from './sort'
|
|
|
|
const translationsDir = path.join(__dirname, '../src/renderer/src/i18n/locales')
|
|
const baseLocale = process.env.BASE_LOCALE ?? 'zh-cn'
|
|
const baseFileName = `${baseLocale}.json`
|
|
const baseFilePath = path.join(translationsDir, baseFileName)
|
|
|
|
type I18NValue = string | { [key: string]: I18NValue }
|
|
type I18N = { [key: string]: I18NValue }
|
|
|
|
/**
|
|
* 递归检查并同步目标对象与模板对象的键值结构
|
|
* 1. 如果目标对象缺少模板对象中的键,抛出错误
|
|
* 2. 如果目标对象存在模板对象中不存在的键,抛出错误
|
|
* 3. 对于嵌套对象,递归执行同步操作
|
|
*
|
|
* 该函数用于确保所有翻译文件与基准模板(通常是中文翻译文件)保持完全一致的键值结构。
|
|
* 任何结构上的差异都会导致错误被抛出,以便及时发现和修复翻译文件中的问题。
|
|
*
|
|
* @param target 需要检查的目标翻译对象
|
|
* @param template 作为基准的模板对象(通常是中文翻译文件)
|
|
* @throws {Error} 当发现键值结构不匹配时抛出错误
|
|
*/
|
|
function checkRecursively(target: I18N, template: I18N): void {
|
|
for (const key in template) {
|
|
if (!(key in target)) {
|
|
throw new Error(`缺少属性 ${key}`)
|
|
}
|
|
if (key.includes('.')) {
|
|
throw new Error(`应该使用严格嵌套结构 ${key}`)
|
|
}
|
|
if (typeof template[key] === 'object' && template[key] !== null) {
|
|
if (typeof target[key] !== 'object' || target[key] === null) {
|
|
throw new Error(`属性 ${key} 不是对象`)
|
|
}
|
|
// 递归检查子对象
|
|
checkRecursively(target[key], template[key])
|
|
}
|
|
}
|
|
|
|
// 删除 target 中存在但 template 中没有的 key
|
|
for (const targetKey in target) {
|
|
if (!(targetKey in template)) {
|
|
throw new Error(`多余属性 ${targetKey}`)
|
|
}
|
|
}
|
|
}
|
|
|
|
function isSortedI18N(obj: I18N): boolean {
|
|
// fs.writeFileSync('./test_origin.json', JSON.stringify(obj))
|
|
// fs.writeFileSync('./test_sorted.json', JSON.stringify(sortedObjectByKeys(obj)))
|
|
return JSON.stringify(obj) === JSON.stringify(sortedObjectByKeys(obj))
|
|
}
|
|
|
|
/**
|
|
* 检查 JSON 对象中是否存在重复键,并收集所有重复键
|
|
* @param obj 要检查的对象
|
|
* @returns 返回重复键的数组(若无重复则返回空数组)
|
|
*/
|
|
function checkDuplicateKeys(obj: I18N): string[] {
|
|
const keys = new Set<string>()
|
|
const duplicateKeys: string[] = []
|
|
|
|
const checkObject = (obj: I18N, path: string = '') => {
|
|
for (const key in obj) {
|
|
const fullPath = path ? `${path}.${key}` : key
|
|
|
|
if (keys.has(fullPath)) {
|
|
// 发现重复键时,添加到数组中(避免重复添加)
|
|
if (!duplicateKeys.includes(fullPath)) {
|
|
duplicateKeys.push(fullPath)
|
|
}
|
|
} else {
|
|
keys.add(fullPath)
|
|
}
|
|
|
|
// 递归检查子对象
|
|
if (typeof obj[key] === 'object' && obj[key] !== null) {
|
|
checkObject(obj[key], fullPath)
|
|
}
|
|
}
|
|
}
|
|
|
|
checkObject(obj)
|
|
return duplicateKeys
|
|
}
|
|
|
|
function checkTranslations() {
|
|
if (!fs.existsSync(baseFilePath)) {
|
|
throw new Error(`主模板文件 ${baseFileName} 不存在,请检查路径或文件名`)
|
|
}
|
|
|
|
const baseContent = fs.readFileSync(baseFilePath, 'utf-8')
|
|
let baseJson: I18N = {}
|
|
try {
|
|
baseJson = JSON.parse(baseContent)
|
|
} catch (error) {
|
|
throw new Error(`解析 ${baseFileName} 出错。${error}`)
|
|
}
|
|
|
|
// 检查主模板是否存在重复键
|
|
const duplicateKeys = checkDuplicateKeys(baseJson)
|
|
if (duplicateKeys.length > 0) {
|
|
throw new Error(`主模板文件 ${baseFileName} 存在以下重复键:\n${duplicateKeys.join('\n')}`)
|
|
}
|
|
|
|
// 检查主模板是否有序
|
|
if (!isSortedI18N(baseJson)) {
|
|
throw new Error(`主模板文件 ${baseFileName} 的键值未按字典序排序。`)
|
|
}
|
|
|
|
const files = fs.readdirSync(translationsDir).filter((file) => file.endsWith('.json') && file !== baseFileName)
|
|
|
|
// 同步键
|
|
for (const file of files) {
|
|
const filePath = path.join(translationsDir, file)
|
|
let targetJson: I18N = {}
|
|
try {
|
|
const fileContent = fs.readFileSync(filePath, 'utf-8')
|
|
targetJson = JSON.parse(fileContent)
|
|
} catch (error) {
|
|
throw new Error(`解析 ${file} 出错。`)
|
|
}
|
|
|
|
// 检查有序性
|
|
if (!isSortedI18N(targetJson)) {
|
|
throw new Error(`翻译文件 ${file} 的键值未按字典序排序。`)
|
|
}
|
|
|
|
try {
|
|
checkRecursively(targetJson, baseJson)
|
|
} catch (e) {
|
|
console.error(e)
|
|
throw new Error(`在检查 ${filePath} 时出错`)
|
|
}
|
|
}
|
|
}
|
|
|
|
export function main() {
|
|
try {
|
|
checkTranslations()
|
|
console.log('i18n 检查已通过')
|
|
} catch (e) {
|
|
console.error(e)
|
|
throw new Error(`检查未通过。尝试运行 yarn i18n:sync 以解决问题。`)
|
|
}
|
|
}
|
|
|
|
main()
|