2023-09-10 15:41:36 +02:00
|
|
|
const fetch = require('node-fetch').default;
|
|
|
|
const https = require('https');
|
|
|
|
const { readSecret, SECRET_KEYS } = require('./secrets');
|
|
|
|
const { generateRequestUrl, normaliseResponse } = require('google-translate-api-browser');
|
2023-10-06 13:39:03 +02:00
|
|
|
const bingTranslateApi = require('bing-translate-api')
|
2023-09-10 15:41:36 +02:00
|
|
|
|
2023-09-10 17:22:39 +02:00
|
|
|
const DEEPLX_URL_DEFAULT = 'http://127.0.0.1:1188/translate';
|
|
|
|
const ONERING_URL_DEFAULT = 'http://127.0.0.1:4990/translate';
|
|
|
|
|
2023-09-10 15:41:36 +02:00
|
|
|
/**
|
|
|
|
* @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) {
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
app.post('/api/translate/google', jsonParser, async (request, response) => {
|
|
|
|
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', () => {
|
|
|
|
const result = normaliseResponse(JSON.parse(data));
|
|
|
|
console.log('Translated text: ' + result.text);
|
|
|
|
return response.send(result.text);
|
|
|
|
});
|
|
|
|
}).on("error", (err) => {
|
|
|
|
console.log("Translation error: " + err.message);
|
|
|
|
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;
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
});
|
2023-09-10 16:27:50 +02:00
|
|
|
|
|
|
|
app.post('/api/translate/onering', jsonParser, async (request, response) => {
|
2023-09-10 17:22:39 +02:00
|
|
|
const secretUrl = readSecret(SECRET_KEYS.ONERING_URL);
|
|
|
|
const url = secretUrl || ONERING_URL_DEFAULT;
|
2023-09-10 16:27:50 +02:00
|
|
|
|
|
|
|
if (!url) {
|
|
|
|
console.log('OneRing URL is not configured.');
|
|
|
|
return response.sendStatus(401);
|
|
|
|
}
|
|
|
|
|
2023-09-10 17:22:39 +02:00
|
|
|
if (!secretUrl && url === ONERING_URL_DEFAULT) {
|
|
|
|
console.log('OneRing URL is using default value.', ONERING_URL_DEFAULT);
|
|
|
|
}
|
|
|
|
|
2023-09-10 16:27:50 +02:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
});
|
2023-09-10 17:22:39 +02:00
|
|
|
|
|
|
|
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;
|
|
|
|
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({
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
});
|
2023-10-06 13:39:03 +02:00
|
|
|
|
|
|
|
app.post('/api/translate/bing', jsonParser, async (request, response) => {
|
|
|
|
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);
|
|
|
|
});
|
|
|
|
});
|
2023-09-10 15:41:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
registerEndpoints,
|
|
|
|
};
|