mirror of
https://github.com/SillyTavern/SillyTavern.git
synced 2025-06-05 21:59:27 +02:00
Use Express router for translate endpoint
This commit is contained in:
@@ -3615,7 +3615,7 @@ require('./src/endpoints/horde').registerEndpoints(app, jsonParser);
|
|||||||
require('./src/endpoints/vectors').registerEndpoints(app, jsonParser);
|
require('./src/endpoints/vectors').registerEndpoints(app, jsonParser);
|
||||||
|
|
||||||
// Chat translation
|
// Chat translation
|
||||||
require('./src/endpoints/translate').registerEndpoints(app, jsonParser);
|
app.use('/api/translate', require('./src/endpoints/translate').router);
|
||||||
|
|
||||||
// Emotion classification
|
// Emotion classification
|
||||||
require('./src/endpoints/classify').registerEndpoints(app, jsonParser);
|
require('./src/endpoints/classify').registerEndpoints(app, jsonParser);
|
||||||
|
@@ -1,25 +1,65 @@
|
|||||||
const fetch = require('node-fetch').default;
|
const fetch = require('node-fetch').default;
|
||||||
const https = require('https');
|
const https = require('https');
|
||||||
|
const express = require('express');
|
||||||
const { readSecret, SECRET_KEYS } = require('./secrets');
|
const { readSecret, SECRET_KEYS } = require('./secrets');
|
||||||
const { getConfigValue } = require('../util');
|
const { getConfigValue } = require('../util');
|
||||||
|
const { jsonParser } = require('../express-common');
|
||||||
|
|
||||||
const DEEPLX_URL_DEFAULT = 'http://127.0.0.1:1188/translate';
|
const DEEPLX_URL_DEFAULT = 'http://127.0.0.1:1188/translate';
|
||||||
const ONERING_URL_DEFAULT = 'http://127.0.0.1:4990/translate';
|
const ONERING_URL_DEFAULT = 'http://127.0.0.1:4990/translate';
|
||||||
|
|
||||||
/**
|
const router = express.Router();
|
||||||
* @param {import("express").Express} app
|
|
||||||
* @param {any} jsonParser
|
|
||||||
*/
|
|
||||||
function registerEndpoints(app, jsonParser) {
|
|
||||||
app.post('/api/translate/libre', jsonParser, async (request, response) => {
|
|
||||||
const key = readSecret(SECRET_KEYS.LIBRE);
|
|
||||||
const url = readSecret(SECRET_KEYS.LIBRE_URL);
|
|
||||||
|
|
||||||
if (!url) {
|
router.post('/libre', jsonParser, async (request, response) => {
|
||||||
console.log('LibreTranslate URL is not configured.');
|
const key = readSecret(SECRET_KEYS.LIBRE);
|
||||||
return response.sendStatus(401);
|
const url = readSecret(SECRET_KEYS.LIBRE_URL);
|
||||||
|
|
||||||
|
if (!url) {
|
||||||
|
console.log('LibreTranslate URL is not configured.');
|
||||||
|
return response.sendStatus(401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const text = request.body.text;
|
||||||
|
const lang = request.body.lang;
|
||||||
|
|
||||||
|
if (!text || !lang) {
|
||||||
|
return response.sendStatus(400);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('Input text: ' + text);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await fetch(url, {
|
||||||
|
method: 'POST',
|
||||||
|
body: JSON.stringify({
|
||||||
|
q: text,
|
||||||
|
source: 'auto',
|
||||||
|
target: lang,
|
||||||
|
format: 'text',
|
||||||
|
api_key: key,
|
||||||
|
}),
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!result.ok) {
|
||||||
|
const error = await result.text();
|
||||||
|
console.log('LibreTranslate error: ', result.statusText, error);
|
||||||
|
return response.sendStatus(result.status);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const json = await result.json();
|
||||||
|
console.log('Translated text: ' + json.translatedText);
|
||||||
|
|
||||||
|
return response.send(json.translatedText);
|
||||||
|
} catch (error) {
|
||||||
|
console.log('Translation error: ' + error.message);
|
||||||
|
return response.sendStatus(500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post('/google', jsonParser, async (request, response) => {
|
||||||
|
try {
|
||||||
|
const { generateRequestUrl, normaliseResponse } = require('google-translate-api-browser');
|
||||||
const text = request.body.text;
|
const text = request.body.text;
|
||||||
const lang = request.body.lang;
|
const lang = request.body.lang;
|
||||||
|
|
||||||
@@ -29,264 +69,220 @@ function registerEndpoints(app, jsonParser) {
|
|||||||
|
|
||||||
console.log('Input text: ' + text);
|
console.log('Input text: ' + text);
|
||||||
|
|
||||||
try {
|
const url = generateRequestUrl(text, { to: lang });
|
||||||
const result = await fetch(url, {
|
|
||||||
method: 'POST',
|
https.get(url, (resp) => {
|
||||||
body: JSON.stringify({
|
let data = '';
|
||||||
q: text,
|
|
||||||
source: 'auto',
|
resp.on('data', (chunk) => {
|
||||||
target: lang,
|
data += chunk;
|
||||||
format: 'text',
|
|
||||||
api_key: key,
|
|
||||||
}),
|
|
||||||
headers: { 'Content-Type': 'application/json' },
|
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!result.ok) {
|
resp.on('end', () => {
|
||||||
const error = await result.text();
|
try {
|
||||||
console.log('LibreTranslate error: ', result.statusText, error);
|
const result = normaliseResponse(JSON.parse(data));
|
||||||
return response.sendStatus(result.status);
|
console.log('Translated text: ' + result.text);
|
||||||
}
|
return response.send(result.text);
|
||||||
|
} catch (error) {
|
||||||
const json = await result.json();
|
console.log('Translation error', error);
|
||||||
console.log('Translated text: ' + json.translatedText);
|
return response.sendStatus(500);
|
||||||
|
}
|
||||||
return response.send(json.translatedText);
|
|
||||||
} catch (error) {
|
|
||||||
console.log('Translation error: ' + error.message);
|
|
||||||
return response.sendStatus(500);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
app.post('/api/translate/google', jsonParser, async (request, response) => {
|
|
||||||
try {
|
|
||||||
const { generateRequestUrl, normaliseResponse } = require('google-translate-api-browser');
|
|
||||||
const text = request.body.text;
|
|
||||||
const lang = request.body.lang;
|
|
||||||
|
|
||||||
if (!text || !lang) {
|
|
||||||
return response.sendStatus(400);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('Input text: ' + text);
|
|
||||||
|
|
||||||
const url = generateRequestUrl(text, { to: lang });
|
|
||||||
|
|
||||||
https.get(url, (resp) => {
|
|
||||||
let data = '';
|
|
||||||
|
|
||||||
resp.on('data', (chunk) => {
|
|
||||||
data += chunk;
|
|
||||||
});
|
|
||||||
|
|
||||||
resp.on('end', () => {
|
|
||||||
try {
|
|
||||||
const result = normaliseResponse(JSON.parse(data));
|
|
||||||
console.log('Translated text: ' + result.text);
|
|
||||||
return response.send(result.text);
|
|
||||||
} catch (error) {
|
|
||||||
console.log('Translation error', error);
|
|
||||||
return response.sendStatus(500);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}).on('error', (err) => {
|
|
||||||
console.log('Translation error: ' + err.message);
|
|
||||||
return response.sendStatus(500);
|
|
||||||
});
|
});
|
||||||
} catch (error) {
|
}).on('error', (err) => {
|
||||||
console.log('Translation error', error);
|
|
||||||
return response.sendStatus(500);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
app.post('/api/translate/deepl', jsonParser, async (request, response) => {
|
|
||||||
const key = readSecret(SECRET_KEYS.DEEPL);
|
|
||||||
|
|
||||||
if (!key) {
|
|
||||||
return response.sendStatus(401);
|
|
||||||
}
|
|
||||||
|
|
||||||
const text = request.body.text;
|
|
||||||
const lang = request.body.lang;
|
|
||||||
const formality = getConfigValue('deepl.formality', 'default');
|
|
||||||
|
|
||||||
if (!text || !lang) {
|
|
||||||
return response.sendStatus(400);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('Input text: ' + text);
|
|
||||||
|
|
||||||
const params = new URLSearchParams();
|
|
||||||
params.append('text', text);
|
|
||||||
params.append('target_lang', lang);
|
|
||||||
|
|
||||||
if (['de', 'fr', 'it', 'es', 'nl', 'ja', 'ru'].includes(lang)) {
|
|
||||||
// We don't specify a Portuguese variant, so ignore formality for it.
|
|
||||||
params.append('formality', formality);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const result = await fetch('https://api-free.deepl.com/v2/translate', {
|
|
||||||
method: 'POST',
|
|
||||||
body: params,
|
|
||||||
headers: {
|
|
||||||
'Accept': 'application/json',
|
|
||||||
'Authorization': `DeepL-Auth-Key ${key}`,
|
|
||||||
'Content-Type': 'application/x-www-form-urlencoded',
|
|
||||||
},
|
|
||||||
timeout: 0,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!result.ok) {
|
|
||||||
const error = await result.text();
|
|
||||||
console.log('DeepL error: ', result.statusText, error);
|
|
||||||
return response.sendStatus(result.status);
|
|
||||||
}
|
|
||||||
|
|
||||||
const json = await result.json();
|
|
||||||
console.log('Translated text: ' + json.translations[0].text);
|
|
||||||
|
|
||||||
return response.send(json.translations[0].text);
|
|
||||||
} catch (error) {
|
|
||||||
console.log('Translation error: ' + error.message);
|
|
||||||
return response.sendStatus(500);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
app.post('/api/translate/onering', jsonParser, async (request, response) => {
|
|
||||||
const secretUrl = readSecret(SECRET_KEYS.ONERING_URL);
|
|
||||||
const url = secretUrl || ONERING_URL_DEFAULT;
|
|
||||||
|
|
||||||
if (!url) {
|
|
||||||
console.log('OneRing URL is not configured.');
|
|
||||||
return response.sendStatus(401);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!secretUrl && url === ONERING_URL_DEFAULT) {
|
|
||||||
console.log('OneRing URL is using default value.', ONERING_URL_DEFAULT);
|
|
||||||
}
|
|
||||||
|
|
||||||
const text = request.body.text;
|
|
||||||
const from_lang = request.body.from_lang;
|
|
||||||
const to_lang = request.body.to_lang;
|
|
||||||
|
|
||||||
if (!text || !from_lang || !to_lang) {
|
|
||||||
return response.sendStatus(400);
|
|
||||||
}
|
|
||||||
|
|
||||||
const params = new URLSearchParams();
|
|
||||||
params.append('text', text);
|
|
||||||
params.append('from_lang', from_lang);
|
|
||||||
params.append('to_lang', to_lang);
|
|
||||||
|
|
||||||
console.log('Input text: ' + text);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const fetchUrl = new URL(url);
|
|
||||||
fetchUrl.search = params.toString();
|
|
||||||
|
|
||||||
const result = await fetch(fetchUrl, {
|
|
||||||
method: 'GET',
|
|
||||||
timeout: 0,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!result.ok) {
|
|
||||||
const error = await result.text();
|
|
||||||
console.log('OneRing error: ', result.statusText, error);
|
|
||||||
return response.sendStatus(result.status);
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = await result.json();
|
|
||||||
console.log('Translated text: ' + data.result);
|
|
||||||
|
|
||||||
return response.send(data.result);
|
|
||||||
} catch (error) {
|
|
||||||
console.log('Translation error: ' + error.message);
|
|
||||||
return response.sendStatus(500);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
app.post('/api/translate/deeplx', jsonParser, async (request, response) => {
|
|
||||||
const secretUrl = readSecret(SECRET_KEYS.DEEPLX_URL);
|
|
||||||
const url = secretUrl || DEEPLX_URL_DEFAULT;
|
|
||||||
|
|
||||||
if (!url) {
|
|
||||||
console.log('DeepLX URL is not configured.');
|
|
||||||
return response.sendStatus(401);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!secretUrl && url === DEEPLX_URL_DEFAULT) {
|
|
||||||
console.log('DeepLX URL is using default value.', DEEPLX_URL_DEFAULT);
|
|
||||||
}
|
|
||||||
|
|
||||||
const text = request.body.text;
|
|
||||||
let lang = request.body.lang;
|
|
||||||
if (request.body.lang === 'zh-CN') {
|
|
||||||
lang = 'ZH';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!text || !lang) {
|
|
||||||
return response.sendStatus(400);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('Input text: ' + text);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const result = await fetch(url, {
|
|
||||||
method: 'POST',
|
|
||||||
body: JSON.stringify({
|
|
||||||
text: text,
|
|
||||||
source_lang: 'auto',
|
|
||||||
target_lang: lang,
|
|
||||||
}),
|
|
||||||
headers: {
|
|
||||||
'Accept': 'application/json',
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
timeout: 0,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!result.ok) {
|
|
||||||
const error = await result.text();
|
|
||||||
console.log('DeepLX error: ', result.statusText, error);
|
|
||||||
return response.sendStatus(result.status);
|
|
||||||
}
|
|
||||||
|
|
||||||
const json = await result.json();
|
|
||||||
console.log('Translated text: ' + json.data);
|
|
||||||
|
|
||||||
return response.send(json.data);
|
|
||||||
} catch (error) {
|
|
||||||
console.log('DeepLX translation error: ' + error.message);
|
|
||||||
return response.sendStatus(500);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
app.post('/api/translate/bing', jsonParser, async (request, response) => {
|
|
||||||
const bingTranslateApi = require('bing-translate-api');
|
|
||||||
const text = request.body.text;
|
|
||||||
let lang = request.body.lang;
|
|
||||||
|
|
||||||
if (request.body.lang === 'zh-CN') {
|
|
||||||
lang = 'zh-Hans';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!text || !lang) {
|
|
||||||
return response.sendStatus(400);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('Input text: ' + text);
|
|
||||||
|
|
||||||
bingTranslateApi.translate(text, null, lang).then(result => {
|
|
||||||
console.log('Translated text: ' + result.translation);
|
|
||||||
return response.send(result.translation);
|
|
||||||
}).catch(err => {
|
|
||||||
console.log('Translation error: ' + err.message);
|
console.log('Translation error: ' + err.message);
|
||||||
return response.sendStatus(500);
|
return response.sendStatus(500);
|
||||||
});
|
});
|
||||||
});
|
} catch (error) {
|
||||||
}
|
console.log('Translation error', error);
|
||||||
|
return response.sendStatus(500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
module.exports = {
|
router.post('/deepl', jsonParser, async (request, response) => {
|
||||||
registerEndpoints,
|
const key = readSecret(SECRET_KEYS.DEEPL);
|
||||||
};
|
|
||||||
|
if (!key) {
|
||||||
|
return response.sendStatus(401);
|
||||||
|
}
|
||||||
|
|
||||||
|
const text = request.body.text;
|
||||||
|
const lang = request.body.lang;
|
||||||
|
const formality = getConfigValue('deepl.formality', 'default');
|
||||||
|
|
||||||
|
if (!text || !lang) {
|
||||||
|
return response.sendStatus(400);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('Input text: ' + text);
|
||||||
|
|
||||||
|
const params = new URLSearchParams();
|
||||||
|
params.append('text', text);
|
||||||
|
params.append('target_lang', lang);
|
||||||
|
|
||||||
|
if (['de', 'fr', 'it', 'es', 'nl', 'ja', 'ru'].includes(lang)) {
|
||||||
|
// We don't specify a Portuguese variant, so ignore formality for it.
|
||||||
|
params.append('formality', formality);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await fetch('https://api-free.deepl.com/v2/translate', {
|
||||||
|
method: 'POST',
|
||||||
|
body: params,
|
||||||
|
headers: {
|
||||||
|
'Accept': 'application/json',
|
||||||
|
'Authorization': `DeepL-Auth-Key ${key}`,
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
},
|
||||||
|
timeout: 0,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!result.ok) {
|
||||||
|
const error = await result.text();
|
||||||
|
console.log('DeepL error: ', result.statusText, error);
|
||||||
|
return response.sendStatus(result.status);
|
||||||
|
}
|
||||||
|
|
||||||
|
const json = await result.json();
|
||||||
|
console.log('Translated text: ' + json.translations[0].text);
|
||||||
|
|
||||||
|
return response.send(json.translations[0].text);
|
||||||
|
} catch (error) {
|
||||||
|
console.log('Translation error: ' + error.message);
|
||||||
|
return response.sendStatus(500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post('/onering', jsonParser, async (request, response) => {
|
||||||
|
const secretUrl = readSecret(SECRET_KEYS.ONERING_URL);
|
||||||
|
const url = secretUrl || ONERING_URL_DEFAULT;
|
||||||
|
|
||||||
|
if (!url) {
|
||||||
|
console.log('OneRing URL is not configured.');
|
||||||
|
return response.sendStatus(401);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!secretUrl && url === ONERING_URL_DEFAULT) {
|
||||||
|
console.log('OneRing URL is using default value.', ONERING_URL_DEFAULT);
|
||||||
|
}
|
||||||
|
|
||||||
|
const text = request.body.text;
|
||||||
|
const from_lang = request.body.from_lang;
|
||||||
|
const to_lang = request.body.to_lang;
|
||||||
|
|
||||||
|
if (!text || !from_lang || !to_lang) {
|
||||||
|
return response.sendStatus(400);
|
||||||
|
}
|
||||||
|
|
||||||
|
const params = new URLSearchParams();
|
||||||
|
params.append('text', text);
|
||||||
|
params.append('from_lang', from_lang);
|
||||||
|
params.append('to_lang', to_lang);
|
||||||
|
|
||||||
|
console.log('Input text: ' + text);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const fetchUrl = new URL(url);
|
||||||
|
fetchUrl.search = params.toString();
|
||||||
|
|
||||||
|
const result = await fetch(fetchUrl, {
|
||||||
|
method: 'GET',
|
||||||
|
timeout: 0,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!result.ok) {
|
||||||
|
const error = await result.text();
|
||||||
|
console.log('OneRing error: ', result.statusText, error);
|
||||||
|
return response.sendStatus(result.status);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await result.json();
|
||||||
|
console.log('Translated text: ' + data.result);
|
||||||
|
|
||||||
|
return response.send(data.result);
|
||||||
|
} catch (error) {
|
||||||
|
console.log('Translation error: ' + error.message);
|
||||||
|
return response.sendStatus(500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post('/deeplx', jsonParser, async (request, response) => {
|
||||||
|
const secretUrl = readSecret(SECRET_KEYS.DEEPLX_URL);
|
||||||
|
const url = secretUrl || DEEPLX_URL_DEFAULT;
|
||||||
|
|
||||||
|
if (!url) {
|
||||||
|
console.log('DeepLX URL is not configured.');
|
||||||
|
return response.sendStatus(401);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!secretUrl && url === DEEPLX_URL_DEFAULT) {
|
||||||
|
console.log('DeepLX URL is using default value.', DEEPLX_URL_DEFAULT);
|
||||||
|
}
|
||||||
|
|
||||||
|
const text = request.body.text;
|
||||||
|
let lang = request.body.lang;
|
||||||
|
if (request.body.lang === 'zh-CN') {
|
||||||
|
lang = 'ZH';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!text || !lang) {
|
||||||
|
return response.sendStatus(400);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('Input text: ' + text);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await fetch(url, {
|
||||||
|
method: 'POST',
|
||||||
|
body: JSON.stringify({
|
||||||
|
text: text,
|
||||||
|
source_lang: 'auto',
|
||||||
|
target_lang: lang,
|
||||||
|
}),
|
||||||
|
headers: {
|
||||||
|
'Accept': 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
timeout: 0,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!result.ok) {
|
||||||
|
const error = await result.text();
|
||||||
|
console.log('DeepLX error: ', result.statusText, error);
|
||||||
|
return response.sendStatus(result.status);
|
||||||
|
}
|
||||||
|
|
||||||
|
const json = await result.json();
|
||||||
|
console.log('Translated text: ' + json.data);
|
||||||
|
|
||||||
|
return response.send(json.data);
|
||||||
|
} catch (error) {
|
||||||
|
console.log('DeepLX translation error: ' + error.message);
|
||||||
|
return response.sendStatus(500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post('/bing', jsonParser, async (request, response) => {
|
||||||
|
const bingTranslateApi = require('bing-translate-api');
|
||||||
|
const text = request.body.text;
|
||||||
|
let lang = request.body.lang;
|
||||||
|
|
||||||
|
if (request.body.lang === 'zh-CN') {
|
||||||
|
lang = 'zh-Hans';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!text || !lang) {
|
||||||
|
return response.sendStatus(400);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('Input text: ' + text);
|
||||||
|
|
||||||
|
bingTranslateApi.translate(text, null, lang).then(result => {
|
||||||
|
console.log('Translated text: ' + result.translation);
|
||||||
|
return response.send(result.translation);
|
||||||
|
}).catch(err => {
|
||||||
|
console.log('Translation error: ' + err.message);
|
||||||
|
return response.sendStatus(500);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = { router };
|
||||||
|
Reference in New Issue
Block a user