Files
kiss-translator/src/apis/trans.js

980 lines
20 KiB
JavaScript
Raw Normal View History

import queryString from "query-string";
import {
OPT_TRANS_GOOGLE,
2025-07-02 13:38:30 +08:00
OPT_TRANS_GOOGLE_2,
OPT_TRANS_MICROSOFT,
OPT_TRANS_DEEPL,
OPT_TRANS_DEEPLFREE,
OPT_TRANS_DEEPLX,
2024-04-12 11:31:01 +08:00
OPT_TRANS_NIUTRANS,
OPT_TRANS_BAIDU,
OPT_TRANS_TENCENT,
2025-06-30 21:34:37 +08:00
OPT_TRANS_VOLCENGINE,
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,
OPT_TRANS_OPENROUTER,
OPT_TRANS_CUSTOMIZE,
API_SPE_TYPES,
2024-04-17 17:38:54 +08:00
INPUT_PLACE_FROM,
INPUT_PLACE_TO,
// 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,
2025-10-01 23:22:14 +08:00
DEFAULT_USER_AGENT,
} from "../config";
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";
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-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
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 "";
}
const preIndex = cacheMap.get(apiSlug) ?? -1;
2024-01-04 09:40:03 +08:00
const curIndex = (preIndex + 1) % keys.length;
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
};
const genSystemPrompt = ({ systemPrompt, from, to }) =>
systemPrompt
.replaceAll(INPUT_PLACE_FROM, from)
.replaceAll(INPUT_PLACE_TO, to);
const genUserPrompt = ({
// userPrompt,
tone,
glossary = {},
// from,
to,
texts,
docInfo,
}) => {
const prompt = JSON.stringify({
targetLanguage: to,
title: docInfo.title,
description: docInfo.description,
segments: texts.map((text, i) => ({ id: i, text })),
glossary,
tone,
});
// if (userPrompt.includes(INPUT_PLACE_TEXT)) {
// return userPrompt
// .replaceAll(INPUT_PLACE_FROM, from)
// .replaceAll(INPUT_PLACE_TO, to)
// .replaceAll(INPUT_PLACE_TEXT, prompt);
// }
return prompt;
};
2025-09-03 09:40:25 +08:00
const parseAIRes = (raw) => {
if (!raw) {
return [];
}
try {
const jsonString = extractJson(raw);
const data = JSON.parse(jsonString);
if (Array.isArray(data.translations)) {
// todo: 考虑序号id可能会打乱
return data.translations.map((item) => [
item?.text ?? "",
item?.sourceLanguage ?? "",
]);
}
} catch (err) {
kissLog("parseAIRes", err);
}
return [];
};
const parseSTRes = (raw) => {
if (!raw) {
return [];
}
try {
const jsonString = extractJson(raw);
const data = JSON.parse(jsonString);
if (Array.isArray(data)) {
return data;
}
} catch (err) {
kissLog("parseAIRes: subtitle", err);
}
return [];
};
const genGoogle = ({ texts, from, to, url, key }) => {
2025-09-25 23:08:39 +08:00
const params = queryString.stringify({
2025-06-26 11:13:51 +08:00
client: "gtx",
dt: "t",
dj: 1,
ie: "UTF-8",
sl: from,
tl: to,
q: texts.join(" "),
2025-09-25 23:08:39 +08:00
});
url = `${url}?${params}`;
const headers = {
"Content-type": "application/json",
};
2025-06-26 11:13:51 +08:00
if (key) {
2025-09-25 23:08:39 +08:00
headers.Authorization = `Bearer ${key}`;
2025-06-26 11:13:51 +08:00
}
2025-09-25 23:08:39 +08:00
return { url, headers, method: "GET" };
};
const genGoogle2 = ({ texts, from, to, url, key }) => {
2025-10-05 17:47:29 +08:00
const body = [[texts, from, to], "wt_lib"];
2025-09-25 23:08:39 +08:00
const headers = {
"Content-Type": "application/json+protobuf",
"X-Goog-API-Key": key,
};
2025-10-05 17:47:29 +08:00
return { url, body, headers };
};
2025-09-25 23:08:39 +08:00
const genMicrosoft = ({ texts, from, to, token }) => {
const params = queryString.stringify({
from,
to,
"api-version": "3.0",
2025-09-25 23:08:39 +08:00
});
const url = `https://api-edge.cognitive.microsofttranslator.com/translate?${params}`;
const headers = {
"Content-type": "application/json",
Authorization: `Bearer ${token}`,
};
2025-10-05 17:47:29 +08:00
const body = texts.map((text) => ({ Text: text }));
2025-10-05 17:47:29 +08:00
return { url, body, headers };
};
const genDeepl = ({ texts, from, to, url, key }) => {
2025-10-05 17:47:29 +08:00
const body = {
text: texts,
target_lang: to,
source_lang: from,
// split_sentences: "0",
};
2025-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
Authorization: `DeepL-Auth-Key ${key}`,
};
2025-10-05 17:47:29 +08:00
return { url, body, headers };
};
const genDeeplX = ({ texts, from, to, url, key }) => {
2025-10-05 17:47:29 +08:00
const body = {
text: texts.join(" "),
target_lang: to,
source_lang: from,
};
2025-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
};
if (key) {
2025-09-25 23:08:39 +08:00
headers.Authorization = `Bearer ${key}`;
}
2025-10-05 17:47:29 +08:00
return { url, body, headers };
};
const genNiuTrans = ({ texts, from, to, url, key, dictNo, memoryNo }) => {
2025-10-05 17:47:29 +08:00
const body = {
2024-04-12 11:31:01 +08:00
from,
to,
apikey: key,
src_text: texts.join(" "),
2024-04-12 11:31:01 +08:00
dictNo,
memoryNo,
};
2025-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
2024-04-12 11:31:01 +08:00
};
2025-10-05 17:47:29 +08:00
return { url, body, headers };
2024-04-12 11:31:01 +08:00
};
const genTencent = ({ texts, from, to }) => {
2025-10-05 17:47:29 +08:00
const body = {
header: {
2025-06-27 20:03:58 +08:00
fn: "auto_translation",
client_key:
"browser-chrome-110.0.0-Mac OS-df4bd4c5-a65d-44b2-a40f-42f34f3535f2-1677486696487",
},
2025-06-27 20:03:58 +08:00
type: "plain",
model_category: "normal",
source: {
text_list: texts,
lang: from,
},
target: {
lang: to,
},
};
2025-09-25 23:08:39 +08:00
const url = "https://transmart.qq.com/api/imt";
const headers = {
"Content-Type": "application/json",
2025-10-01 23:22:14 +08:00
"user-agent": DEFAULT_USER_AGENT,
2025-09-25 23:08:39 +08:00
referer: "https://transmart.qq.com/zh-CN/index",
};
2025-10-05 17:47:29 +08:00
return { url, body, headers };
};
const genVolcengine = ({ texts, from, to }) => {
2025-10-05 17:47:29 +08:00
const body = {
2025-06-30 21:34:37 +08:00
source_language: from,
target_language: to,
text: texts.join(" "),
2025-06-30 21:34:37 +08:00
};
2025-09-25 23:08:39 +08:00
const url = "https://translate.volcengine.com/crx/translate/v1";
const headers = {
"Content-type": "application/json",
2025-06-30 21:34:37 +08:00
};
2025-10-05 17:47:29 +08:00
return { url, body, headers };
2025-06-30 21:34:37 +08:00
};
const genOpenAI = ({
url,
key,
2024-09-25 14:03:12 +08:00
systemPrompt,
2024-11-30 00:41:29 +08:00
userPrompt,
model,
temperature,
maxTokens,
hisMsgs = [],
}) => {
2025-09-03 20:43:07 +08:00
const userMsg = {
role: "user",
content: userPrompt,
};
2025-10-05 17:47:29 +08:00
const body = {
model,
messages: [
{
role: "system",
2024-09-25 14:03:12 +08:00
content: systemPrompt,
},
2025-09-03 20:43:07 +08:00
...hisMsgs,
userMsg,
],
temperature,
max_completion_tokens: maxTokens,
};
2025-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
Authorization: `Bearer ${key}`, // OpenAI
// "api-key": key, // Azure OpenAI
};
2025-10-05 17:47:29 +08:00
return { url, body, headers, userMsg };
};
const genGemini = ({
url,
key,
systemPrompt,
userPrompt,
model,
2025-07-02 16:37:57 +08:00
temperature,
maxTokens,
hisMsgs = [],
}) => {
2024-05-12 16:25:20 +08:00
url = url
.replaceAll(INPUT_PLACE_MODEL, model)
.replaceAll(INPUT_PLACE_KEY, key);
2025-08-11 12:12:03 +08:00
2025-09-03 20:43:07 +08:00
const userMsg = { role: "user", parts: [{ text: userPrompt }] };
2025-10-05 17:47:29 +08:00
const body = {
2025-10-07 21:10:51 +08:00
// system_instruction: {
// parts: {
// text: systemPrompt,
// },
// },
contents: [
{
role: "model",
parts: [{ text: systemPrompt }],
},
2025-10-07 21:10:51 +08:00
...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",
},
],
2023-12-21 14:15:14 +08:00
};
2025-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
2023-12-21 14:15:14 +08:00
};
2025-10-05 17:47:29 +08:00
return { url, body, headers, userMsg };
2023-12-21 14:15:14 +08:00
};
2025-07-02 21:54:18 +08:00
const genGemini2 = ({
url,
key,
systemPrompt,
userPrompt,
model,
temperature,
2025-07-23 20:03:54 +08:00
maxTokens,
hisMsgs = [],
2025-07-02 21:54:18 +08:00
}) => {
2025-09-03 20:43:07 +08:00
const userMsg = {
role: "user",
content: userPrompt,
};
2025-10-05 17:47:29 +08:00
const body = {
2025-07-02 21:54:18 +08:00
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-07-02 21:54:18 +08:00
};
2025-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
Authorization: `Bearer ${key}`,
2025-07-02 21:54:18 +08:00
};
2025-10-05 17:47:29 +08:00
return { url, body, headers, userMsg };
2025-07-02 21:54:18 +08:00
};
2024-09-23 18:22:19 +08:00
const genClaude = ({
url,
key,
systemPrompt,
2024-11-30 00:41:29 +08:00
userPrompt,
2024-09-23 18:22:19 +08:00
model,
temperature,
maxTokens,
hisMsgs = [],
2024-09-23 18:22:19 +08:00
}) => {
2025-09-03 20:43:07 +08:00
const userMsg = {
role: "user",
content: userPrompt,
};
2025-10-05 17:47:29 +08:00
const body = {
2024-09-23 18:22:19 +08:00
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-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
"anthropic-version": "2023-06-01",
"anthropic-dangerous-direct-browser-access": "true",
"x-api-key": key,
2024-09-23 18:22:19 +08:00
};
2025-10-05 17:47:29 +08:00
return { url, body, headers, userMsg };
2024-09-23 18:22:19 +08:00
};
const genOpenRouter = ({
url,
key,
systemPrompt,
userPrompt,
model,
temperature,
maxTokens,
hisMsgs = [],
}) => {
2025-09-03 20:43:07 +08:00
const userMsg = {
role: "user",
content: userPrompt,
};
2025-10-05 17:47:29 +08:00
const body = {
model,
messages: [
{
role: "system",
content: systemPrompt,
},
2025-09-03 20:43:07 +08:00
...hisMsgs,
userMsg,
],
temperature,
max_tokens: maxTokens,
};
2025-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
Authorization: `Bearer ${key}`,
};
2025-10-05 17:47:29 +08:00
return { url, body, headers, userMsg };
};
const genOllama = ({
think,
url,
key,
systemPrompt,
userPrompt,
model,
2025-09-03 14:02:55 +08:00
temperature,
maxTokens,
hisMsgs = [],
}) => {
2025-09-03 20:43:07 +08:00
const userMsg = {
role: "user",
content: userPrompt,
};
2025-10-05 17:47:29 +08:00
const body = {
2024-04-28 21:43:20 +08:00
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-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
2024-04-28 21:43:20 +08:00
};
if (key) {
2025-09-25 23:08:39 +08:00
headers.Authorization = `Bearer ${key}`;
2024-04-28 21:43:20 +08:00
}
2025-10-05 17:47:29 +08:00
return { url, body, headers, userMsg };
2024-04-28 21:43:20 +08:00
};
const genCloudflareAI = ({ texts, from, to, url, key }) => {
2025-10-05 17:47:29 +08:00
const body = {
text: texts.join(" "),
2023-10-26 11:13:50 +08:00
source_lang: from,
target_lang: to,
};
2025-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
Authorization: `Bearer ${key}`,
2023-10-26 11:13:50 +08:00
};
2025-10-05 17:47:29 +08:00
return { url, body, headers };
2023-10-26 11:13:50 +08:00
};
2025-09-25 23:08:39 +08:00
const genCustom = ({ texts, from, to, url, key }) => {
2025-10-05 17:47:29 +08:00
const body = { texts, from, to };
2025-09-25 23:08:39 +08:00
const headers = {
"Content-type": "application/json",
Authorization: `Bearer ${key}`,
};
2025-10-05 17:47:29 +08:00
return { url, body, headers };
2025-09-25 23:08:39 +08:00
};
const genReqFuncs = {
[OPT_TRANS_GOOGLE]: genGoogle,
[OPT_TRANS_GOOGLE_2]: genGoogle2,
[OPT_TRANS_MICROSOFT]: genMicrosoft,
[OPT_TRANS_DEEPL]: genDeepl,
[OPT_TRANS_DEEPLFREE]: genDeeplFree,
[OPT_TRANS_DEEPLX]: genDeeplX,
[OPT_TRANS_NIUTRANS]: genNiuTrans,
[OPT_TRANS_BAIDU]: genBaidu,
[OPT_TRANS_TENCENT]: genTencent,
[OPT_TRANS_VOLCENGINE]: genVolcengine,
[OPT_TRANS_OPENAI]: genOpenAI,
[OPT_TRANS_GEMINI]: genGemini,
[OPT_TRANS_GEMINI_2]: genGemini2,
[OPT_TRANS_CLAUDE]: genClaude,
[OPT_TRANS_CLOUDFLAREAI]: genCloudflareAI,
[OPT_TRANS_OLLAMA]: genOllama,
[OPT_TRANS_OPENROUTER]: genOpenRouter,
[OPT_TRANS_CUSTOMIZE]: genCustom,
};
const genInit = ({
url = "",
2025-10-05 17:47:29 +08:00
body = null,
2025-09-25 23:08:39 +08:00
headers = {},
userMsg = null,
method = "POST",
2025-09-03 20:43:07 +08:00
}) => {
2025-09-25 23:08:39 +08:00
if (!url) {
throw new Error("genInit: url is empty");
2024-05-12 16:10:11 +08:00
}
const init = {
2025-09-25 23:08:39 +08:00
method,
headers,
};
2025-10-05 17:47:29 +08:00
if (method !== "GET" && method !== "HEAD" && body) {
let payload = JSON.stringify(body);
const id = body?.params?.id;
2025-10-01 23:22:14 +08:00
if (id) {
2025-10-05 17:47:29 +08:00
payload = payload.replace(
2025-10-01 23:22:14 +08:00
'method":"',
(id + 3) % 13 === 0 || (id + 5) % 29 === 0
? 'method" : "'
: 'method": "'
);
}
2025-10-05 17:47:29 +08:00
Object.assign(init, { body: payload });
2025-09-25 23:08:39 +08:00
}
2025-09-25 23:08:39 +08:00
return [url, init, userMsg];
};
2023-10-26 11:13:50 +08:00
/**
* 构造翻译接口请求参数
2023-10-26 11:13:50 +08:00
* @param {*}
* @returns
*/
export const genTransReq = async ({ reqHook, ...args }) => {
2025-09-25 23:08:39 +08:00
const {
apiType,
apiSlug,
key,
systemPrompt,
userPrompt,
from,
to,
texts,
docInfo,
2025-10-01 16:18:19 +08:00
glossary,
2025-09-25 23:08:39 +08:00
customHeader,
customBody,
events,
2025-09-25 23:08:39 +08:00
} = args;
if (API_SPE_TYPES.mulkeys.has(apiType)) {
args.key = keyPick(apiSlug, key, keyMap);
2023-12-22 11:35:46 +08:00
}
2025-09-25 23:08:39 +08:00
if (apiType === OPT_TRANS_DEEPLX) {
args.url = keyPick(apiSlug, args.url, urlMap);
}
if (API_SPE_TYPES.ai.has(apiType)) {
args.systemPrompt = genSystemPrompt({ systemPrompt, from, to });
args.userPrompt = !!events
? JSON.stringify(events)
: genUserPrompt({
userPrompt,
from,
to,
texts,
docInfo,
glossary,
});
}
2025-09-25 23:08:39 +08:00
const {
url = "",
2025-10-05 17:47:29 +08:00
body = null,
2025-09-25 23:08:39 +08:00
headers = {},
userMsg = null,
method = "POST",
} = genReqFuncs[apiType](args);
// 合并用户自定义headers和body
if (customHeader?.trim()) {
Object.assign(headers, parseJsonObj(customHeader));
}
if (customBody?.trim()) {
2025-10-05 17:47:29 +08:00
Object.assign(body, parseJsonObj(customBody));
2025-09-25 23:08:39 +08:00
}
// 执行 request hook
if (reqHook?.trim() && !events) {
2025-09-25 23:08:39 +08:00
try {
interpreter.run(`exports.reqHook = ${reqHook}`);
const hookResult = await interpreter.exports.reqHook(args, {
url,
2025-10-05 17:47:29 +08:00
body,
2025-09-25 23:08:39 +08:00
headers,
userMsg,
method,
});
if (hookResult && hookResult.url) {
return genInit(hookResult);
}
} catch (err) {
kissLog("run req hook", err);
}
}
2025-10-05 17:47:29 +08:00
return genInit({ url, body, headers, userMsg, method });
};
/**
* 解析翻译接口返回数据
* @param {*} res
* @param {*} param3
* @returns
*/
2025-09-25 23:08:39 +08:00
export const parseTransRes = async (
res,
2025-09-25 23:08:39 +08:00
{
texts,
from,
to,
fromLang,
toLang,
langMap,
resHook,
thinkIgnore,
history,
userMsg,
apiType,
}
) => {
2025-09-25 23:08:39 +08:00
// 执行 response hook
if (resHook?.trim()) {
try {
interpreter.run(`exports.resHook = ${resHook}`);
const hookResult = await interpreter.exports.resHook({
apiType,
userMsg,
res,
texts,
from,
to,
fromLang,
toLang,
langMap,
});
if (hookResult && Array.isArray(hookResult.translations)) {
if (history && userMsg && hookResult.modelMsg) {
history.add(userMsg, hookResult.modelMsg);
}
return hookResult.translations;
}
} catch (err) {
kissLog("run res hook", err);
}
}
2025-09-03 20:43:07 +08:00
let modelMsg = "";
switch (apiType) {
case OPT_TRANS_GOOGLE:
return [[res?.sentences?.map((item) => item.trans).join(" "), res?.src]];
case OPT_TRANS_GOOGLE_2:
return res?.[0]?.map((_, i) => [res?.[0]?.[i], res?.[1]?.[i]]);
case OPT_TRANS_MICROSOFT:
return res?.map((item) => [
item.translations.map((item) => item.text).join(" "),
2025-09-27 21:21:56 +08:00
item.detectedLanguage?.language,
]);
case OPT_TRANS_DEEPL:
return res?.translations?.map((item) => [
item.text,
item.detected_source_language,
]);
case OPT_TRANS_DEEPLFREE:
return [
[
res?.result?.texts?.map((item) => item.text).join(" "),
res?.result?.lang,
],
];
case OPT_TRANS_DEEPLX:
return [[res?.data, res?.source_lang]];
case OPT_TRANS_NIUTRANS:
const json = JSON.parse(res);
if (json.error_msg) {
throw new Error(json.error_msg);
}
return [[json.tgt_text, json.from]];
case OPT_TRANS_BAIDU:
if (res.type === 1) {
return [
[
Object.keys(JSON.parse(res.result).content[0].mean[0].cont)[0],
res.from,
],
];
} else if (res.type === 2) {
return [[res.data.map((item) => item.dst).join(" "), res.from]];
}
break;
case OPT_TRANS_TENCENT:
return res?.auto_translation?.map((text) => [text, res?.src_lang]);
case OPT_TRANS_VOLCENGINE:
return [[res?.translation, res?.detected_language]];
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 ?? "");
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 ?? "");
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 ?? "");
case OPT_TRANS_CLOUDFLAREAI:
return [[res?.result?.translated_text]];
case OPT_TRANS_OLLAMA:
2025-09-03 20:43:07 +08:00
modelMsg = res?.choices?.[0]?.message;
2025-09-25 23:08:39 +08:00
const deepModels = thinkIgnore
.split(",")
.filter((model) => model?.trim());
2025-09-03 12:04:56 +08:00
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);
case OPT_TRANS_CUSTOMIZE:
2025-09-25 23:08:39 +08:00
return res?.map((item) => [item.text, item.src]);
default:
}
return [];
};
/**
* 发送翻译请求并解析
* @param {*} param0
* @returns
*/
2025-09-27 23:33:33 +08:00
export const handleTranslate = async (
texts = [],
2025-10-01 16:18:19 +08:00
{
from,
to,
fromLang,
toLang,
langMap,
docInfo,
glossary,
apiSetting,
usePool,
}
2025-09-27 23:33:33 +08:00
) => {
2025-09-03 20:43:07 +08:00
let history = null;
let hisMsgs = [];
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-25 23:08:39 +08:00
let token = "";
if (apiType === OPT_TRANS_MICROSOFT) {
2025-09-27 21:21:56 +08:00
token = await msAuth();
2025-09-25 23:08:39 +08:00
}
const [input, init, userMsg] = await genTransReq({
texts,
from,
to,
2025-09-25 23:08:39 +08:00
fromLang,
toLang,
langMap,
docInfo,
2025-10-01 16:18:19 +08:00
glossary,
2025-09-03 20:43:07 +08:00
hisMsgs,
2025-09-25 23:08:39 +08:00
token,
...apiSetting,
});
const res = await fetchData(input, init, {
useCache: false,
usePool,
fetchInterval,
fetchLimit,
httpTimeout,
});
if (!res) {
kissLog("tranlate got empty response");
return [];
}
return parseTransRes(res, {
texts,
from,
to,
2025-09-25 23:08:39 +08:00
fromLang,
toLang,
langMap,
2025-09-03 20:43:07 +08:00
history,
userMsg,
...apiSetting,
});
};
2025-09-27 23:33:33 +08:00
/**
* Microsoft语言识别聚合及解析
* @param {*} texts
* @returns
*/
export const handleMicrosoftLangdetect = async (texts = []) => {
const token = await msAuth();
const input =
"https://api-edge.cognitive.microsofttranslator.com/detect?api-version=3.0";
const init = {
headers: {
"Content-type": "application/json",
Authorization: `Bearer ${token}`,
},
method: "POST",
body: JSON.stringify(texts.map((text) => ({ Text: text }))),
};
const res = await fetchData(input, init, {
useCache: false,
});
if (Array.isArray(res)) {
return res.map((r) => r.language);
}
return [];
};
/**
* 字幕翻译
* @param {*} param0
* @returns
*/
2025-10-10 13:49:04 +08:00
export const handleSubtitle = async ({ events, from, to, apiSetting }) => {
const { apiType, fetchInterval, fetchLimit, httpTimeout } = apiSetting;
const [input, init] = await genTransReq({
...apiSetting,
events,
from,
to,
systemPrompt: apiSetting.subtitlePrompt,
});
const res = await fetchData(input, init, {
useCache: false,
usePool: true,
fetchInterval,
fetchLimit,
httpTimeout,
});
if (!res) {
kissLog("subtitle got empty response");
return [];
}
switch (apiType) {
case OPT_TRANS_OPENAI:
case OPT_TRANS_GEMINI_2:
case OPT_TRANS_OPENROUTER:
case OPT_TRANS_OLLAMA:
return parseSTRes(res?.choices?.[0]?.message?.content ?? "");
case OPT_TRANS_GEMINI:
return parseSTRes(res?.candidates?.[0]?.content?.parts?.[0]?.text ?? "");
case OPT_TRANS_CLAUDE:
return parseSTRes(res?.content?.[0]?.text ?? "");
case OPT_TRANS_CUSTOMIZE:
return res;
default:
}
return [];
};