2023-10-21 11:54:04 +08:00
|
|
|
import queryString from "query-string";
|
|
|
|
|
import {
|
|
|
|
|
OPT_TRANS_GOOGLE,
|
2025-07-02 13:38:30 +08:00
|
|
|
OPT_TRANS_GOOGLE_2,
|
2023-10-21 11:54:04 +08:00
|
|
|
OPT_TRANS_MICROSOFT,
|
|
|
|
|
OPT_TRANS_DEEPL,
|
|
|
|
|
OPT_TRANS_DEEPLFREE,
|
|
|
|
|
OPT_TRANS_DEEPLX,
|
2024-04-12 11:31:01 +08:00
|
|
|
OPT_TRANS_NIUTRANS,
|
2023-10-21 11:54:04 +08:00
|
|
|
OPT_TRANS_BAIDU,
|
|
|
|
|
OPT_TRANS_TENCENT,
|
2025-06-30 21:34:37 +08:00
|
|
|
OPT_TRANS_VOLCENGINE,
|
2023-10-21 11:54:04 +08:00
|
|
|
OPT_TRANS_OPENAI,
|
2023-12-21 14:15:14 +08:00
|
|
|
OPT_TRANS_GEMINI,
|
2025-07-02 21:54:18 +08:00
|
|
|
OPT_TRANS_GEMINI_2,
|
2024-09-23 18:22:19 +08:00
|
|
|
OPT_TRANS_CLAUDE,
|
2023-10-26 11:13:50 +08:00
|
|
|
OPT_TRANS_CLOUDFLAREAI,
|
2024-04-28 21:43:20 +08:00
|
|
|
OPT_TRANS_OLLAMA,
|
2025-08-23 11:01:54 +08:00
|
|
|
OPT_TRANS_OPENROUTER,
|
2023-10-21 11:54:04 +08:00
|
|
|
OPT_TRANS_CUSTOMIZE,
|
2025-09-24 23:24:00 +08:00
|
|
|
API_SPE_TYPES,
|
2024-04-17 17:38:54 +08:00
|
|
|
INPUT_PLACE_FROM,
|
|
|
|
|
INPUT_PLACE_TO,
|
2025-09-24 23:24:00 +08:00
|
|
|
// INPUT_PLACE_TEXT,
|
2024-04-17 17:38:54 +08:00
|
|
|
INPUT_PLACE_KEY,
|
2024-05-12 16:25:20 +08:00
|
|
|
INPUT_PLACE_MODEL,
|
2023-10-21 11:54:04 +08:00
|
|
|
} from "../config";
|
2024-04-21 13:16:44 +08:00
|
|
|
import { msAuth } from "../libs/auth";
|
|
|
|
|
import { genDeeplFree } from "./deepl";
|
|
|
|
|
import { genBaidu } from "./baidu";
|
2024-05-12 16:10:11 +08:00
|
|
|
import interpreter from "../libs/interpreter";
|
2025-09-03 00:37:35 +08:00
|
|
|
import { parseJsonObj, extractJson } from "../libs/utils";
|
|
|
|
|
import { kissLog } from "../libs/log";
|
|
|
|
|
import { fetchData } from "../libs/fetch";
|
2025-09-03 20:43:07 +08:00
|
|
|
import { getMsgHistory } from "./history";
|
2023-10-21 11:54:04 +08:00
|
|
|
|
2023-12-22 11:35:46 +08:00
|
|
|
const keyMap = new Map();
|
2024-04-20 14:01:34 +08:00
|
|
|
const urlMap = new Map();
|
2023-12-22 11:35:46 +08:00
|
|
|
|
2024-04-20 14:01:34 +08:00
|
|
|
// 轮询key/url
|
2025-09-24 23:24:00 +08:00
|
|
|
const keyPick = (apiSlug, key = "", cacheMap) => {
|
2023-12-22 11:35:46 +08:00
|
|
|
const keys = key
|
2024-01-19 16:02:53 +08:00
|
|
|
.split(/\n|,/)
|
2023-12-22 11:35:46 +08:00
|
|
|
.map((item) => item.trim())
|
|
|
|
|
.filter(Boolean);
|
|
|
|
|
|
|
|
|
|
if (keys.length === 0) {
|
|
|
|
|
return "";
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-24 23:24:00 +08:00
|
|
|
const preIndex = cacheMap.get(apiSlug) ?? -1;
|
2024-01-04 09:40:03 +08:00
|
|
|
const curIndex = (preIndex + 1) % keys.length;
|
2025-09-24 23:24:00 +08:00
|
|
|
cacheMap.set(apiSlug, curIndex);
|
2023-12-22 11:35:46 +08:00
|
|
|
|
2024-01-04 09:40:03 +08:00
|
|
|
return keys[curIndex];
|
2023-12-22 11:35:46 +08:00
|
|
|
};
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const genSystemPrompt = ({ systemPrompt, from, to }) =>
|
|
|
|
|
systemPrompt
|
|
|
|
|
.replaceAll(INPUT_PLACE_FROM, from)
|
|
|
|
|
.replaceAll(INPUT_PLACE_TO, to);
|
|
|
|
|
|
2025-09-24 23:24:00 +08:00
|
|
|
const genUserPrompt = ({
|
|
|
|
|
// userPrompt,
|
|
|
|
|
tone,
|
|
|
|
|
glossary = {},
|
|
|
|
|
// from,
|
|
|
|
|
to,
|
|
|
|
|
texts,
|
|
|
|
|
docInfo,
|
|
|
|
|
}) => {
|
2025-09-03 00:37:35 +08:00
|
|
|
const prompt = JSON.stringify({
|
|
|
|
|
targetLanguage: to,
|
|
|
|
|
title: docInfo.title,
|
|
|
|
|
description: docInfo.description,
|
|
|
|
|
segments: texts.map((text, i) => ({ id: i, text })),
|
2025-09-24 23:24:00 +08:00
|
|
|
glossary,
|
|
|
|
|
tone,
|
2025-09-03 00:37:35 +08:00
|
|
|
});
|
|
|
|
|
|
2025-09-24 23:24:00 +08:00
|
|
|
// if (userPrompt.includes(INPUT_PLACE_TEXT)) {
|
|
|
|
|
// return userPrompt
|
|
|
|
|
// .replaceAll(INPUT_PLACE_FROM, from)
|
|
|
|
|
// .replaceAll(INPUT_PLACE_TO, to)
|
|
|
|
|
// .replaceAll(INPUT_PLACE_TEXT, prompt);
|
|
|
|
|
// }
|
2025-09-03 00:37:35 +08:00
|
|
|
|
|
|
|
|
return prompt;
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 09:40:25 +08:00
|
|
|
const parseAIRes = (raw) => {
|
2025-09-03 00:37:35 +08:00
|
|
|
let data;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const jsonString = extractJson(raw);
|
|
|
|
|
data = JSON.parse(jsonString);
|
|
|
|
|
} catch (err) {
|
2025-09-24 23:24:00 +08:00
|
|
|
kissLog("parseAIRes", err);
|
|
|
|
|
return [];
|
2025-09-03 00:37:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!Array.isArray(data.translations)) {
|
2025-09-24 23:24:00 +08:00
|
|
|
return [];
|
2025-09-03 00:37:35 +08:00
|
|
|
}
|
|
|
|
|
|
2025-09-24 23:24:00 +08:00
|
|
|
// todo: 考虑序号id可能会打乱
|
|
|
|
|
return data.translations.map((item) => [
|
|
|
|
|
item?.text ?? "",
|
|
|
|
|
item?.sourceLanguage ?? "",
|
|
|
|
|
]);
|
2025-09-03 00:37:35 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const genGoogle = ({ texts, from, to, url, key }) => {
|
2025-06-26 11:13:51 +08:00
|
|
|
const params = {
|
|
|
|
|
client: "gtx",
|
|
|
|
|
dt: "t",
|
|
|
|
|
dj: 1,
|
|
|
|
|
ie: "UTF-8",
|
|
|
|
|
sl: from,
|
|
|
|
|
tl: to,
|
2025-09-03 00:37:35 +08:00
|
|
|
q: texts.join(" "),
|
2025-06-26 11:13:51 +08:00
|
|
|
};
|
|
|
|
|
const input = `${url}?${queryString.stringify(params)}`;
|
2023-10-21 11:54:04 +08:00
|
|
|
const init = {
|
|
|
|
|
headers: {
|
2025-06-26 11:13:51 +08:00
|
|
|
"Content-type": "application/json",
|
2023-10-21 11:54:04 +08:00
|
|
|
},
|
|
|
|
|
};
|
2025-06-26 11:13:51 +08:00
|
|
|
if (key) {
|
|
|
|
|
init.headers.Authorization = `Bearer ${key}`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [input, init];
|
2023-10-21 11:54:04 +08:00
|
|
|
};
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const genGoogle2 = ({ texts, from, to, url, key }) => {
|
|
|
|
|
const body = JSON.stringify([[texts, from, to], "wt_lib"]);
|
2025-06-27 19:29:00 +08:00
|
|
|
const init = {
|
|
|
|
|
method: "POST",
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-Type": "application/json+protobuf",
|
|
|
|
|
"X-Goog-API-Key": key,
|
|
|
|
|
},
|
|
|
|
|
body,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [url, init];
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const genMicrosoft = async ({ texts, from, to }) => {
|
2023-10-21 11:54:04 +08:00
|
|
|
const [token] = await msAuth();
|
|
|
|
|
const params = {
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
"api-version": "3.0",
|
|
|
|
|
};
|
2025-09-01 18:56:48 +08:00
|
|
|
const input = `https://api-edge.cognitive.microsofttranslator.com/translate?${queryString.stringify(params)}`;
|
2023-10-21 11:54:04 +08:00
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
Authorization: `Bearer ${token}`,
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
2025-09-03 00:37:35 +08:00
|
|
|
body: JSON.stringify(texts.map((text) => ({ Text: text }))),
|
2023-10-21 11:54:04 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [input, init];
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const genDeepl = ({ texts, from, to, url, key }) => {
|
2023-10-21 11:54:04 +08:00
|
|
|
const data = {
|
2025-09-03 00:37:35 +08:00
|
|
|
text: texts,
|
2023-10-21 11:54:04 +08:00
|
|
|
target_lang: to,
|
|
|
|
|
source_lang: from,
|
|
|
|
|
// split_sentences: "0",
|
|
|
|
|
};
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
Authorization: `DeepL-Auth-Key ${key}`,
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [url, init];
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const genDeeplX = ({ texts, from, to, url, key }) => {
|
2023-10-21 11:54:04 +08:00
|
|
|
const data = {
|
2025-09-03 00:37:35 +08:00
|
|
|
text: texts.join(" "),
|
2023-10-21 11:54:04 +08:00
|
|
|
target_lang: to,
|
|
|
|
|
source_lang: from,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
if (key) {
|
|
|
|
|
init.headers.Authorization = `Bearer ${key}`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [url, init];
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const genNiuTrans = ({ texts, from, to, url, key, dictNo, memoryNo }) => {
|
2024-04-12 11:31:01 +08:00
|
|
|
const data = {
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
apikey: key,
|
2025-09-03 00:37:35 +08:00
|
|
|
src_text: texts.join(" "),
|
2024-04-12 11:31:01 +08:00
|
|
|
dictNo,
|
|
|
|
|
memoryNo,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [url, init];
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const genTencent = ({ texts, from, to }) => {
|
2023-10-21 11:54:04 +08:00
|
|
|
const data = {
|
|
|
|
|
header: {
|
2025-06-27 20:03:58 +08:00
|
|
|
fn: "auto_translation",
|
2025-07-01 22:42:57 +08:00
|
|
|
client_key:
|
|
|
|
|
"browser-chrome-110.0.0-Mac OS-df4bd4c5-a65d-44b2-a40f-42f34f3535f2-1677486696487",
|
2023-10-21 11:54:04 +08:00
|
|
|
},
|
2025-06-27 20:03:58 +08:00
|
|
|
type: "plain",
|
|
|
|
|
model_category: "normal",
|
2023-10-21 11:54:04 +08:00
|
|
|
source: {
|
2025-09-03 00:37:35 +08:00
|
|
|
text_list: texts,
|
2023-10-21 11:54:04 +08:00
|
|
|
lang: from,
|
|
|
|
|
},
|
|
|
|
|
target: {
|
|
|
|
|
lang: to,
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-01 18:56:48 +08:00
|
|
|
const input = "https://transmart.qq.com/api/imt";
|
2023-10-21 11:54:04 +08:00
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-Type": "application/json",
|
2025-07-01 22:42:57 +08:00
|
|
|
"user-agent":
|
|
|
|
|
"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/110.0.0.0 Safari/537.36",
|
|
|
|
|
referer: "https://transmart.qq.com/zh-CN/index",
|
2023-10-21 11:54:04 +08:00
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-01 18:56:48 +08:00
|
|
|
return [input, init];
|
2023-10-21 11:54:04 +08:00
|
|
|
};
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const genVolcengine = ({ texts, from, to }) => {
|
2025-06-30 21:34:37 +08:00
|
|
|
const data = {
|
|
|
|
|
source_language: from,
|
|
|
|
|
target_language: to,
|
2025-09-03 00:37:35 +08:00
|
|
|
text: texts.join(" "),
|
2025-06-30 21:34:37 +08:00
|
|
|
};
|
|
|
|
|
|
2025-09-01 18:56:48 +08:00
|
|
|
const input = "https://translate.volcengine.com/crx/translate/v1";
|
2025-06-30 21:34:37 +08:00
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-01 18:56:48 +08:00
|
|
|
return [input, init];
|
2025-06-30 21:34:37 +08:00
|
|
|
};
|
|
|
|
|
|
2024-05-21 23:15:46 +08:00
|
|
|
const genOpenAI = ({
|
2025-09-03 00:37:35 +08:00
|
|
|
texts,
|
2024-05-21 23:15:46 +08:00
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
url,
|
|
|
|
|
key,
|
2024-09-25 14:03:12 +08:00
|
|
|
systemPrompt,
|
2024-11-30 00:41:29 +08:00
|
|
|
userPrompt,
|
2024-05-21 23:15:46 +08:00
|
|
|
model,
|
|
|
|
|
temperature,
|
|
|
|
|
maxTokens,
|
2025-08-11 16:28:23 +08:00
|
|
|
customHeader,
|
2025-08-11 16:03:30 +08:00
|
|
|
customBody,
|
2025-09-03 00:37:35 +08:00
|
|
|
docInfo,
|
2025-09-03 20:43:07 +08:00
|
|
|
hisMsgs,
|
2024-05-21 23:15:46 +08:00
|
|
|
}) => {
|
2025-09-03 00:37:35 +08:00
|
|
|
systemPrompt = genSystemPrompt({ systemPrompt, from, to });
|
|
|
|
|
userPrompt = genUserPrompt({ userPrompt, from, to, texts, docInfo });
|
2025-08-23 00:10:42 +08:00
|
|
|
customHeader = parseJsonObj(customHeader);
|
|
|
|
|
customBody = parseJsonObj(customBody);
|
2025-08-11 12:12:03 +08:00
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
const userMsg = {
|
|
|
|
|
role: "user",
|
|
|
|
|
content: userPrompt,
|
|
|
|
|
};
|
2023-10-21 11:54:04 +08:00
|
|
|
const data = {
|
|
|
|
|
model,
|
|
|
|
|
messages: [
|
|
|
|
|
{
|
|
|
|
|
role: "system",
|
2024-09-25 14:03:12 +08:00
|
|
|
content: systemPrompt,
|
2023-10-21 11:54:04 +08:00
|
|
|
},
|
2025-09-03 20:43:07 +08:00
|
|
|
...hisMsgs,
|
|
|
|
|
userMsg,
|
2023-10-21 11:54:04 +08:00
|
|
|
],
|
2024-05-21 23:15:46 +08:00
|
|
|
temperature,
|
2025-08-09 11:35:26 +08:00
|
|
|
max_completion_tokens: maxTokens,
|
2025-08-11 16:03:30 +08:00
|
|
|
...customBody,
|
2023-10-21 11:54:04 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
Authorization: `Bearer ${key}`, // OpenAI
|
|
|
|
|
"api-key": key, // Azure OpenAI
|
2025-08-11 16:28:23 +08:00
|
|
|
...customHeader,
|
2023-10-21 11:54:04 +08:00
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
return [url, init, userMsg];
|
2023-10-21 11:54:04 +08:00
|
|
|
};
|
|
|
|
|
|
2025-07-01 22:42:57 +08:00
|
|
|
const genGemini = ({
|
2025-09-03 00:37:35 +08:00
|
|
|
texts,
|
2025-07-01 22:42:57 +08:00
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
url,
|
|
|
|
|
key,
|
|
|
|
|
systemPrompt,
|
|
|
|
|
userPrompt,
|
|
|
|
|
model,
|
2025-07-02 16:37:57 +08:00
|
|
|
temperature,
|
|
|
|
|
maxTokens,
|
2025-08-11 16:28:23 +08:00
|
|
|
customHeader,
|
2025-08-11 16:03:30 +08:00
|
|
|
customBody,
|
2025-09-03 00:37:35 +08:00
|
|
|
docInfo,
|
2025-09-03 20:43:07 +08:00
|
|
|
hisMsgs,
|
2025-07-01 22:42:57 +08:00
|
|
|
}) => {
|
2024-05-12 16:25:20 +08:00
|
|
|
url = url
|
|
|
|
|
.replaceAll(INPUT_PLACE_MODEL, model)
|
|
|
|
|
.replaceAll(INPUT_PLACE_KEY, key);
|
2025-09-03 00:37:35 +08:00
|
|
|
systemPrompt = genSystemPrompt({ systemPrompt, from, to });
|
|
|
|
|
userPrompt = genUserPrompt({ userPrompt, from, to, texts, docInfo });
|
2025-08-23 00:10:42 +08:00
|
|
|
customHeader = parseJsonObj(customHeader);
|
|
|
|
|
customBody = parseJsonObj(customBody);
|
2025-08-11 12:12:03 +08:00
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
const userMsg = { role: "user", parts: [{ text: userPrompt }] };
|
2023-12-21 14:15:14 +08:00
|
|
|
const data = {
|
2024-11-30 00:41:29 +08:00
|
|
|
system_instruction: {
|
|
|
|
|
parts: {
|
|
|
|
|
text: systemPrompt,
|
2025-07-01 22:42:57 +08:00
|
|
|
},
|
2024-11-30 00:41:29 +08:00
|
|
|
},
|
2025-09-03 20:43:07 +08:00
|
|
|
contents: [...hisMsgs, userMsg],
|
2025-07-02 16:37:57 +08:00
|
|
|
generationConfig: {
|
|
|
|
|
maxOutputTokens: maxTokens,
|
|
|
|
|
temperature,
|
|
|
|
|
// topP: 0.8,
|
|
|
|
|
// topK: 10,
|
|
|
|
|
},
|
2025-09-03 12:04:56 +08:00
|
|
|
// thinkingConfig: {
|
|
|
|
|
// thinkingBudget: 0,
|
|
|
|
|
// },
|
|
|
|
|
safetySettings: [
|
|
|
|
|
{
|
|
|
|
|
category: "HARM_CATEGORY_HARASSMENT",
|
|
|
|
|
threshold: "BLOCK_NONE",
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
category: "HARM_CATEGORY_HATE_SPEECH",
|
|
|
|
|
threshold: "BLOCK_NONE",
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
category: "HARM_CATEGORY_SEXUALLY_EXPLICIT",
|
|
|
|
|
threshold: "BLOCK_NONE",
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
category: "HARM_CATEGORY_DANGEROUS_CONTENT",
|
|
|
|
|
threshold: "BLOCK_NONE",
|
|
|
|
|
},
|
|
|
|
|
],
|
2025-08-11 16:03:30 +08:00
|
|
|
...customBody,
|
2023-12-21 14:15:14 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
2025-08-11 16:28:23 +08:00
|
|
|
...customHeader,
|
2023-12-21 14:15:14 +08:00
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
return [url, init, userMsg];
|
2023-12-21 14:15:14 +08:00
|
|
|
};
|
|
|
|
|
|
2025-07-02 21:54:18 +08:00
|
|
|
const genGemini2 = ({
|
2025-09-03 00:37:35 +08:00
|
|
|
texts,
|
2025-07-02 21:54:18 +08:00
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
url,
|
|
|
|
|
key,
|
|
|
|
|
systemPrompt,
|
|
|
|
|
userPrompt,
|
|
|
|
|
model,
|
|
|
|
|
temperature,
|
2025-07-23 20:03:54 +08:00
|
|
|
maxTokens,
|
2025-08-11 16:28:23 +08:00
|
|
|
customHeader,
|
2025-08-11 16:03:30 +08:00
|
|
|
customBody,
|
2025-09-03 00:37:35 +08:00
|
|
|
docInfo,
|
2025-09-03 20:43:07 +08:00
|
|
|
hisMsgs,
|
2025-07-02 21:54:18 +08:00
|
|
|
}) => {
|
2025-09-03 00:37:35 +08:00
|
|
|
systemPrompt = genSystemPrompt({ systemPrompt, from, to });
|
|
|
|
|
userPrompt = genUserPrompt({ userPrompt, from, to, texts, docInfo });
|
2025-08-23 00:10:42 +08:00
|
|
|
customHeader = parseJsonObj(customHeader);
|
|
|
|
|
customBody = parseJsonObj(customBody);
|
2025-08-11 12:12:03 +08:00
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
const userMsg = {
|
|
|
|
|
role: "user",
|
|
|
|
|
content: userPrompt,
|
|
|
|
|
};
|
2025-07-02 21:54:18 +08:00
|
|
|
const data = {
|
|
|
|
|
model,
|
|
|
|
|
messages: [
|
|
|
|
|
{
|
|
|
|
|
role: "system",
|
|
|
|
|
content: systemPrompt,
|
|
|
|
|
},
|
2025-09-03 20:43:07 +08:00
|
|
|
...hisMsgs,
|
|
|
|
|
userMsg,
|
2025-07-02 21:54:18 +08:00
|
|
|
],
|
|
|
|
|
temperature,
|
2025-07-23 20:03:54 +08:00
|
|
|
max_tokens: maxTokens,
|
2025-08-11 16:03:30 +08:00
|
|
|
...customBody,
|
2025-07-02 21:54:18 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
Authorization: `Bearer ${key}`,
|
2025-08-11 16:28:23 +08:00
|
|
|
...customHeader,
|
2025-07-02 21:54:18 +08:00
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
return [url, init, userMsg];
|
2025-07-02 21:54:18 +08:00
|
|
|
};
|
|
|
|
|
|
2024-09-23 18:22:19 +08:00
|
|
|
const genClaude = ({
|
2025-09-03 00:37:35 +08:00
|
|
|
texts,
|
2024-09-23 18:22:19 +08:00
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
url,
|
|
|
|
|
key,
|
|
|
|
|
systemPrompt,
|
2024-11-30 00:41:29 +08:00
|
|
|
userPrompt,
|
2024-09-23 18:22:19 +08:00
|
|
|
model,
|
|
|
|
|
temperature,
|
|
|
|
|
maxTokens,
|
2025-08-11 16:28:23 +08:00
|
|
|
customHeader,
|
2025-08-11 16:03:30 +08:00
|
|
|
customBody,
|
2025-09-03 00:37:35 +08:00
|
|
|
docInfo,
|
2025-09-03 20:43:07 +08:00
|
|
|
hisMsgs,
|
2024-09-23 18:22:19 +08:00
|
|
|
}) => {
|
2025-09-03 00:37:35 +08:00
|
|
|
systemPrompt = genSystemPrompt({ systemPrompt, from, to });
|
|
|
|
|
userPrompt = genUserPrompt({ userPrompt, from, to, texts, docInfo });
|
2025-08-23 00:10:42 +08:00
|
|
|
customHeader = parseJsonObj(customHeader);
|
|
|
|
|
customBody = parseJsonObj(customBody);
|
2025-08-11 12:12:03 +08:00
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
const userMsg = {
|
|
|
|
|
role: "user",
|
|
|
|
|
content: userPrompt,
|
|
|
|
|
};
|
2024-09-23 18:22:19 +08:00
|
|
|
const data = {
|
|
|
|
|
model,
|
|
|
|
|
system: systemPrompt,
|
2025-09-03 20:43:07 +08:00
|
|
|
messages: [...hisMsgs, userMsg],
|
2024-09-23 18:22:19 +08:00
|
|
|
temperature,
|
|
|
|
|
max_tokens: maxTokens,
|
2025-08-11 16:03:30 +08:00
|
|
|
...customBody,
|
2024-09-23 18:22:19 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
"anthropic-version": "2023-06-01",
|
2025-08-22 08:03:10 +08:00
|
|
|
"anthropic-dangerous-direct-browser-access": "true",
|
2024-09-23 18:22:19 +08:00
|
|
|
"x-api-key": key,
|
2025-08-11 16:28:23 +08:00
|
|
|
...customHeader,
|
2024-09-23 18:22:19 +08:00
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
return [url, init, userMsg];
|
2024-09-23 18:22:19 +08:00
|
|
|
};
|
|
|
|
|
|
2025-08-23 11:01:54 +08:00
|
|
|
const genOpenRouter = ({
|
2025-09-03 00:37:35 +08:00
|
|
|
texts,
|
2025-08-23 11:01:54 +08:00
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
url,
|
|
|
|
|
key,
|
|
|
|
|
systemPrompt,
|
|
|
|
|
userPrompt,
|
|
|
|
|
model,
|
|
|
|
|
temperature,
|
|
|
|
|
maxTokens,
|
|
|
|
|
customHeader,
|
|
|
|
|
customBody,
|
2025-09-03 00:37:35 +08:00
|
|
|
docInfo,
|
2025-09-03 20:43:07 +08:00
|
|
|
hisMsgs,
|
2025-08-23 11:01:54 +08:00
|
|
|
}) => {
|
2025-09-03 00:37:35 +08:00
|
|
|
systemPrompt = genSystemPrompt({ systemPrompt, from, to });
|
|
|
|
|
userPrompt = genUserPrompt({ userPrompt, from, to, texts, docInfo });
|
2025-08-23 11:01:54 +08:00
|
|
|
customHeader = parseJsonObj(customHeader);
|
|
|
|
|
customBody = parseJsonObj(customBody);
|
|
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
const userMsg = {
|
|
|
|
|
role: "user",
|
|
|
|
|
content: userPrompt,
|
|
|
|
|
};
|
2025-08-23 11:01:54 +08:00
|
|
|
const data = {
|
|
|
|
|
model,
|
|
|
|
|
messages: [
|
|
|
|
|
{
|
|
|
|
|
role: "system",
|
|
|
|
|
content: systemPrompt,
|
|
|
|
|
},
|
2025-09-03 20:43:07 +08:00
|
|
|
...hisMsgs,
|
|
|
|
|
userMsg,
|
2025-08-23 11:01:54 +08:00
|
|
|
],
|
|
|
|
|
temperature,
|
|
|
|
|
max_tokens: maxTokens,
|
|
|
|
|
...customBody,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
Authorization: `Bearer ${key}`,
|
|
|
|
|
...customHeader,
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
return [url, init, userMsg];
|
2025-08-23 11:01:54 +08:00
|
|
|
};
|
|
|
|
|
|
2025-07-01 22:42:57 +08:00
|
|
|
const genOllama = ({
|
2025-09-03 00:37:35 +08:00
|
|
|
texts,
|
2025-07-01 22:42:57 +08:00
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
think,
|
|
|
|
|
url,
|
|
|
|
|
key,
|
|
|
|
|
systemPrompt,
|
|
|
|
|
userPrompt,
|
|
|
|
|
model,
|
2025-09-03 14:02:55 +08:00
|
|
|
temperature,
|
|
|
|
|
maxTokens,
|
2025-08-11 16:28:23 +08:00
|
|
|
customHeader,
|
2025-08-11 16:03:30 +08:00
|
|
|
customBody,
|
2025-09-03 00:37:35 +08:00
|
|
|
docInfo,
|
2025-09-03 20:43:07 +08:00
|
|
|
hisMsgs,
|
2025-07-01 22:42:57 +08:00
|
|
|
}) => {
|
2025-09-03 00:37:35 +08:00
|
|
|
systemPrompt = genSystemPrompt({ systemPrompt, from, to });
|
|
|
|
|
userPrompt = genUserPrompt({ userPrompt, from, to, texts, docInfo });
|
2025-08-23 00:10:42 +08:00
|
|
|
customHeader = parseJsonObj(customHeader);
|
|
|
|
|
customBody = parseJsonObj(customBody);
|
2025-08-11 12:12:03 +08:00
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
const userMsg = {
|
|
|
|
|
role: "user",
|
|
|
|
|
content: userPrompt,
|
|
|
|
|
};
|
2024-04-28 21:43:20 +08:00
|
|
|
const data = {
|
|
|
|
|
model,
|
2025-09-03 14:02:55 +08:00
|
|
|
messages: [
|
|
|
|
|
{
|
|
|
|
|
role: "system",
|
|
|
|
|
content: systemPrompt,
|
|
|
|
|
},
|
2025-09-03 20:43:07 +08:00
|
|
|
...hisMsgs,
|
|
|
|
|
userMsg,
|
2025-09-03 14:02:55 +08:00
|
|
|
],
|
|
|
|
|
temperature,
|
|
|
|
|
max_tokens: maxTokens,
|
|
|
|
|
think,
|
2024-04-28 21:43:20 +08:00
|
|
|
stream: false,
|
2025-08-11 16:03:30 +08:00
|
|
|
...customBody,
|
2024-04-28 21:43:20 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
2025-08-11 16:28:23 +08:00
|
|
|
...customHeader,
|
2024-04-28 21:43:20 +08:00
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
if (key) {
|
|
|
|
|
init.headers.Authorization = `Bearer ${key}`;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
return [url, init, userMsg];
|
2024-04-28 21:43:20 +08:00
|
|
|
};
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const genCloudflareAI = ({ texts, from, to, url, key }) => {
|
2023-10-26 11:13:50 +08:00
|
|
|
const data = {
|
2025-09-03 00:37:35 +08:00
|
|
|
text: texts.join(" "),
|
2023-10-26 11:13:50 +08:00
|
|
|
source_lang: from,
|
|
|
|
|
target_lang: to,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
Authorization: `Bearer ${key}`,
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [url, init];
|
|
|
|
|
};
|
|
|
|
|
|
2025-09-03 20:43:07 +08:00
|
|
|
const genCustom = ({
|
|
|
|
|
texts,
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
url,
|
|
|
|
|
key,
|
|
|
|
|
reqHook,
|
|
|
|
|
docInfo,
|
|
|
|
|
hisMsgs,
|
|
|
|
|
}) => {
|
2024-05-12 16:10:11 +08:00
|
|
|
if (reqHook?.trim()) {
|
|
|
|
|
interpreter.run(`exports.reqHook = ${reqHook}`);
|
2025-09-03 20:43:07 +08:00
|
|
|
return interpreter.exports.reqHook({
|
|
|
|
|
texts,
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
url,
|
|
|
|
|
key,
|
|
|
|
|
docInfo,
|
|
|
|
|
hisMsgs,
|
|
|
|
|
});
|
2024-05-12 16:10:11 +08:00
|
|
|
}
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
const data = { texts, from, to };
|
|
|
|
|
const init = {
|
2023-10-21 11:54:04 +08:00
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
2025-09-03 00:37:35 +08:00
|
|
|
Authorization: `Bearer ${key}`,
|
2023-10-21 11:54:04 +08:00
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [url, init];
|
|
|
|
|
};
|
|
|
|
|
|
2023-10-26 11:13:50 +08:00
|
|
|
/**
|
2024-04-21 13:16:44 +08:00
|
|
|
* 构造翻译接口请求参数
|
2023-10-26 11:13:50 +08:00
|
|
|
* @param {*}
|
|
|
|
|
* @returns
|
|
|
|
|
*/
|
2025-09-24 23:24:00 +08:00
|
|
|
export const genTransReq = ({ apiType, apiSlug, ...args }) => {
|
|
|
|
|
switch (apiType) {
|
2023-12-22 11:35:46 +08:00
|
|
|
case OPT_TRANS_DEEPL:
|
|
|
|
|
case OPT_TRANS_OPENAI:
|
|
|
|
|
case OPT_TRANS_GEMINI:
|
2025-07-02 21:54:18 +08:00
|
|
|
case OPT_TRANS_GEMINI_2:
|
2024-09-23 18:22:19 +08:00
|
|
|
case OPT_TRANS_CLAUDE:
|
2023-12-22 11:35:46 +08:00
|
|
|
case OPT_TRANS_CLOUDFLAREAI:
|
2024-04-28 21:43:20 +08:00
|
|
|
case OPT_TRANS_OLLAMA:
|
2025-08-23 11:01:54 +08:00
|
|
|
case OPT_TRANS_OPENROUTER:
|
2024-04-12 11:31:01 +08:00
|
|
|
case OPT_TRANS_NIUTRANS:
|
2025-07-01 22:42:57 +08:00
|
|
|
case OPT_TRANS_CUSTOMIZE:
|
2025-09-24 23:24:00 +08:00
|
|
|
args.key = keyPick(apiSlug, args.key, keyMap);
|
2024-04-20 14:01:34 +08:00
|
|
|
break;
|
|
|
|
|
case OPT_TRANS_DEEPLX:
|
2025-09-24 23:24:00 +08:00
|
|
|
args.url = keyPick(apiSlug, args.url, urlMap);
|
2023-12-22 11:35:46 +08:00
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-24 23:24:00 +08:00
|
|
|
switch (apiType) {
|
2023-10-21 11:54:04 +08:00
|
|
|
case OPT_TRANS_GOOGLE:
|
|
|
|
|
return genGoogle(args);
|
2025-07-02 13:38:30 +08:00
|
|
|
case OPT_TRANS_GOOGLE_2:
|
2025-07-01 22:42:57 +08:00
|
|
|
return genGoogle2(args);
|
2023-10-21 11:54:04 +08:00
|
|
|
case OPT_TRANS_MICROSOFT:
|
|
|
|
|
return genMicrosoft(args);
|
|
|
|
|
case OPT_TRANS_DEEPL:
|
|
|
|
|
return genDeepl(args);
|
|
|
|
|
case OPT_TRANS_DEEPLFREE:
|
|
|
|
|
return genDeeplFree(args);
|
|
|
|
|
case OPT_TRANS_DEEPLX:
|
|
|
|
|
return genDeeplX(args);
|
2024-04-12 11:31:01 +08:00
|
|
|
case OPT_TRANS_NIUTRANS:
|
|
|
|
|
return genNiuTrans(args);
|
2023-10-21 11:54:04 +08:00
|
|
|
case OPT_TRANS_BAIDU:
|
|
|
|
|
return genBaidu(args);
|
|
|
|
|
case OPT_TRANS_TENCENT:
|
|
|
|
|
return genTencent(args);
|
2025-06-30 21:34:37 +08:00
|
|
|
case OPT_TRANS_VOLCENGINE:
|
|
|
|
|
return genVolcengine(args);
|
2023-10-21 11:54:04 +08:00
|
|
|
case OPT_TRANS_OPENAI:
|
2023-10-26 11:13:50 +08:00
|
|
|
return genOpenAI(args);
|
2023-12-21 14:15:14 +08:00
|
|
|
case OPT_TRANS_GEMINI:
|
|
|
|
|
return genGemini(args);
|
2025-07-02 21:54:18 +08:00
|
|
|
case OPT_TRANS_GEMINI_2:
|
|
|
|
|
return genGemini2(args);
|
2024-09-23 18:22:19 +08:00
|
|
|
case OPT_TRANS_CLAUDE:
|
|
|
|
|
return genClaude(args);
|
2023-10-26 11:13:50 +08:00
|
|
|
case OPT_TRANS_CLOUDFLAREAI:
|
|
|
|
|
return genCloudflareAI(args);
|
2024-04-28 21:43:20 +08:00
|
|
|
case OPT_TRANS_OLLAMA:
|
|
|
|
|
return genOllama(args);
|
2025-08-23 11:01:54 +08:00
|
|
|
case OPT_TRANS_OPENROUTER:
|
|
|
|
|
return genOpenRouter(args);
|
2023-10-21 11:54:04 +08:00
|
|
|
case OPT_TRANS_CUSTOMIZE:
|
|
|
|
|
return genCustom(args);
|
|
|
|
|
default:
|
2025-09-24 23:24:00 +08:00
|
|
|
throw new Error(`[trans] ${apiType} not support`);
|
2023-10-21 11:54:04 +08:00
|
|
|
}
|
|
|
|
|
};
|
2025-08-31 23:37:29 +08:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 解析翻译接口返回数据
|
|
|
|
|
* @param {*} res
|
|
|
|
|
* @param {*} param3
|
|
|
|
|
* @returns
|
|
|
|
|
*/
|
|
|
|
|
export const parseTransRes = (
|
|
|
|
|
res,
|
2025-09-24 23:24:00 +08:00
|
|
|
{ texts, from, to, resHook, thinkIgnore, history, userMsg, apiType }
|
2025-08-31 23:37:29 +08:00
|
|
|
) => {
|
2025-09-03 20:43:07 +08:00
|
|
|
let modelMsg = "";
|
|
|
|
|
|
2025-09-24 23:24:00 +08:00
|
|
|
switch (apiType) {
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_GOOGLE:
|
2025-09-03 00:37:35 +08:00
|
|
|
return [[res?.sentences?.map((item) => item.trans).join(" "), res?.src]];
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_GOOGLE_2:
|
2025-09-03 00:37:35 +08:00
|
|
|
return res?.[0]?.map((_, i) => [res?.[0]?.[i], res?.[1]?.[i]]);
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_MICROSOFT:
|
2025-09-03 00:37:35 +08:00
|
|
|
return res?.map((item) => [
|
|
|
|
|
item.translations.map((item) => item.text).join(" "),
|
|
|
|
|
item.detectedLanguage.language,
|
|
|
|
|
]);
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_DEEPL:
|
2025-09-03 00:37:35 +08:00
|
|
|
return res?.translations?.map((item) => [
|
|
|
|
|
item.text,
|
|
|
|
|
item.detected_source_language,
|
|
|
|
|
]);
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_DEEPLFREE:
|
2025-09-03 00:37:35 +08:00
|
|
|
return [
|
|
|
|
|
[
|
|
|
|
|
res?.result?.texts?.map((item) => item.text).join(" "),
|
|
|
|
|
res?.result?.lang,
|
|
|
|
|
],
|
|
|
|
|
];
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_DEEPLX:
|
2025-09-03 00:37:35 +08:00
|
|
|
return [[res?.data, res?.source_lang]];
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_NIUTRANS:
|
|
|
|
|
const json = JSON.parse(res);
|
|
|
|
|
if (json.error_msg) {
|
|
|
|
|
throw new Error(json.error_msg);
|
|
|
|
|
}
|
2025-09-03 00:37:35 +08:00
|
|
|
return [[json.tgt_text, json.from]];
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_BAIDU:
|
|
|
|
|
if (res.type === 1) {
|
2025-09-03 00:37:35 +08:00
|
|
|
return [
|
|
|
|
|
[
|
|
|
|
|
Object.keys(JSON.parse(res.result).content[0].mean[0].cont)[0],
|
|
|
|
|
res.from,
|
|
|
|
|
],
|
|
|
|
|
];
|
2025-08-31 23:37:29 +08:00
|
|
|
} else if (res.type === 2) {
|
2025-09-03 00:37:35 +08:00
|
|
|
return [[res.data.map((item) => item.dst).join(" "), res.from]];
|
2025-08-31 23:37:29 +08:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case OPT_TRANS_TENCENT:
|
2025-09-03 00:37:35 +08:00
|
|
|
return res?.auto_translation?.map((text) => [text, res?.src_lang]);
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_VOLCENGINE:
|
2025-09-21 19:51:57 +08:00
|
|
|
return [[res?.translation, res?.detected_language]];
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_OPENAI:
|
|
|
|
|
case OPT_TRANS_GEMINI_2:
|
|
|
|
|
case OPT_TRANS_OPENROUTER:
|
2025-09-03 20:43:07 +08:00
|
|
|
modelMsg = res?.choices?.[0]?.message;
|
|
|
|
|
if (history && userMsg && modelMsg) {
|
|
|
|
|
history.add(userMsg, {
|
|
|
|
|
role: modelMsg.role,
|
|
|
|
|
content: modelMsg.content,
|
|
|
|
|
});
|
|
|
|
|
}
|
2025-09-03 09:40:25 +08:00
|
|
|
return parseAIRes(res?.choices?.[0]?.message?.content ?? "");
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_GEMINI:
|
2025-09-03 20:43:07 +08:00
|
|
|
modelMsg = res?.candidates?.[0]?.content;
|
|
|
|
|
if (history && userMsg && modelMsg) {
|
|
|
|
|
history.add(userMsg, modelMsg);
|
|
|
|
|
}
|
2025-09-03 09:40:25 +08:00
|
|
|
return parseAIRes(res?.candidates?.[0]?.content?.parts?.[0]?.text ?? "");
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_CLAUDE:
|
2025-09-03 20:43:07 +08:00
|
|
|
modelMsg = { role: res?.role, content: res?.content?.text };
|
|
|
|
|
if (history && userMsg && modelMsg) {
|
|
|
|
|
history.add(userMsg, {
|
|
|
|
|
role: modelMsg.role,
|
|
|
|
|
content: modelMsg.content,
|
|
|
|
|
});
|
|
|
|
|
}
|
2025-09-03 09:40:25 +08:00
|
|
|
return parseAIRes(res?.content?.[0]?.text ?? "");
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_CLOUDFLAREAI:
|
2025-09-03 00:37:35 +08:00
|
|
|
return [[res?.result?.translated_text]];
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_OLLAMA:
|
2025-09-03 20:43:07 +08:00
|
|
|
modelMsg = res?.choices?.[0]?.message;
|
|
|
|
|
|
2025-09-03 12:04:56 +08:00
|
|
|
const deepModels = thinkIgnore.split(",").filter((model) => model.trim());
|
|
|
|
|
if (deepModels.some((model) => res?.model?.startsWith(model))) {
|
2025-09-03 20:43:07 +08:00
|
|
|
modelMsg?.content.replace(/<think>[\s\S]*<\/think>/i, "");
|
2025-09-03 12:04:56 +08:00
|
|
|
}
|
2025-09-03 20:43:07 +08:00
|
|
|
|
|
|
|
|
if (history && userMsg && modelMsg) {
|
|
|
|
|
history.add(userMsg, {
|
|
|
|
|
role: modelMsg.role,
|
|
|
|
|
content: modelMsg.content,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
return parseAIRes(modelMsg?.content);
|
2025-08-31 23:37:29 +08:00
|
|
|
case OPT_TRANS_CUSTOMIZE:
|
|
|
|
|
if (resHook?.trim()) {
|
|
|
|
|
interpreter.run(`exports.resHook = ${resHook}`);
|
2025-09-03 20:43:07 +08:00
|
|
|
if (history) {
|
|
|
|
|
const [translations, modelMsg] = interpreter.exports.resHook({
|
|
|
|
|
res,
|
|
|
|
|
texts,
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
});
|
|
|
|
|
userMsg && modelMsg && history.add(userMsg, modelMsg);
|
|
|
|
|
return translations;
|
|
|
|
|
} else {
|
|
|
|
|
return interpreter.exports.resHook({ res, texts, from, to });
|
|
|
|
|
}
|
2025-08-31 23:37:29 +08:00
|
|
|
} else {
|
2025-09-03 00:37:35 +08:00
|
|
|
return res?.map((item) => [item.text, item.src]);
|
2025-08-31 23:37:29 +08:00
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-03 00:37:35 +08:00
|
|
|
return [];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 发送翻译请求并解析
|
|
|
|
|
* @param {*} param0
|
|
|
|
|
* @returns
|
|
|
|
|
*/
|
2025-09-03 13:05:41 +08:00
|
|
|
export const handleTranslate = async ({
|
2025-09-03 00:37:35 +08:00
|
|
|
texts,
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
docInfo,
|
|
|
|
|
apiSetting,
|
|
|
|
|
usePool,
|
|
|
|
|
}) => {
|
2025-09-03 20:43:07 +08:00
|
|
|
let history = null;
|
|
|
|
|
let hisMsgs = [];
|
2025-09-24 23:24:00 +08:00
|
|
|
const {
|
|
|
|
|
apiType,
|
|
|
|
|
apiSlug,
|
|
|
|
|
contextSize,
|
|
|
|
|
useContext,
|
|
|
|
|
fetchInterval,
|
|
|
|
|
fetchLimit,
|
|
|
|
|
httpTimeout,
|
|
|
|
|
} = apiSetting;
|
|
|
|
|
if (useContext && API_SPE_TYPES.context.has(apiType)) {
|
|
|
|
|
history = getMsgHistory(apiSlug, contextSize);
|
2025-09-03 20:43:07 +08:00
|
|
|
hisMsgs = history.getAll();
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-24 23:24:00 +08:00
|
|
|
const [input, init, userMsg] = await genTransReq({
|
2025-09-03 00:37:35 +08:00
|
|
|
texts,
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
docInfo,
|
2025-09-03 20:43:07 +08:00
|
|
|
hisMsgs,
|
2025-09-03 00:37:35 +08:00
|
|
|
...apiSetting,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const res = await fetchData(input, init, {
|
|
|
|
|
useCache: false,
|
|
|
|
|
usePool,
|
2025-09-24 23:24:00 +08:00
|
|
|
fetchInterval,
|
|
|
|
|
fetchLimit,
|
|
|
|
|
httpTimeout,
|
2025-09-03 00:37:35 +08:00
|
|
|
});
|
|
|
|
|
if (!res) {
|
|
|
|
|
throw new Error("tranlate got empty response");
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-24 23:24:00 +08:00
|
|
|
return parseTransRes(res, {
|
2025-09-03 00:37:35 +08:00
|
|
|
texts,
|
|
|
|
|
from,
|
|
|
|
|
to,
|
2025-09-03 20:43:07 +08:00
|
|
|
history,
|
|
|
|
|
userMsg,
|
2025-09-03 00:37:35 +08:00
|
|
|
...apiSetting,
|
|
|
|
|
});
|
2025-08-31 23:37:29 +08:00
|
|
|
};
|