- /* eslint-disable no-multi-spaces */
-
- // ==UserScript==
- // @name Baidu Translate
- // @name:zh-CN 百度翻译
- // @name:en Baidu Translate
- // @namespace PY-DNG APIS
- // @version 0.2.5
- // @description Baidu translate api support for userscripts. No need for baidu's token.
- // @description:zh-CN 用户脚本版的百度翻译API支持,无需申请token,@require即用
- // @description:en Baidu translate api support for userscripts. No need for baidu's token.
- // @author PY-DNG
- // @license MIT
- // @icon 
- // @grant GM_xmlhttpRequest
- // @connect fanyi.baidu.com
- // ==/UserScript==
-
- /* Usage:
- ** baidu_translate(details): details = {
- text: <string> Text you want to translate.
- Required argument, no default value.
- callback: <function> Callback function when translation succeed,
- e.g. function(result) {console.log(result);}.
- Argument `result` will be <string> translate result.
- Required argument, no default value.
- [src]: <string> Source language; e.g. 'zh' for Chinese, 'en' for English, 'ja' for Japanese, ....
- Default: Auto detect.
- [dst]: <string> Destination language,
- Default: 'zh'.
- [split]: <number> Text spliting max length, just leave blank if you don't know what this means.
- What it is: Baidu has limited that up to 5000 letters can be translated each single API
- request, so the translate function will split text by about every ${split} letters, and
- translates each of them with a single API request then concatenate the translate result
- together again. Don't worry if any sentence to be splited, actually the translate
- function splits text with '\n' first, then concatenate the '\n'-splited strings as long
- as possible while keeping its length <= ${split}.
- Default: 5000
- [onerror]: <function> Callback function when translation failed,
- e.g. function(reason) {console.log(reason);}.
- Argument: `reason` may be anything that causes its failure, just for debugging and do
- not use it in production.
- Default: function() {}.
- [retry]: <number> Times to retry before onerror function being called or an error being thrown,
- Default: 3.
- }
- ** Or:
- ** baidu_translate(text, src, dst, callback, onerror, split, retry)
- **
- ** Overloads:
- ** baidu_translate(text, src, dst, callback, onerror, split)
- ** baidu_translate(text, dst, callback, onerror, split)
- ** baidu_translate(text, callback, onerror, split)
- ** baidu_translate(text, callback, onerror)
- ** baidu_translate(text, callback)
- */
-
- let baidu_translate, bdTransReady;
- [baidu_translate, bdTransReady] = (function() {
- const BDT = new BaiduTranslateAPI();
- return [baidu_translate, bdTransReady];
-
- function baidu_translate() {
- // Check BDT ready
- if (!BDT.gtk || !BDT.token) {
- //throw new Error('BaiduTranslateAPI not ready');
- return false;
- }
-
- // Get argument
- const [text, src, dst, callback, onerror, split, retry] = parseArgs([...arguments], [
- function(args, defaultValues) {
- const details = args[0];
- const parsed = [...defaultValues];
- details.hasOwnProperty('text') && (parsed[0] = details.text);
- details.hasOwnProperty('src') && (parsed[1] = details.src);
- details.hasOwnProperty('dst') && (parsed[2] = details.dst);
- details.hasOwnProperty('callback') && (parsed[3] = details.callback);
- details.hasOwnProperty('onerror') && (parsed[4] = details.onerror);
- details.hasOwnProperty('split') && (parsed[5] = details.split);
- details.hasOwnProperty('retry') && (parsed[6] = details.retry);
- return parsed;
- },
- [1,4],
- [1,4,5],
- [1,4,5,6],
- [1,3,4,5,6],
- [1,2,3,4,5,6],
- [1,2,3,4,5,6,7]
- ], ['', undefined, undefined, function() {}, function() {}, 5000, 3]);
-
- // Split lines
- const textarr = text.replaceAll('\r\n', '\n').replaceAll('\n\r', '\n').replaceAll('\r', '\n').split('\n');
- if (textarr.some((t) => (t.length > split))) {
- throw new Error('Some paragraph is longer than given split length (' + split + ')');
- }
-
- // Prepare
- const AM = new AsyncManager();
- const result = [];
- AM.onfinish = function() {
- callback(result.map(re => '\n'.repeat(re.newline_begin) + re.dst + '\n'.repeat(re.newline_end)).join('\n'));
- }
-
- // Send requests
- let index = 0;
- while (textarr.length > 0) {
- // Join translate api paragraph
- let para = '';
- while (textarr.length > 0 && para.length + textarr[0].length < split) {
- para += '\n' + textarr.shift();
- }
- para = para.substr(1);
-
- // Whether this paragraph contains \n at beginning
- const countNewLine = str => str.split('').filter(t => t === '\n').length;
- const newline_begin = countNewLine(para.match(/^\s*/) ? para.match(/^\s*/)[0] : '');
- const newline_end = countNewLine(para.match(/^\s*/) ? para.match(/\s*$/)[0] : '');
- result[index] = {src: para, dst: null, newline_begin, newline_end};
-
- // API request
- AM.add();
- BDT.translate({
- text: para,
- args: [index],
- src: src,
- dst: dst,
- onerror: onerror,
- retry: retry,
- callback: function(json, i) {
- const temp_result = json.trans_result.data.reduce(function(pre, cur) {
- return (pre.push('\n'.repeat(cur.prefixWrap) + cur.dst), pre);
- }, []).join('\n');
- result[i].dst = temp_result;
- AM.finish();
- }
- });
- index++;
- }
- AM.finishEvent = true;
-
- return true;
- }
-
- function bdTransReady(callback) {
- const callbacks = [];
- BDT.oninit = dispatchCallback;
- bdTransReady = bdTransReady_inner;
- bdTransReady_inner();
-
- function dispatchCallback() {
- callbacks.forEach(cb => cb());
- callbacks.splice(0, Infinity);
- }
-
- function bdTransReady_inner() {
- callbacks.push(callback);
- }
- }
-
- function BaiduTranslateAPI() {
- const BT = this;
- BT.gtk = BT.token = null;
- BT.inited = false;
-
- let oninit=function() {};
- Object.defineProperty(BT, 'oninit', {
- enumerable: true,
- set: (f) => (typeof f === 'function' && (oninit = f) && BT.inited && oninit()),
- get: () => (oninit)
- });
-
- init();
-
- BT.calcSign = calcSign;
- BT.translate = translate;
-
- async function translate(details) {
- const callback = details.callback;
- const text = details.text;
- const src = details.src || await langDetect(text);
- const dst = details.dst || 'zh';
- const onerror = details.onerror || function() {};
- const retry = details.retry || 0;
- const args = details.args || [];
-
- GM_xmlhttpRequest({
- method: 'POST',
- url: 'https://fanyi.baidu.com/v2transapi',
- headers: {
- 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36',
- 'Content-Type': 'application/x-www-form-urlencoded'
- },
- data: toQueryString({
- 'from': src,
- 'to': dst,
- 'query': text,
- 'simple_means_flag': 3,
- 'sign': calcSign(text),
- 'token': BT.token
- }),
- onload: function(response) {
- response.status !== 200 && Err(response);
- const json = JSON.parse(response.responseText);
- json.error && Err(json);
- callback.apply(null, [json].concat(args));
-
- function Err(e) {
- !_onerror(e) && console.log('Retrying...\nleft: ' + (retry-1).toString());
- console.log(e);
- throw new Error('Server returned with an error (logged above)');
- }
- },
- onerror: _onerror,
- ontimeout: _onerror,
- onabort: _onerror,
- });
-
- // Returns true for error, false for retry
- function _onerror(e) {
- console.log('sign = ' + calcSign(text));
- if (retry > 0) {
- setTimeout(retryRequest.bind(null, e), 500);
- return false;
- }
- onerror(e);
- return true;
- }
-
- function retryRequest(e) {
- translate({
- callback: callback,
- text: text,
- src: src,
- dst: dst,
- onerror: onerror,
- retry: retry - 1,
- args: args
- });
- }
- }
-
- function langDetect(text) {
- return new Promise((resolve, reject) => {
- GM_xmlhttpRequest({
- method: 'POST',
- url: 'https://fanyi.baidu.com/langdetect',
- headers: {
- 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36',
- 'Content-Type': 'application/x-www-form-urlencoded'
- },
- data: toQueryString({'query': text}),
- onload: function(response) {
- const json = JSON.parse(response.responseText);
- resolve(json.lan);
- },
- onerror: function(e) {reject(e)},
- ontimeout: function(e) {reject(e)},
- onabort: function(e) {reject(e)},
- });
- });
- }
-
- // Calc sign based on query-text and gtk
- function calcSign(query) {
- function a(r, o) {
- for (var t = 0; t < o.length - 2; t += 3) {
- var a = o.charAt(t + 2);
- a = a >= "a" ? a.charCodeAt(0) - 87 : Number(a),
- a = "+" === o.charAt(t + 1) ? r >>> a: r << a,
- r = "+" === o.charAt(t) ? r + a & 4294967295 : r ^ a
- }
- return r
- }
- var C = null;
- var token = function(r, _gtk) {
- var o = r.length;
- o > 30 && (r = "" + r.substr(0, 10) + r.substr(Math.floor(o / 2) - 5, 10) + r.substring(r.length, r.length - 10));
- var t = void 0,
- t = null !== C ? C: (C = _gtk || "") || "";
- for (var e = t.split("."), h = Number(e[0]) || 0, i = Number(e[1]) || 0, d = [], f = 0, g = 0; g < r.length; g++) {
- var m = r.charCodeAt(g);
- 128 > m ? d[f++] = m: (2048 > m ? d[f++] = m >> 6 | 192 : (55296 === (64512 & m) && g + 1 < r.length && 56320 === (64512 & r.charCodeAt(g + 1)) ? (m = 65536 + ((1023 & m) << 10) + (1023 & r.charCodeAt(++g)), d[f++] = m >> 18 | 240, d[f++] = m >> 12 & 63 | 128) : d[f++] = m >> 12 | 224, d[f++] = m >> 6 & 63 | 128), d[f++] = 63 & m | 128)
- }
- for (var S = h,
- u = "+-a^+6",
- l = "+-3^+b+-f",
- s = 0; s < d.length; s++) S += d[s],
- S = a(S, u);
-
- return S = a(S, l),
- S ^= i,
- 0 > S && (S = (2147483647 & S) + 2147483648),
- S %= 1e6,
- S.toString() + "." + (S ^ h)
- }
-
- return token(query, BT.gtk);
- }
-
- function toQueryString(obj) {
- const USP = new URLSearchParams();
- for (const [key, value] of Object.entries(obj)) {
- USP.append(key, value);
- }
- return USP.toString();
- }
-
- // Request token and gtk
- function init() {
- // Request twice, make sure gtk is latest,
- // or may get 998 error while requesting translate API
- requestIndex(requestIndex.bind(null, function(html) {
- BT.token = html.match(/token: ["'](.*?)["'],/)[1];
- BT.gtk = html.match(/window.gtk = ["'](.*?)["'];/)[1];
- BT.inited = true;
- oninit();
- }));
-
- function requestIndex(callback) {
- const url = 'https://fanyi.baidu.com';
- GM_xmlhttpRequest({
- method: 'GET',
- headers: {
- 'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36'
- },
- url: url,
- onload: function(response) {
- callback(response.responseText);
- }
- });
- }
- }
- }
-
- function AsyncManager() {
- const AM = this;
-
- // Ongoing xhr count
- this.taskCount = 0;
-
- // Whether generate finish events
- let finishEvent = false;
- Object.defineProperty(this, 'finishEvent', {
- configurable: true,
- enumerable: true,
- get: () => (finishEvent),
- set: (b) => {
- finishEvent = b;
- b && AM.taskCount === 0 && AM.onfinish && AM.onfinish();
- }
- });
-
- // Add one task
- this.add = () => (++AM.taskCount);
-
- // Finish one task
- this.finish = () => ((--AM.taskCount === 0 && AM.finishEvent && AM.onfinish && AM.onfinish(), AM.taskCount));
- }
-
- function parseArgs(args, rules, defaultValues=[]) {
- // args and rules should be array, but not just iterable (string is also iterable)
- if (!Array.isArray(args) || !Array.isArray(rules)) {
- throw new TypeError('parseArgs: args and rules should be array')
- }
-
- // fill rules[0]
- (!Array.isArray(rules[0]) || rules[0].length === 1) && rules.splice(0, 0, []);
-
- // max arguments length
- const count = rules.length - 1;
-
- // args.length must <= count
- if (args.length > count) {
- throw new TypeError(`parseArgs: args has more elements(${args.length}) longer than ruless'(${count})`);
- }
-
- // rules[i].length should be === i if rules[i] is an array, otherwise it should be a function
- for (let i = 1; i <= count; i++) {
- const rule = rules[i];
- if (Array.isArray(rule)) {
- if (rule.length !== i) {
- throw new TypeError(`parseArgs: rules[${i}](${rule}) should have ${i} numbers, but given ${rules[i].length}`);
- }
- if (!rule.every((num) => (typeof num === 'number' && num <= count))) {
- throw new TypeError(`parseArgs: rules[${i}](${rule}) should contain numbers smaller than count(${count}) only`);
- }
- } else if (typeof rule !== 'function') {
- throw new TypeError(`parseArgs: rules[${i}](${rule}) should be an array or a function.`)
- }
- }
-
- // Parse
- const rule = rules[args.length];
- let parsed;
- if (Array.isArray(rule)) {
- parsed = [...defaultValues];
- for (let i = 0; i < rule.length; i++) {
- parsed[rule[i]-1] = args[i];
- }
- } else {
- parsed = rule(args, defaultValues);
- }
- return parsed;
- }
- }) ();