mirror of
https://github.com/coaidev/coai.git
synced 2025-05-19 13:00:14 +09:00
219 lines
24 KiB
JavaScript
219 lines
24 KiB
JavaScript
// vite.config.ts
|
|
import { defineConfig } from "file:///I:/Projects/chatnio/app/node_modules/.pnpm/vite@4.5.0_@types+node@20.8.9_less@4.2.0/node_modules/vite/dist/node/index.js";
|
|
import react from "file:///I:/Projects/chatnio/app/node_modules/.pnpm/@vitejs+plugin-react-swc@3.4.0_vite@4.5.0/node_modules/@vitejs/plugin-react-swc/index.mjs";
|
|
import path3 from "path";
|
|
import { createHtmlPlugin } from "file:///I:/Projects/chatnio/app/node_modules/.pnpm/vite-plugin-html@3.2.0_vite@4.5.0/node_modules/vite-plugin-html/dist/index.mjs";
|
|
|
|
// src/translator/translator.ts
|
|
import path2 from "path";
|
|
import fs2 from "fs";
|
|
|
|
// src/translator/io.ts
|
|
import fs from "fs";
|
|
import path from "path";
|
|
function readJSON(...paths) {
|
|
return JSON.parse(fs.readFileSync(path.resolve(...paths)).toString());
|
|
}
|
|
function writeJSON(data, ...paths) {
|
|
fs.writeFileSync(path.resolve(...paths), JSON.stringify(data, null, 2));
|
|
}
|
|
function getMigration(mother, data, prefix) {
|
|
return Object.keys(mother).map((key) => {
|
|
const template = mother[key], translation = data !== void 0 && key in data ? data[key] : void 0;
|
|
const val = [prefix.length === 0 ? key : `${prefix}.${key}`];
|
|
switch (typeof template) {
|
|
case "string":
|
|
if (typeof translation !== "string")
|
|
return val;
|
|
else if (template.startsWith("!!"))
|
|
return val;
|
|
break;
|
|
case "object":
|
|
return getMigration(template, translation, val[0]);
|
|
default:
|
|
return typeof translation === typeof template ? [] : val;
|
|
}
|
|
return [];
|
|
}).flat().filter((key) => key !== void 0 && key.length > 0);
|
|
}
|
|
function getFields(data) {
|
|
switch (typeof data) {
|
|
case "string":
|
|
return 1;
|
|
case "object":
|
|
if (Array.isArray(data))
|
|
return data.length;
|
|
return Object.keys(data).reduce(
|
|
(acc, key) => acc + getFields(data[key]),
|
|
0
|
|
);
|
|
default:
|
|
return 1;
|
|
}
|
|
}
|
|
function getTranslation(data, path4) {
|
|
const keys = path4.split(".");
|
|
let current = data;
|
|
for (const key of keys) {
|
|
if (current[key] === void 0)
|
|
return void 0;
|
|
current = current[key];
|
|
}
|
|
return current;
|
|
}
|
|
function setTranslation(data, path4, value) {
|
|
const keys = path4.split(".");
|
|
let current = data;
|
|
for (let i = 0; i < keys.length - 1; i++) {
|
|
if (current[keys[i]] === void 0)
|
|
current[keys[i]] = {};
|
|
current = current[keys[i]];
|
|
}
|
|
current[keys[keys.length - 1]] = value;
|
|
}
|
|
|
|
// src/translator/adapter.ts
|
|
var languageTranslatorMap = {
|
|
cn: "zh-CN",
|
|
en: "en",
|
|
ru: "ru",
|
|
ja: "ja",
|
|
ko: "ko",
|
|
fr: "fr",
|
|
de: "de",
|
|
es: "es",
|
|
pt: "pt",
|
|
it: "it"
|
|
};
|
|
function getFormattedLanguage(lang) {
|
|
return languageTranslatorMap[lang.toLowerCase()] || lang;
|
|
}
|
|
async function translate(text, from, to) {
|
|
if (from === to || text.length === 0)
|
|
return text;
|
|
const resp = await fetch(
|
|
`https://api.mymemory.translated.net/get?q=${encodeURIComponent(
|
|
text
|
|
)}&langpair=${from}|${to}`
|
|
);
|
|
const data = await resp.json();
|
|
return data.responseData.translatedText;
|
|
}
|
|
function doTranslate(content, from, to) {
|
|
from = getFormattedLanguage(from);
|
|
to = getFormattedLanguage(to);
|
|
if (content.startsWith("!!"))
|
|
content = content.substring(2);
|
|
return translate(content, from, to);
|
|
}
|
|
|
|
// src/translator/translator.ts
|
|
var defaultDevLang = "cn";
|
|
async function processTranslation(config) {
|
|
const source = path2.resolve(config.root, "src/resources/i18n");
|
|
const files = fs2.readdirSync(source);
|
|
const motherboard = `${defaultDevLang}.json`;
|
|
if (files.length === 0) {
|
|
console.warn("no translation files found");
|
|
return;
|
|
} else if (!files.includes(motherboard)) {
|
|
console.warn(`no default translation file found (${defaultDevLang}.json)`);
|
|
return;
|
|
}
|
|
const data = readJSON(source, motherboard);
|
|
const target = files.filter((file) => file !== motherboard);
|
|
for (const file of target) {
|
|
const lang = file.split(".")[0];
|
|
const translation = { ...readJSON(source, file) };
|
|
const fields = getFields(data);
|
|
const migration = getMigration(data, translation, "");
|
|
const total = migration.length;
|
|
let current = 0;
|
|
for (const key of migration) {
|
|
const from = getTranslation(data, key);
|
|
const to = typeof from === "string" ? await doTranslate(from, defaultDevLang, lang) : from;
|
|
current++;
|
|
console.log(
|
|
`[i18n] successfully translated: ${from} -> ${to} (lang: ${defaultDevLang} -> ${lang}, progress: ${current}/${total})`
|
|
);
|
|
setTranslation(translation, key, to);
|
|
}
|
|
if (migration.length > 0) {
|
|
writeJSON(translation, source, file);
|
|
}
|
|
console.info(
|
|
`translation file ${file} loaded, ${fields} fields detected, ${migration.length} migration(s) applied`
|
|
);
|
|
}
|
|
}
|
|
|
|
// src/translator/index.ts
|
|
function createTranslationPlugin() {
|
|
return {
|
|
name: "translate-plugin",
|
|
apply: "build",
|
|
async configResolved(config) {
|
|
try {
|
|
console.info("[i18n] start translation process");
|
|
await processTranslation(config);
|
|
} catch (e) {
|
|
console.warn(`error during translation: ${e}`);
|
|
} finally {
|
|
console.info("[i18n] translation process finished");
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
// vite.config.ts
|
|
var __vite_injected_original_dirname = "I:\\Projects\\chatnio\\app";
|
|
var vite_config_default = defineConfig({
|
|
plugins: [
|
|
react(),
|
|
createHtmlPlugin({
|
|
minify: true
|
|
}),
|
|
createTranslationPlugin()
|
|
],
|
|
resolve: {
|
|
alias: {
|
|
"@": path3.resolve(__vite_injected_original_dirname, "./src")
|
|
}
|
|
},
|
|
css: {
|
|
preprocessorOptions: {
|
|
less: {
|
|
javascriptEnabled: true
|
|
}
|
|
}
|
|
},
|
|
build: {
|
|
manifest: true,
|
|
chunkSizeWarningLimit: 2048,
|
|
rollupOptions: {
|
|
output: {
|
|
entryFileNames: `assets/[name].[hash].js`,
|
|
chunkFileNames: `assets/[name].[hash].js`
|
|
}
|
|
}
|
|
},
|
|
server: {
|
|
proxy: {
|
|
"/api": {
|
|
target: "http://localhost:8094",
|
|
changeOrigin: true,
|
|
rewrite: (path4) => path4.replace(/^\/api/, ""),
|
|
ws: true
|
|
},
|
|
"/v1": {
|
|
target: "http://localhost:8094",
|
|
changeOrigin: true
|
|
}
|
|
}
|
|
}
|
|
});
|
|
export {
|
|
vite_config_default as default
|
|
};
|
|
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["vite.config.ts", "src/translator/translator.ts", "src/translator/io.ts", "src/translator/adapter.ts", "src/translator/index.ts"],
  "sourcesContent": ["const __vite_injected_original_dirname = \"I:\\\\Projects\\\\chatnio\\\\app\";const __vite_injected_original_filename = \"I:\\\\Projects\\\\chatnio\\\\app\\\\vite.config.ts\";const __vite_injected_original_import_meta_url = \"file:///I:/Projects/chatnio/app/vite.config.ts\";import { defineConfig } from 'vite'\r\nimport react from '@vitejs/plugin-react-swc'\r\nimport path from \"path\"\r\nimport { createHtmlPlugin } from 'vite-plugin-html'\r\nimport { createTranslationPlugin } from \"./src/translator\";\r\n\r\n// https://vitejs.dev/config/\r\nexport default defineConfig({\r\n  plugins: [\r\n    react(),\r\n    createHtmlPlugin({\r\n      minify: true,\r\n    }),\r\n    createTranslationPlugin(),\r\n  ],\r\n  resolve: {\r\n    alias: {\r\n      \"@\": path.resolve(__dirname, \"./src\"),\r\n    },\r\n  },\r\n  css: {\r\n    preprocessorOptions: {\r\n      less: {\r\n        javascriptEnabled: true,\r\n      }\r\n    }\r\n  },\r\n  build: {\r\n    manifest: true,\r\n    chunkSizeWarningLimit: 2048,\r\n    rollupOptions: {\r\n      output: {\r\n        entryFileNames: `assets/[name].[hash].js`,\r\n        chunkFileNames: `assets/[name].[hash].js`,\r\n      },\r\n    },\r\n  },\r\n  server: {\r\n    proxy: {\r\n      \"/api\": {\r\n        target: \"http://localhost:8094\",\r\n        changeOrigin: true,\r\n        rewrite: (path) => path.replace(/^\\/api/, \"\"),\r\n        ws: true,\r\n      },\r\n      \"/v1\": {\r\n        target: \"http://localhost:8094\",\r\n        changeOrigin: true,\r\n      }\r\n    }\r\n  }\r\n});\r\n", "const __vite_injected_original_dirname = \"I:\\\\Projects\\\\chatnio\\\\app\\\\src\\\\translator\";const __vite_injected_original_filename = \"I:\\\\Projects\\\\chatnio\\\\app\\\\src\\\\translator\\\\translator.ts\";const __vite_injected_original_import_meta_url = \"file:///I:/Projects/chatnio/app/src/translator/translator.ts\";import { ResolvedConfig } from \"vite\";\r\nimport path from \"path\";\r\nimport fs from \"fs\";\r\nimport {\r\n  getFields,\r\n  getMigration,\r\n  getTranslation,\r\n  readJSON,\r\n  setTranslation,\r\n  writeJSON,\r\n} from \"./io\";\r\nimport { doTranslate } from \"./adapter\";\r\n\r\nexport const defaultDevLang = \"cn\";\r\n\r\nexport async function processTranslation(\r\n  config: ResolvedConfig,\r\n): Promise<void> {\r\n  const source = path.resolve(config.root, \"src/resources/i18n\");\r\n  const files = fs.readdirSync(source);\r\n\r\n  const motherboard = `${defaultDevLang}.json`;\r\n\r\n  if (files.length === 0) {\r\n    console.warn(\"no translation files found\");\r\n    return;\r\n  } else if (!files.includes(motherboard)) {\r\n    console.warn(`no default translation file found (${defaultDevLang}.json)`);\r\n    return;\r\n  }\r\n\r\n  const data = readJSON(source, motherboard);\r\n\r\n  const target = files.filter((file) => file !== motherboard);\r\n  for (const file of target) {\r\n    const lang = file.split(\".\")[0];\r\n    const translation = { ...readJSON(source, file) };\r\n\r\n    const fields = getFields(data);\r\n    const migration = getMigration(data, translation, \"\");\r\n    const total = migration.length;\r\n    let current = 0;\r\n    for (const key of migration) {\r\n      const from = getTranslation(data, key);\r\n      const to =\r\n        typeof from === \"string\"\r\n          ? await doTranslate(from, defaultDevLang, lang)\r\n          : from;\r\n      current++;\r\n\r\n      console.log(\r\n        `[i18n] successfully translated: ${from} -> ${to} (lang: ${defaultDevLang} -> ${lang}, progress: ${current}/${total})`,\r\n      );\r\n      setTranslation(translation, key, to);\r\n    }\r\n\r\n    if (migration.length > 0) {\r\n      writeJSON(translation, source, file);\r\n    }\r\n\r\n    console.info(\r\n      `translation file ${file} loaded, ${fields} fields detected, ${migration.length} migration(s) applied`,\r\n    );\r\n  }\r\n}\r\n", "const __vite_injected_original_dirname = \"I:\\\\Projects\\\\chatnio\\\\app\\\\src\\\\translator\";const __vite_injected_original_filename = \"I:\\\\Projects\\\\chatnio\\\\app\\\\src\\\\translator\\\\io.ts\";const __vite_injected_original_import_meta_url = \"file:///I:/Projects/chatnio/app/src/translator/io.ts\";import fs from \"fs\";\r\nimport path from \"path\";\r\n\r\nexport function readJSON(...paths: string[]): any {\r\n  return JSON.parse(fs.readFileSync(path.resolve(...paths)).toString());\r\n}\r\n\r\nexport function writeJSON(data: any, ...paths: string[]): void {\r\n  fs.writeFileSync(path.resolve(...paths), JSON.stringify(data, null, 2));\r\n}\r\n\r\nexport function getMigration(\r\n  mother: Record<string, any>,\r\n  data: Record<string, any>,\r\n  prefix: string,\r\n): string[] {\r\n  return Object.keys(mother)\r\n    .map((key): string[] => {\r\n      const template = mother[key],\r\n        translation = data !== undefined && key in data ? data[key] : undefined;\r\n      const val = [prefix.length === 0 ? key : `${prefix}.${key}`];\r\n\r\n      switch (typeof template) {\r\n        case \"string\":\r\n          if (typeof translation !== \"string\") return val;\r\n          else if (template.startsWith(\"!!\")) return val;\r\n          break;\r\n        case \"object\":\r\n          return getMigration(template, translation, val[0]);\r\n        default:\r\n          return typeof translation === typeof template ? [] : val;\r\n      }\r\n\r\n      return [];\r\n    })\r\n    .flat()\r\n    .filter((key) => key !== undefined && key.length > 0);\r\n}\r\n\r\nexport function getFields(data: any): number {\r\n  switch (typeof data) {\r\n    case \"string\":\r\n      return 1;\r\n    case \"object\":\r\n      if (Array.isArray(data)) return data.length;\r\n      return Object.keys(data).reduce(\r\n        (acc, key) => acc + getFields(data[key]),\r\n        0,\r\n      );\r\n    default:\r\n      return 1;\r\n  }\r\n}\r\n\r\nexport function getTranslation(data: Record<string, any>, path: string): any {\r\n  const keys = path.split(\".\");\r\n  let current = data;\r\n  for (const key of keys) {\r\n    if (current[key] === undefined) return undefined;\r\n    current = current[key];\r\n  }\r\n  return current;\r\n}\r\n\r\nexport function setTranslation(\r\n  data: Record<string, any>,\r\n  path: string,\r\n  value: any,\r\n): void {\r\n  const keys = path.split(\".\");\r\n  let current = data;\r\n  for (let i = 0; i < keys.length - 1; i++) {\r\n    if (current[keys[i]] === undefined) current[keys[i]] = {};\r\n    current = current[keys[i]];\r\n  }\r\n  current[keys[keys.length - 1]] = value;\r\n}\r\n", "const __vite_injected_original_dirname = \"I:\\\\Projects\\\\chatnio\\\\app\\\\src\\\\translator\";const __vite_injected_original_filename = \"I:\\\\Projects\\\\chatnio\\\\app\\\\src\\\\translator\\\\adapter.ts\";const __vite_injected_original_import_meta_url = \"file:///I:/Projects/chatnio/app/src/translator/adapter.ts\";// format language code to name/ISO 639-1 code map\r\nconst languageTranslatorMap: Record<string, string> = {\r\n  cn: \"zh-CN\",\r\n  en: \"en\",\r\n  ru: \"ru\",\r\n  ja: \"ja\",\r\n  ko: \"ko\",\r\n  fr: \"fr\",\r\n  de: \"de\",\r\n  es: \"es\",\r\n  pt: \"pt\",\r\n  it: \"it\",\r\n};\r\n\r\nexport function getFormattedLanguage(lang: string): string {\r\n  return languageTranslatorMap[lang.toLowerCase()] || lang;\r\n}\r\n\r\ntype translationResponse = {\r\n  responseData: {\r\n    translatedText: string;\r\n  };\r\n};\r\n\r\nasync function translate(\r\n  text: string,\r\n  from: string,\r\n  to: string,\r\n): Promise<string> {\r\n  if (from === to || text.length === 0) return text;\r\n  const resp = await fetch(\r\n    `https://api.mymemory.translated.net/get?q=${encodeURIComponent(\r\n      text,\r\n    )}&langpair=${from}|${to}`,\r\n  );\r\n  const data: translationResponse = await resp.json();\r\n\r\n  return data.responseData.translatedText;\r\n}\r\n\r\nexport function doTranslate(\r\n  content: string,\r\n  from: string,\r\n  to: string,\r\n): Promise<string> {\r\n  from = getFormattedLanguage(from);\r\n  to = getFormattedLanguage(to);\r\n\r\n  if (content.startsWith(\"!!\")) content = content.substring(2);\r\n\r\n  return translate(content, from, to);\r\n}\r\n", "const __vite_injected_original_dirname = \"I:\\\\Projects\\\\chatnio\\\\app\\\\src\\\\translator\";const __vite_injected_original_filename = \"I:\\\\Projects\\\\chatnio\\\\app\\\\src\\\\translator\\\\index.ts\";const __vite_injected_original_import_meta_url = \"file:///I:/Projects/chatnio/app/src/translator/index.ts\";import { Plugin, ResolvedConfig } from \"vite\";\r\nimport { processTranslation } from \"./translator\";\r\n\r\nexport function createTranslationPlugin(): Plugin {\r\n  return {\r\n    name: \"translate-plugin\",\r\n    apply: \"build\",\r\n    async configResolved(config: ResolvedConfig) {\r\n      try {\r\n        console.info(\"[i18n] start translation process\");\r\n        await processTranslation(config);\r\n      } catch (e) {\r\n        console.warn(`error during translation: ${e}`);\r\n      } finally {\r\n        console.info(\"[i18n] translation process finished\");\r\n      }\r\n    },\r\n  };\r\n}\r\n"],
  "mappings": "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",
  "names": ["path", "path", "fs", "path", "path", "fs", "path"]
}

|