2023-10-21 11:54:04 +08:00
|
|
|
import queryString from "query-string";
|
|
|
|
|
import {
|
|
|
|
|
OPT_TRANS_GOOGLE,
|
|
|
|
|
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,
|
|
|
|
|
OPT_TRANS_OPENAI,
|
2024-04-28 16:58:09 +08:00
|
|
|
OPT_TRANS_OPENAI_2,
|
|
|
|
|
OPT_TRANS_OPENAI_3,
|
2023-12-21 14:15:14 +08:00
|
|
|
OPT_TRANS_GEMINI,
|
2023-10-26 11:13:50 +08:00
|
|
|
OPT_TRANS_CLOUDFLAREAI,
|
2024-04-28 21:43:20 +08:00
|
|
|
OPT_TRANS_OLLAMA,
|
|
|
|
|
OPT_TRANS_OLLAMA_2,
|
|
|
|
|
OPT_TRANS_OLLAMA_3,
|
2023-10-21 11:54:04 +08:00
|
|
|
OPT_TRANS_CUSTOMIZE,
|
2024-04-10 13:37:16 +08:00
|
|
|
OPT_TRANS_CUSTOMIZE_2,
|
|
|
|
|
OPT_TRANS_CUSTOMIZE_3,
|
|
|
|
|
OPT_TRANS_CUSTOMIZE_4,
|
|
|
|
|
OPT_TRANS_CUSTOMIZE_5,
|
2023-10-21 11:54:04 +08:00
|
|
|
URL_MICROSOFT_TRAN,
|
|
|
|
|
URL_TENCENT_TRANSMART,
|
2024-04-17 17:38:54 +08:00
|
|
|
INPUT_PLACE_URL,
|
|
|
|
|
INPUT_PLACE_FROM,
|
|
|
|
|
INPUT_PLACE_TO,
|
|
|
|
|
INPUT_PLACE_TEXT,
|
|
|
|
|
INPUT_PLACE_KEY,
|
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";
|
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
|
|
|
|
|
const keyPick = (translator, 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 "";
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-20 14:01:34 +08:00
|
|
|
const preIndex = cacheMap.get(translator) ?? -1;
|
2024-01-04 09:40:03 +08:00
|
|
|
const curIndex = (preIndex + 1) % keys.length;
|
2024-04-20 14:01:34 +08:00
|
|
|
cacheMap.set(translator, 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
|
|
|
};
|
|
|
|
|
|
2023-10-21 11:54:04 +08:00
|
|
|
const genGoogle = ({ text, from, to, url, key }) => {
|
|
|
|
|
const params = {
|
|
|
|
|
client: "gtx",
|
|
|
|
|
dt: "t",
|
|
|
|
|
dj: 1,
|
|
|
|
|
ie: "UTF-8",
|
|
|
|
|
sl: from,
|
|
|
|
|
tl: to,
|
|
|
|
|
q: text,
|
|
|
|
|
};
|
|
|
|
|
const input = `${url}?${queryString.stringify(params)}`;
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
if (key) {
|
|
|
|
|
init.headers.Authorization = `Bearer ${key}`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [input, init];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const genMicrosoft = async ({ text, from, to }) => {
|
|
|
|
|
const [token] = await msAuth();
|
|
|
|
|
const params = {
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
"api-version": "3.0",
|
|
|
|
|
};
|
|
|
|
|
const input = `${URL_MICROSOFT_TRAN}?${queryString.stringify(params)}`;
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
Authorization: `Bearer ${token}`,
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify([{ Text: text }]),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [input, init];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const genDeepl = ({ text, from, to, url, key }) => {
|
|
|
|
|
const data = {
|
|
|
|
|
text: [text],
|
|
|
|
|
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];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const genDeeplX = ({ text, from, to, url, key }) => {
|
|
|
|
|
const data = {
|
|
|
|
|
text,
|
|
|
|
|
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];
|
|
|
|
|
};
|
|
|
|
|
|
2024-04-12 11:31:01 +08:00
|
|
|
const genNiuTrans = ({ text, from, to, url, key, dictNo, memoryNo }) => {
|
|
|
|
|
const data = {
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
apikey: key,
|
|
|
|
|
src_text: text,
|
|
|
|
|
dictNo,
|
|
|
|
|
memoryNo,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [url, init];
|
|
|
|
|
};
|
|
|
|
|
|
2023-10-21 11:54:04 +08:00
|
|
|
const genTencent = ({ text, from, to }) => {
|
|
|
|
|
const data = {
|
|
|
|
|
header: {
|
|
|
|
|
fn: "auto_translation_block",
|
|
|
|
|
},
|
|
|
|
|
source: {
|
|
|
|
|
text_block: text,
|
|
|
|
|
lang: from,
|
|
|
|
|
},
|
|
|
|
|
target: {
|
|
|
|
|
lang: to,
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-Type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [URL_TENCENT_TRANSMART, init];
|
|
|
|
|
};
|
|
|
|
|
|
2023-10-26 11:13:50 +08:00
|
|
|
const genOpenAI = ({ text, from, to, url, key, prompt, model }) => {
|
2023-10-21 11:54:04 +08:00
|
|
|
prompt = prompt
|
2024-04-17 17:38:54 +08:00
|
|
|
.replaceAll(INPUT_PLACE_FROM, from)
|
|
|
|
|
.replaceAll(INPUT_PLACE_TO, to);
|
2023-10-21 11:54:04 +08:00
|
|
|
|
|
|
|
|
const data = {
|
|
|
|
|
model,
|
|
|
|
|
messages: [
|
|
|
|
|
{
|
|
|
|
|
role: "system",
|
|
|
|
|
content: prompt,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
role: "user",
|
|
|
|
|
content: text,
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
temperature: 0,
|
|
|
|
|
max_tokens: 256,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
Authorization: `Bearer ${key}`, // OpenAI
|
|
|
|
|
"api-key": key, // Azure OpenAI
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [url, init];
|
|
|
|
|
};
|
|
|
|
|
|
2023-12-21 14:15:14 +08:00
|
|
|
const genGemini = ({ text, from, to, url, key, prompt, model }) => {
|
|
|
|
|
prompt = prompt
|
2024-04-17 17:38:54 +08:00
|
|
|
.replaceAll(INPUT_PLACE_FROM, from)
|
|
|
|
|
.replaceAll(INPUT_PLACE_TO, to)
|
2024-04-18 00:31:36 +08:00
|
|
|
.replaceAll(INPUT_PLACE_TEXT, text);
|
2023-12-21 14:15:14 +08:00
|
|
|
|
|
|
|
|
const data = {
|
|
|
|
|
contents: [
|
|
|
|
|
{
|
|
|
|
|
// role: "user",
|
|
|
|
|
parts: [
|
|
|
|
|
{
|
|
|
|
|
text: prompt,
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const input = `${url}/${model}:generateContent?key=${key}`;
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return [input, init];
|
|
|
|
|
};
|
|
|
|
|
|
2024-04-28 21:43:20 +08:00
|
|
|
const genOllama = ({ text, from, to, url, key, prompt, model }) => {
|
|
|
|
|
prompt = prompt
|
|
|
|
|
.replaceAll(INPUT_PLACE_FROM, from)
|
|
|
|
|
.replaceAll(INPUT_PLACE_TO, to)
|
|
|
|
|
.replaceAll(INPUT_PLACE_TEXT, text);
|
|
|
|
|
|
|
|
|
|
const data = {
|
|
|
|
|
model,
|
|
|
|
|
prompt,
|
|
|
|
|
stream: false,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
if (key) {
|
|
|
|
|
init.headers.Authorization = `Bearer ${key}`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [url, init];
|
|
|
|
|
};
|
|
|
|
|
|
2023-10-26 11:13:50 +08:00
|
|
|
const genCloudflareAI = ({ text, from, to, url, key }) => {
|
|
|
|
|
const data = {
|
|
|
|
|
text,
|
|
|
|
|
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];
|
|
|
|
|
};
|
|
|
|
|
|
2024-04-18 09:48:07 +08:00
|
|
|
const genCustom = ({ text, from, to, url, key, customOption }) => {
|
2024-04-17 17:38:54 +08:00
|
|
|
const replaceInput = (str) =>
|
|
|
|
|
str
|
|
|
|
|
.replaceAll(INPUT_PLACE_URL, url)
|
|
|
|
|
.replaceAll(INPUT_PLACE_FROM, from)
|
|
|
|
|
.replaceAll(INPUT_PLACE_TO, to)
|
2024-04-18 09:48:07 +08:00
|
|
|
.replaceAll(INPUT_PLACE_TEXT, text.replaceAll(`"`, `\n`))
|
2024-04-17 17:38:54 +08:00
|
|
|
.replaceAll(INPUT_PLACE_KEY, key);
|
2023-10-21 11:54:04 +08:00
|
|
|
const data = {
|
|
|
|
|
text,
|
|
|
|
|
from,
|
|
|
|
|
to,
|
|
|
|
|
};
|
|
|
|
|
const init = {
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
method: "POST",
|
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
|
};
|
|
|
|
|
if (key) {
|
|
|
|
|
init.headers.Authorization = `Bearer ${key}`;
|
|
|
|
|
}
|
2024-04-17 17:38:54 +08:00
|
|
|
url = replaceInput(url);
|
|
|
|
|
|
2024-04-18 09:48:07 +08:00
|
|
|
if (customOption?.trim()) {
|
2024-04-17 17:38:54 +08:00
|
|
|
try {
|
2024-04-17 22:35:12 +08:00
|
|
|
const opt = JSON.parse(replaceInput(customOption));
|
|
|
|
|
opt.url && (url = opt.url);
|
|
|
|
|
opt.headers && (init.headers = opt.headers);
|
|
|
|
|
opt.method && (init.method = opt.method);
|
2024-04-17 17:38:54 +08:00
|
|
|
if (init.method === "GET") {
|
|
|
|
|
delete init.body;
|
|
|
|
|
} else {
|
2024-04-17 22:35:12 +08:00
|
|
|
opt.body && (init.body = JSON.stringify(opt.body));
|
2024-04-17 17:38:54 +08:00
|
|
|
}
|
|
|
|
|
} catch (err) {
|
2024-04-17 22:35:12 +08:00
|
|
|
throw new Error(`custom option parse err: ${err}`);
|
2024-04-17 17:38:54 +08:00
|
|
|
}
|
|
|
|
|
}
|
2023-10-21 11:54:04 +08:00
|
|
|
|
|
|
|
|
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
|
|
|
|
|
*/
|
2024-04-21 13:16:44 +08:00
|
|
|
export const genTransReq = ({ translator, text, from, to }, apiSetting) => {
|
2023-10-21 11:54:04 +08:00
|
|
|
const args = { text, from, to, ...apiSetting };
|
2023-12-22 11:35:46 +08:00
|
|
|
|
|
|
|
|
switch (translator) {
|
|
|
|
|
case OPT_TRANS_DEEPL:
|
|
|
|
|
case OPT_TRANS_OPENAI:
|
2024-04-28 16:58:09 +08:00
|
|
|
case OPT_TRANS_OPENAI_2:
|
|
|
|
|
case OPT_TRANS_OPENAI_3:
|
2023-12-22 11:35:46 +08:00
|
|
|
case OPT_TRANS_GEMINI:
|
|
|
|
|
case OPT_TRANS_CLOUDFLAREAI:
|
2024-04-28 21:43:20 +08:00
|
|
|
case OPT_TRANS_OLLAMA:
|
|
|
|
|
case OPT_TRANS_OLLAMA_2:
|
|
|
|
|
case OPT_TRANS_OLLAMA_3:
|
2024-04-12 11:31:01 +08:00
|
|
|
case OPT_TRANS_NIUTRANS:
|
2024-04-20 14:01:34 +08:00
|
|
|
args.key = keyPick(translator, args.key, keyMap);
|
|
|
|
|
break;
|
|
|
|
|
case OPT_TRANS_DEEPLX:
|
|
|
|
|
args.url = keyPick(translator, args.url, urlMap);
|
2023-12-22 11:35:46 +08:00
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-21 11:54:04 +08:00
|
|
|
switch (translator) {
|
|
|
|
|
case OPT_TRANS_GOOGLE:
|
|
|
|
|
return genGoogle(args);
|
|
|
|
|
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);
|
|
|
|
|
case OPT_TRANS_OPENAI:
|
2024-04-28 16:58:09 +08:00
|
|
|
case OPT_TRANS_OPENAI_2:
|
|
|
|
|
case OPT_TRANS_OPENAI_3:
|
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);
|
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:
|
|
|
|
|
case OPT_TRANS_OLLAMA_2:
|
|
|
|
|
case OPT_TRANS_OLLAMA_3:
|
|
|
|
|
return genOllama(args);
|
2023-10-21 11:54:04 +08:00
|
|
|
case OPT_TRANS_CUSTOMIZE:
|
2024-04-10 13:37:16 +08:00
|
|
|
case OPT_TRANS_CUSTOMIZE_2:
|
|
|
|
|
case OPT_TRANS_CUSTOMIZE_3:
|
|
|
|
|
case OPT_TRANS_CUSTOMIZE_4:
|
|
|
|
|
case OPT_TRANS_CUSTOMIZE_5:
|
2023-10-21 11:54:04 +08:00
|
|
|
return genCustom(args);
|
|
|
|
|
default:
|
|
|
|
|
throw new Error(`[trans] translator: ${translator} not support`);
|
|
|
|
|
}
|
|
|
|
|
};
|