| import fetch from 'node-fetch'; |
| import express from 'express'; |
| import { translate as bingTranslate } from 'bing-translate-api'; |
| import urlJoin from 'url-join'; |
| import { Translator } from 'google-translate-api-x'; |
|
|
| import { readSecret, SECRET_KEYS } from './secrets.js'; |
| import { getConfigValue, uuidv4 } from '../util.js'; |
|
|
| const DEEPLX_URL_DEFAULT = 'http://127.0.0.1:1188/translate'; |
| const ONERING_URL_DEFAULT = 'http://127.0.0.1:4990/translate'; |
| const LINGVA_DEFAULT = 'https://lingva.ml/api/v1'; |
|
|
| export const router = express.Router(); |
|
|
| router.post('/libre', async (request, response) => { |
| try { |
| const key = readSecret(request.user.directories, SECRET_KEYS.LIBRE); |
| const url = readSecret(request.user.directories, SECRET_KEYS.LIBRE_URL); |
|
|
| if (!url) { |
| console.warn('LibreTranslate URL is not configured.'); |
| return response.sendStatus(400); |
| } |
|
|
| if (request.body.lang === 'zh-CN') { |
| request.body.lang = 'zh'; |
| } |
|
|
| if (request.body.lang === 'zh-TW') { |
| request.body.lang = 'zt'; |
| } |
|
|
| if (request.body.lang === 'pt-BR' || request.body.lang === 'pt-PT') { |
| request.body.lang = 'pt'; |
| } |
|
|
| const text = request.body.text; |
| const lang = request.body.lang; |
|
|
| if (!text || !lang) { |
| return response.sendStatus(400); |
| } |
|
|
| console.debug('Input text: ' + text); |
|
|
| 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.warn('LibreTranslate error: ', result.statusText, error); |
| return response.sendStatus(500); |
| } |
|
|
| |
| const json = await result.json(); |
| console.debug('Translated text: ' + json.translatedText); |
|
|
| return response.send(json.translatedText); |
| } catch (error) { |
| console.error('Translation error: ' + error.message); |
| return response.sendStatus(500); |
| } |
| }); |
|
|
| router.post('/google', async (request, response) => { |
| try { |
| if (request.body.lang === 'pt-BR') { |
| request.body.lang = 'pt'; |
| } |
|
|
| const text = String(request.body.text ?? ''); |
| const lang = String(request.body.lang ?? ''); |
|
|
| if (!text || !lang) { |
| return response.sendStatus(400); |
| } |
|
|
| console.debug('Input text: ' + text); |
|
|
| const translator = new Translator({ to: lang }); |
| const translatedText = await translator.translate(text).then(result => result.text); |
|
|
| response.setHeader('Content-Type', 'text/plain; charset=utf-8'); |
| console.debug('Translated text: ' + translatedText); |
| return response.send(translatedText); |
| } catch (error) { |
| console.error('Translation error', error); |
| return response.sendStatus(500); |
| } |
| }); |
|
|
| router.post('/yandex', async (request, response) => { |
| try { |
| if (request.body.lang === 'pt-PT') { |
| request.body.lang = 'pt'; |
| } |
|
|
| if (request.body.lang === 'zh-CN' || request.body.lang === 'zh-TW') { |
| request.body.lang = 'zh'; |
| } |
|
|
| const chunks = request.body.chunks; |
| const lang = request.body.lang; |
|
|
| if (!chunks || !lang) { |
| return response.sendStatus(400); |
| } |
|
|
| |
| let inputText = ''; |
|
|
| const params = new URLSearchParams(); |
| for (const chunk of chunks) { |
| params.append('text', chunk); |
| inputText += chunk; |
| } |
| params.append('lang', lang); |
| const ucid = uuidv4().replaceAll('-', ''); |
|
|
| console.debug('Input text: ' + inputText); |
|
|
| const result = await fetch(`https://translate.yandex.net/api/v1/tr.json/translate?ucid=${ucid}&srv=android&format=text`, { |
| method: 'POST', |
| body: params, |
| headers: { |
| 'Content-Type': 'application/x-www-form-urlencoded', |
| }, |
| }); |
|
|
| if (!result.ok) { |
| const error = await result.text(); |
| console.warn('Yandex error: ', result.statusText, error); |
| return response.sendStatus(500); |
| } |
|
|
| |
| const json = await result.json(); |
| const translated = json.text.join(); |
| console.debug('Translated text: ' + translated); |
|
|
| return response.send(translated); |
| } catch (error) { |
| console.error('Translation error: ' + error.message); |
| return response.sendStatus(500); |
| } |
| }); |
|
|
| router.post('/lingva', async (request, response) => { |
| try { |
| const secretUrl = readSecret(request.user.directories, SECRET_KEYS.LINGVA_URL); |
| const baseUrl = secretUrl || LINGVA_DEFAULT; |
|
|
| if (!secretUrl && baseUrl === LINGVA_DEFAULT) { |
| console.warn('Lingva URL is using default value.', LINGVA_DEFAULT); |
| } |
|
|
| if (request.body.lang === 'zh-CN' || request.body.lang === 'zh-TW') { |
| request.body.lang = 'zh'; |
| } |
|
|
| if (request.body.lang === 'pt-BR' || request.body.lang === 'pt-PT') { |
| request.body.lang = 'pt'; |
| } |
|
|
| const text = request.body.text; |
| const lang = request.body.lang; |
|
|
| if (!text || !lang) { |
| return response.sendStatus(400); |
| } |
|
|
| console.debug('Input text: ' + text); |
|
|
| const url = urlJoin(baseUrl, 'auto', lang, encodeURIComponent(text)); |
| const result = await fetch(url); |
|
|
| if (!result.ok) { |
| const error = await result.text(); |
| console.warn('Lingva error: ', result.statusText, error); |
| } |
|
|
| |
| const data = await result.json(); |
| console.debug('Translated text: ' + data.translation); |
| return response.send(data.translation); |
| } catch (error) { |
| console.error('Translation error', error); |
| return response.sendStatus(500); |
| } |
| }); |
|
|
| router.post('/deepl', async (request, response) => { |
| try { |
| const key = readSecret(request.user.directories, SECRET_KEYS.DEEPL); |
|
|
| if (!key) { |
| console.warn('DeepL key is not configured.'); |
| return response.sendStatus(400); |
| } |
|
|
| if (request.body.lang === 'zh-CN' || request.body.lang === 'zh-TW') { |
| request.body.lang = 'ZH'; |
| } |
|
|
| const text = request.body.text; |
| const lang = request.body.lang; |
| const formality = getConfigValue('deepl.formality', 'default'); |
|
|
| if (!text || !lang) { |
| return response.sendStatus(400); |
| } |
|
|
| console.debug('Input text: ' + text); |
|
|
| const params = new URLSearchParams(); |
| params.append('text', text); |
| params.append('target_lang', lang); |
|
|
| if (['de', 'fr', 'it', 'es', 'nl', 'ja', 'ru', 'pt-BR', 'pt-PT'].includes(lang)) { |
| params.append('formality', formality); |
| } |
|
|
| const endpoint = request.body.endpoint === 'pro' |
| ? 'https://api.deepl.com/v2/translate' |
| : 'https://api-free.deepl.com/v2/translate'; |
|
|
| const result = await fetch(endpoint, { |
| method: 'POST', |
| body: params, |
| headers: { |
| 'Accept': 'application/json', |
| 'Authorization': `DeepL-Auth-Key ${key}`, |
| 'Content-Type': 'application/x-www-form-urlencoded', |
| }, |
| }); |
|
|
| if (!result.ok) { |
| const error = await result.text(); |
| console.warn('DeepL error: ', result.statusText, error); |
| return response.sendStatus(500); |
| } |
|
|
| |
| const json = await result.json(); |
| console.debug('Translated text: ' + json.translations[0].text); |
|
|
| return response.send(json.translations[0].text); |
| } catch (error) { |
| console.error('Translation error: ' + error.message); |
| return response.sendStatus(500); |
| } |
| }); |
|
|
| router.post('/onering', async (request, response) => { |
| try { |
| const secretUrl = readSecret(request.user.directories, SECRET_KEYS.ONERING_URL); |
| const url = secretUrl || ONERING_URL_DEFAULT; |
|
|
| if (!url) { |
| console.warn('OneRing URL is not configured.'); |
| return response.sendStatus(400); |
| } |
|
|
| if (!secretUrl && url === ONERING_URL_DEFAULT) { |
| console.info('OneRing URL is using default value.', ONERING_URL_DEFAULT); |
| } |
|
|
| if (request.body.lang === 'pt-BR' || request.body.lang === 'pt-PT') { |
| request.body.lang = 'pt'; |
| } |
|
|
| 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.debug('Input text: ' + text); |
|
|
| const fetchUrl = new URL(url); |
| fetchUrl.search = params.toString(); |
|
|
| const result = await fetch(fetchUrl, { |
| method: 'GET', |
| }); |
|
|
| if (!result.ok) { |
| const error = await result.text(); |
| console.warn('OneRing error: ', result.statusText, error); |
| return response.sendStatus(500); |
| } |
|
|
| |
| const data = await result.json(); |
| console.debug('Translated text: ' + data.result); |
|
|
| return response.send(data.result); |
| } catch (error) { |
| console.error('Translation error: ' + error.message); |
| return response.sendStatus(500); |
| } |
| }); |
|
|
| router.post('/deeplx', async (request, response) => { |
| try { |
| const secretUrl = readSecret(request.user.directories, SECRET_KEYS.DEEPLX_URL); |
| const url = secretUrl || DEEPLX_URL_DEFAULT; |
|
|
| if (!url) { |
| console.warn('DeepLX URL is not configured.'); |
| return response.sendStatus(400); |
| } |
|
|
| if (!secretUrl && url === DEEPLX_URL_DEFAULT) { |
| console.info('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' || request.body.lang === 'zh-TW') { |
| lang = 'ZH'; |
| } |
|
|
| if (!text || !lang) { |
| return response.sendStatus(400); |
| } |
|
|
| console.debug('Input text: ' + text); |
|
|
| 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', |
| }, |
| }); |
|
|
| if (!result.ok) { |
| const error = await result.text(); |
| console.warn('DeepLX error: ', result.statusText, error); |
| return response.sendStatus(500); |
| } |
|
|
| |
| const json = await result.json(); |
| console.debug('Translated text: ' + json.data); |
|
|
| return response.send(json.data); |
| } catch (error) { |
| console.error('DeepLX translation error: ' + error.message); |
| return response.sendStatus(500); |
| } |
| }); |
|
|
| router.post('/bing', async (request, response) => { |
| try { |
| const text = request.body.text; |
| let lang = request.body.lang; |
|
|
| if (request.body.lang === 'zh-CN') { |
| lang = 'zh-Hans'; |
| } |
|
|
| if (request.body.lang === 'zh-TW') { |
| lang = 'zh-Hant'; |
| } |
|
|
| if (request.body.lang === 'pt-BR') { |
| lang = 'pt'; |
| } |
|
|
| if (!text || !lang) { |
| return response.sendStatus(400); |
| } |
|
|
| console.debug('Input text: ' + text); |
|
|
| const result = await bingTranslate(text, null, lang); |
| const translatedText = result?.translation; |
| console.debug('Translated text: ' + translatedText); |
| return response.send(translatedText); |
| } catch (error) { |
| console.error('Translation error', error); |
| return response.sendStatus(500); |
| } |
| }); |
|
|