Baidu translate api support for userscripts. No need for baidu's token.
สคริปต์นี้ไม่ควรถูกติดตั้งโดยตรง มันเป็นคลังสำหรับสคริปต์อื่น ๆ เพื่อบรรจุด้วยคำสั่งเมทา // @require https://update.greasyfork.org/scripts/452362/1281581/Baidu%20Translate.js
/* eslint-disable no-multi-spaces */ /* eslint-disable no-return-assign */ /* eslint-disable curly */ /* eslint-disable no-sequences */ // ==UserScript== // @name Baidu Translate // @name:zh-CN 百度翻译 // @name:en Baidu Translate // @namespace PY-DNG APIS // @version 0.2.7 // @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 ###////////////+sWN//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/+seQ////////////+bl1//WIFP/2kCX//vLm//758////fv//efQ//zhxP/82bP/+9Gk//rJlP/82bT///////////////7//eLF//aXMv/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/6x5D////////////5uXX/9YgU//aYM//4sGL/+LBj//m4cv/5vHr/+sOJ//vLmP/70qb//Nm0//zfwP/+9Or////////////6xo//9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WMHP/2kCT/9pAk//WLGv/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WNHv/837//+r+B//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/2jiDy9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9o4g8vimUL71iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/4plC+/Ny5TPWIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//zbuEwAAAAA+sGFhPWIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/6wYWEAAAAAAAAAAAAAAAA/Nu4TPimUL71jh/y9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WIFP/1iBT/9YgU//WOH/L4plC+/Ny5TAAAAAAAAAAA4AAAB4AAAAGAAAABAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACAAAABgAAAAeAAAAc= // @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(r###lt) {console.log(r###lt);}. Argument `r###lt` will be <string> translate r###lt. 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 r###lt 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 r###lt = []; AM.onfinish = function() { callback(r###lt.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] : ''); r###lt[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_r###lt = json.trans_r###lt.data.reduce(function(pre, cur) { return (pre.push('\n'.repeat(cur.prefixWrap) + cur.dst), pre); }, []).join('\n'); r###lt[i].dst = temp_r###lt; AM.finish(); } }); index++; } AM.finishEvent = true; return true; } function bdTransReady(callback) { bdTransReady.callbacks = bdTransReady.callbacks || []; bdTransReady.callbacks.push(callback); BDT.oninit = dispatchCallback; function dispatchCallback() { bdTransReady.callbacks.forEach(cb => cb()); bdTransReady.callbacks.splice(0, Infinity); } } 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 e(t, e) { (null == e || e > t.length) && (e = t.length); for (var n = 0, r = new Array(e); n < e; n++) r[n] = t[n]; return r } function n(t, e) { for (var n = 0; n < e.length - 2; n += 3) { var r = e.charAt(n + 2); r = "a" <= r ? r.charCodeAt(0) - 87 : Number(r), r = "+" === e.charAt(n + 1) ? t >>> r : t << r, t = "+" === e.charAt(n) ? t + r & 4294967295 : t ^ r } return t } var r = null; var token = function(t, _gtk) { var o, i = t.match(/[\uD800-\uDBFF][\uDC00-\uDFFF]/g); if (null === i) { var a = t.length; a > 30 && (t = "".concat(t.substr(0, 10)).concat(t.substr(Math.floor(a / 2) - 5, 10)).concat(t.substr(-10, 10))) } else { for (var s = t.split(/[\uD800-\uDBFF][\uDC00-\uDFFF]/), c = 0, u = s.length, l = []; c < u; c++) "" !== s[c] && l.push.apply(l, function(t) { if (Array.isArray(t)) return e(t) }(o = s[c].split("")) || function(t) { if ("undefined" != typeof Symbol && null != t[Symbol.iterator] || null != t["@@iterator"]) return Array.from(t) }(o) || function(t, n) { if (t) { if ("string" == typeof t) return e(t, n); var r = Object.prototype.toString.call(t).slice(8, -1); return "Object" === r && t.constructor && (r = t.constructor.name), "Map" === r || "Set" === r ? Array.from(t) : "Arguments" === r || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(r) ? e(t, n) : void 0 } }(o) || function() { throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.") }()), c !== u - 1 && l.push(i[c]); var p = l.length; p > 30 && (t = l.slice(0, 10).join("") + l.slice(Math.floor(p / 2) - 5, Math.floor(p / 2) + 5).join("") + l.slice(-10).join("")) } for (var h = (_gtk || "").split("."), f = Number(h[0]) || 0, m = Number(h[1]) || 0, g = [], y = 0, v = 0; v < t.length; v++) { var _ = t.charCodeAt(v); _ < 128 ? g[y++] = _ : (_ < 2048 ? g[y++] = _ >> 6 | 192 : (55296 == (64512 & _) && v + 1 < t.length && 56320 == (64512 & t.charCodeAt(v + 1)) ? (_ = 65536 + ((1023 & _) << 10) + (1023 & t.charCodeAt(++v)), g[y++] = _ >> 18 | 240, g[y++] = _ >> 12 & 63 | 128) : g[y++] = _ >> 12 | 224, g[y++] = _ >> 6 & 63 | 128), g[y++] = 63 & _ | 128) } for (var b = f, w = '+-a^+6', k = '+-3^+b+-f', x = 0; x < g.length; x++) b = n(b += g[x], w); return b = n(b, k), (b ^= m) < 0 && (b = 2147483648 + (2147483647 & b)), "".concat((b %= 1e6).toString(), ".").concat(b ^ f) } 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; } }) ();