// @grant require; // @grant nodejs; const $ = new Env('扫码获取京东cookie'); let s_token, cookies, guid, lsid, lstoken, okl_token, token let evuid = 'jdcookie' !(async () => { await moduleCheck(['got', 'tough-cookie', 'qrcode-npm']) await loginEntrance() await generateQrcode() await getCookie() })() .catch((e) => { $.log('', `❌ ${$.name}, 失败! 原因: ${e}!`, '') }) .finally(() => { $.done(); }) function ckJDPush(cookies, key) { if (!cookies) { console.log('请先输入 cookie') } const cName = (ck)=>{ let cname = ck.match(/pt_pin=(\S+);/) if (cname && cname[1]) { return cname[1] } return false } if (!key) { key = 'CookieJD' } if (key === 'CookieJD2' || key === 'CookieJD') { let sn = cName(cookies) if (sn) { $store.put(cookies, key) let msg = '成功保存账号 ' + sn + ' 的 cookie 到 ' + key console.log(msg) return msg } return '无法识别的 cookie' } if (key !== 'CookiesJD') { console.log('key 不要乱输') return 'key 不要乱输' } let csjd = $store.get('CookiesJD'), oldc = {}, fmsg = '' if (csjd) { try { let jcs = JSON.parse(csjd) if (jcs.length){ jcs.forEach((ck, index)=>{ if (ck && ck.cookie) { let cname = cName(ck.cookie) if (cname) { oldc[cname] = { cookie: ck.cookie, index } } } }) } } catch (e) { console.log('原 CookiesJD 数据如下:', csjd, '不符合格式,将被自动清除') } } else { console.log('没有检测 CookiesJD 相关数据,将自动进行创建') } if (typeof(cookies) === 'string') { cookies = [cookies] } else if (typeof(cookies) === 'object' && cookies.length) { console.log('即将添加', cookies.length, '个账号') } else { fmsg = '未知类型 cookies' console.log(fmsg, cookies) return fmsg } cookies.forEach(ck=>{ let cn = cName(ck) if (cn) { let msg if (oldc[cn]) { oldc[cn].cookie = ck msg = '替换京东账号 ' + cn } else { oldc[cn] = { cookie: ck } msg = '新增京东账号 ' + cn } console.log(msg) fmsg += '\n' + msg } else { console.log('无效的 cookie', ck) } }) let fck = [] for (let cval in oldc) { fck.push({ cookie: oldc[cval].cookie }) } $store.put(JSON.stringify(fck, null, 2), 'CookiesJD') return fmsg } const qrcode = { img(text){ let qc = require('qrcode-npm') let qr = qc.qrcode(6, 'L') qr.addData(text) qr.make() return qr.createImgTag(6) }, generate(url){ console.log('将', url, '转换为二维码进行显示') $evui({ id: evuid, title: '打开京东 APP 扫码获取 cookie', width: 800, height: 600, content: `
Powered
BY elecV2P
${this.img(url)}测试使用
请勿用于
实际生产环境中

扫码成功后,下面输入框第一行表示获取到的 cookie 值
第二行为该 cookie 保存的关键字 KEY,默认为 CookieJD
可修改为 CookieJD2 (表示添加或替换第二个京东 cookie)
或者 CookiesJD (表示在 CookiesJD 中新增一个 cookie)

`, style: { cbdata: "height: 132px;", }, cbable: true, cbhint: '扫码成功后,第一行表示 cookie 值\n第二行表示对应保存的 KEY', cblabel: '确定保存' }, data=>{ let fck = data.split(/\r|\n/) console.log('data from client:', fck) if (fck && fck.length) { let res = ckJDPush(fck[0], fck[1]) $message.success(res) } else { console.log('没有收到任何数据') $message.error('后台没有收到任何数据') } }) } } function loginEntrance() { return new Promise((resolve) => { $.get(taskUrl(), async (err, resp, data) => { try { if (err) { console.log(`${JSON.stringify(err)}`) console.log(`${$.name} API请求失败,请检查网路重试`); } else { $.headers = resp.headers; $.data = JSON.parse(data); await formatSetCookies($.headers, $.data); } } catch (e) { $.logErr(e, resp) } finally { resolve(); } }) }) } function generateQrcode() { return new Promise((resolve) => { $.post(taskPostUrl(), (err, resp, data) => { try { if (err) { console.log(`${JSON.stringify(err)}`) console.log(`${$.name} API请求失败,请检查网路重试`); } else { $.stepsHeaders = resp.headers; data = JSON.parse(data); token = data['token']; const setCookie = resp.headers['set-cookie'][0]; okl_token = setCookie.substring(setCookie.indexOf("=") + 1, setCookie.indexOf(";")) const url = 'https://plogin.m.jd.com/cgi-bin/m/tmauth?appid=300&client_type=m&token=' + token; console.debug('token', token, 'okl_token', okl_token, '二维码url', url) qrcode.generate(url); // 输出二维码 console.log("请打开 京东APP 扫码登录(二维码有效期为1分钟)"); } } catch (e) { $.logErr(e, resp) } finally { resolve(); } }) }) } function checkLogin() { return new Promise((resolve) => { const options = { url: `https://plogin.m.jd.com/cgi-bin/m/tmauthchecktoken?&token=${token}&ou_state=0&okl_token=${okl_token}`, body: `lang=chs&appid=300&source=wq_passport&returnurl=https://wqlogin2.jd.com/passport/LoginRedirect?state=${Date.now()}&returnurl=//home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action`, headers: { 'Referer': `https://plogin.m.jd.com/login/login?appid=300&returnurl=https://wqlogin2.jd.com/passport/LoginRedirect?state=${Date.now()}&returnurl=//home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action&source=wq_passport`, 'Cookie': cookies, 'Connection': 'Keep-Alive', 'Content-Type': 'application/x-www-form-urlencoded; Charset=UTF-8', 'Accept': 'application/json, text/plain, */*', 'User-Agent': 'jdapp;android;10.0.5;11;0393465333165363-5333430323261366;network/wifi;model/M2102K1C;osVer/30;appBuild/88681;partner/lc001;eufv/1;jdSupportDarkMode/0;Mozilla/5.0 (Linux; Android 11; M2102K1C Build/RKQ1.201112.002; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/77.0.3865.120 MQQBrowser/6.2 TBS/045534 Mobile Safari/537.36', } } $.post(options, (err, resp, data) => { try { if (err) { console.log(`${JSON.stringify(err)}`) console.log(`${$.name} API请求失败,请检查网路重试`); } else { data = JSON.parse(data); $.checkLoginHeaders = resp.headers; // $.log(`errcode:${data['errcode']}`) } } catch (e) { $.logErr(e, resp) } finally { resolve(data || {}); } }) }) } function getCookie() { let time = 60 $.timer = setInterval(async () => { const checkRes = await checkLogin(); if (checkRes['errcode'] === 0) { //扫描登录成功 $.log(`扫描登录成功\n`) clearInterval($.timer); await formatCookie($.checkLoginHeaders); $.done(); } else if (checkRes['errcode'] === 21) { $.log(`二维码已失效,请重新获取二维码重新扫描\n`); clearInterval($.timer); $.done(); } else if (checkRes['errcode'] === 176) { //未扫描登录 } else { $.log(`其他异常:${JSON.stringify(checkRes)}\n`); clearInterval($.timer); $.done(); } if (time < 0) { clearInterval($.timer); console.log('扫码超时') $ws.send({ type: 'evui', data: { id: evuid, data: '扫码超时,如有需要请重新运行脚本' }}) $message.error('扫码超时,如有需要请重新运行脚本', 10) $.done() } else { time-- } }, 1000) } function formatCookie(headers) { new Promise(resolve => { let pt_key = headers['set-cookie'][1] pt_key = pt_key.substring(pt_key.indexOf("=") + 1, pt_key.indexOf(";")) let pt_pin = headers['set-cookie'][2] pt_pin = pt_pin.substring(pt_pin.indexOf("=") + 1, pt_pin.indexOf(";")) const cookie1 = "pt_key=" + pt_key + ";pt_pin=" + pt_pin + ";"; $.UserName = decodeURIComponent(cookie1.match(/pt_pin=(.+?);/) && cookie1.match(/pt_pin=(.+?);/)[1]) $.log(`京东用户名:${$.UserName} 登录成功,此cookie(有效期为90天)如下:`); $.log(`\n${cookie1}\n`); // 发送给前端 $ws.send({ type: 'evui', data: { id: evuid, data: cookie1 + '\n' + 'CookieJD' }}) resolve() }) } function formatSetCookies(headers, body) { new Promise(resolve => { s_token = body['s_token'] guid = headers['set-cookie'][0] guid = guid.substring(guid.indexOf("=") + 1, guid.indexOf(";")) lsid = headers['set-cookie'][2] lsid = lsid.substring(lsid.indexOf("=") + 1, lsid.indexOf(";")) lstoken = headers['set-cookie'][3] lstoken = lstoken.substring(lstoken.indexOf("=") + 1, lstoken.indexOf(";")) cookies = "guid=" + guid + "; lang=chs; lsid=" + lsid + "; lstoken=" + lstoken + "; " resolve() }) } function taskUrl() { return { url: `https://plogin.m.jd.com/cgi-bin/mm/new_login_entrance?lang=chs&appid=300&returnurl=https://wq.jd.com/passport/LoginRedirect?state=${Date.now()}&returnurl=https://home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action&source=wq_passport`, headers: { 'Connection': 'Keep-Alive', 'Content-Type': 'application/x-www-form-urlencoded', 'Accept': 'application/json, text/plain, */*', 'Accept-Language': 'zh-cn', 'Referer': `https://plogin.m.jd.com/login/login?appid=300&returnurl=https://wq.jd.com/passport/LoginRedirect?state=${Date.now()}&returnurl=https://home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action&source=wq_passport`, 'User-Agent': 'jdapp;android;10.0.5;11;0393465333165363-5333430323261366;network/wifi;model/M2102K1C;osVer/30;appBuild/88681;partner/lc001;eufv/1;jdSupportDarkMode/0;Mozilla/5.0 (Linux; Android 11; M2102K1C Build/RKQ1.201112.002; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/77.0.3865.120 MQQBrowser/6.2 TBS/045534 Mobile Safari/537.36', 'Host': 'plogin.m.jd.com' } } } function taskPostUrl() { return { url: `https://plogin.m.jd.com/cgi-bin/m/tmauthreflogurl?s_token=${s_token}&v=${Date.now()}&remember=true`, body: `lang=chs&appid=300&source=wq_passport&returnurl=https://wqlogin2.jd.com/passport/LoginRedirect?state=${Date.now()}&returnurl=//home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action`, headers: { 'Connection': 'Keep-Alive', 'Content-Type': 'application/x-www-form-urlencoded', 'Accept': 'application/json, text/plain, */*', 'Accept-Language': 'zh-cn', 'Referer': `https://plogin.m.jd.com/login/login?appid=300&returnurl=https://wq.jd.com/passport/LoginRedirect?state=${Date.now()}&returnurl=https://home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action&source=wq_passport`, 'User-Agent': 'jdapp;android;10.0.5;11;0393465333165363-5333430323261366;network/wifi;model/M2102K1C;osVer/30;appBuild/88681;partner/lc001;eufv/1;jdSupportDarkMode/0;Mozilla/5.0 (Linux; Android 11; M2102K1C Build/RKQ1.201112.002; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/77.0.3865.120 MQQBrowser/6.2 TBS/045534 Mobile Safari/537.36', 'Host': 'plogin.m.jd.com' } } } async function moduleCheck(name, install = true) { const fs = require('fs') const path = require('path') if (Array.isArray(name)) { name = name.filter(n=>{ let mfolder = path.join('node_modules', n) if (fs.existsSync(mfolder)) { console.log('module', n, 'installed') return false } return true }) } else if (typeof(name) === 'string') { let mfolder = path.join('node_modules', name) if (fs.existsSync(mfolder)) { console.log('module', name, 'installed') name = [] } else { name = [name] } } else { console.log('unknow module name type', name) return false } if (name.length === 0) { console.log('all check modules are installed') return true } name = name.join(' ') console.log('module', name, 'not installed yet') if (install) { try { await execP('yarn add ' + name) return true } catch(e) { console.error(e) return false } } return false } function execP(command) { console.log('start run command', command) return new Promise((resolve, reject)=>{ $exec(command, { timeout: 0, cb(data, error, finish){ if (finish) { console.log(command, 'finished') resolve(data) } error ? reject(error) : console.log(data) } }) }) } // prettier-ignore function Env(name, opts) { class Http { constructor(env) { this.env = env } send(opts, method = 'GET') { opts = typeof opts === 'string' ? { url: opts } : opts let sender = this.get if (method === 'POST') { sender = this.post } return new Promise((resolve, reject) => { sender.call(this, opts, (err, resp, body) => { if (err) reject(err) else resolve(resp) }) }) } get(opts) { return this.send.call(this.env, opts) } post(opts) { return this.send.call(this.env, opts, 'POST') } } return new (class { constructor(name, opts) { this.name = name this.http = new Http(this) this.data = null this.dataFile = 'box.dat' this.logs = [] this.isMute = false this.isNeedRewrite = false this.logSeparator = '\n' this.startTime = new Date().getTime() Object.assign(this, opts) this.log('', `🔔${this.name}, 开始!`) } isNode() { return true return 'undefined' !== typeof module && !!module.exports } isQuanX() { return false return 'undefined' !== typeof $task } isSurge() { return false return 'undefined' !== typeof $httpClient && 'undefined' === typeof $loon } isLoon() { return false return 'undefined' !== typeof $loon } toObj(str, defaultValue = null) { try { return JSON.parse(str) } catch { return defaultValue } } toStr(obj, defaultValue = null) { try { return JSON.stringify(obj) } catch { return defaultValue } } getjson(key, defaultValue) { let json = defaultValue const val = this.getdata(key) if (val) { try { json = JSON.parse(this.getdata(key)) } catch {} } return json } setjson(val, key) { try { return this.setdata(JSON.stringify(val), key) } catch { return false } } getScript(url) { return new Promise((resolve) => { this.get({ url }, (err, resp, body) => resolve(body)) }) } runScript(script, runOpts) { return new Promise((resolve) => { let httpapi = this.getdata('@chavy_boxjs_userCfgs.httpapi') httpapi = httpapi ? httpapi.replace(/\n/g, '').trim() : httpapi let httpapi_timeout = this.getdata('@chavy_boxjs_userCfgs.httpapi_timeout') httpapi_timeout = httpapi_timeout ? httpapi_timeout * 1 : 20 httpapi_timeout = runOpts && runOpts.timeout ? runOpts.timeout : httpapi_timeout const [key, addr] = httpapi.split('@') const opts = { url: `http://${addr}/v1/scripting/evaluate`, body: { script_text: script, mock_type: 'cron', timeout: httpapi_timeout }, headers: { 'X-Key': key, 'Accept': '*/*' } } this.post(opts, (err, resp, body) => resolve(body)) }).catch((e) => this.logErr(e)) } loaddata() { if (this.isNode()) { this.fs = this.fs ? this.fs : require('fs') this.path = this.path ? this.path : require('path') const curDirDataFilePath = this.path.resolve(this.dataFile) const rootDirDataFilePath = this.path.resolve(process.cwd(), this.dataFile) const isCurDirDataFile = this.fs.existsSync(curDirDataFilePath) const isRootDirDataFile = !isCurDirDataFile && this.fs.existsSync(rootDirDataFilePath) if (isCurDirDataFile || isRootDirDataFile) { const datPath = isCurDirDataFile ? curDirDataFilePath : rootDirDataFilePath try { return JSON.parse(this.fs.readFileSync(datPath)) } catch (e) { return {} } } else return {} } else return {} } writedata() { if (this.isNode()) { this.fs = this.fs ? this.fs : require('fs') this.path = this.path ? this.path : require('path') const curDirDataFilePath = this.path.resolve(this.dataFile) const rootDirDataFilePath = this.path.resolve(process.cwd(), this.dataFile) const isCurDirDataFile = this.fs.existsSync(curDirDataFilePath) const isRootDirDataFile = !isCurDirDataFile && this.fs.existsSync(rootDirDataFilePath) const jsondata = JSON.stringify(this.data) if (isCurDirDataFile) { this.fs.writeFileSync(curDirDataFilePath, jsondata) } else if (isRootDirDataFile) { this.fs.writeFileSync(rootDirDataFilePath, jsondata) } else { this.fs.writeFileSync(curDirDataFilePath, jsondata) } } } lodash_get(source, path, defaultValue = undefined) { const paths = path.replace(/\[(\d+)\]/g, '.$1').split('.') let result = source for (const p of paths) { result = Object(result)[p] if (result === undefined) { return defaultValue } } return result } lodash_set(obj, path, value) { if (Object(obj) !== obj) return obj if (!Array.isArray(path)) path = path.toString().match(/[^.[\]]+/g) || [] path .slice(0, -1) .reduce((a, c, i) => (Object(a[c]) === a[c] ? a[c] : (a[c] = Math.abs(path[i + 1]) >> 0 === +path[i + 1] ? [] : {})), obj)[ path[path.length - 1] ] = value return obj } getdata(key) { let val = this.getval(key) // 如果以 @ if (/^@/.test(key)) { const [, objkey, paths] = /^@(.*?)\.(.*?)$/.exec(key) const objval = objkey ? this.getval(objkey) : '' if (objval) { try { const objedval = JSON.parse(objval) val = objedval ? this.lodash_get(objedval, paths, '') : val } catch (e) { val = '' } } } return val } setdata(val, key) { let issuc = false if (/^@/.test(key)) { const [, objkey, paths] = /^@(.*?)\.(.*?)$/.exec(key) const objdat = this.getval(objkey) const objval = objkey ? (objdat === 'null' ? null : objdat || '{}') : '{}' try { const objedval = JSON.parse(objval) this.lodash_set(objedval, paths, val) issuc = this.setval(JSON.stringify(objedval), objkey) } catch (e) { const objedval = {} this.lodash_set(objedval, paths, val) issuc = this.setval(JSON.stringify(objedval), objkey) } } else { issuc = this.setval(val, key) } return issuc } getval(key) { if (this.isSurge() || this.isLoon()) { return $persistentStore.read(key) } else if (this.isQuanX()) { return $prefs.valueForKey(key) } else if (this.isNode()) { this.data = this.loaddata() return this.data[key] } else { return (this.data && this.data[key]) || null } } setval(val, key) { if (this.isSurge() || this.isLoon()) { return $persistentStore.write(val, key) } else if (this.isQuanX()) { return $prefs.setValueForKey(val, key) } else if (this.isNode()) { this.data = this.loaddata() this.data[key] = val this.writedata() return true } else { return (this.data && this.data[key]) || null } } initGotEnv(opts) { this.got = this.got ? this.got : require('got') this.cktough = this.cktough ? this.cktough : require('tough-cookie') this.ckjar = this.ckjar ? this.ckjar : new this.cktough.CookieJar() if (opts) { opts.headers = opts.headers ? opts.headers : {} if (undefined === opts.headers.Cookie && undefined === opts.cookieJar) { opts.cookieJar = this.ckjar } } } get(opts, callback = () => {}) { if (opts.headers) { delete opts.headers['Content-Type'] delete opts.headers['Content-Length'] } if (this.isSurge() || this.isLoon()) { if (this.isSurge() && this.isNeedRewrite) { opts.headers = opts.headers || {} Object.assign(opts.headers, { 'X-Surge-Skip-Scripting': false }) } $httpClient.get(opts, (err, resp, body) => { if (!err && resp) { resp.body = body resp.statusCode = resp.status } callback(err, resp, body) }) } else if (this.isQuanX()) { if (this.isNeedRewrite) { opts.opts = opts.opts || {} Object.assign(opts.opts, { hints: false }) } $task.fetch(opts).then( (resp) => { const { statusCode: status, statusCode, headers, body } = resp callback(null, { status, statusCode, headers, body }, body) }, (err) => callback(err) ) } else if (this.isNode()) { this.initGotEnv(opts) this.got(opts) .on('redirect', (resp, nextOpts) => { try { if (resp.headers['set-cookie']) { const ck = resp.headers['set-cookie'].map(this.cktough.Cookie.parse).toString() if (ck) { this.ckjar.setCookieSync(ck, null) } nextOpts.cookieJar = this.ckjar } } catch (e) { this.logErr(e) } // this.ckjar.setCookieSync(resp.headers['set-cookie'].map(Cookie.parse).toString()) }) .then( (resp) => { const { statusCode: status, statusCode, headers, body } = resp callback(null, { status, statusCode, headers, body }, body) }, (err) => { const { message: error, response: resp } = err callback(error, resp, resp && resp.body) } ) } } post(opts, callback = () => {}) { // 如果指定了请求体, 但没指定`Content-Type`, 则自动生成 if (opts.body && opts.headers && !opts.headers['Content-Type']) { opts.headers['Content-Type'] = 'application/x-www-form-urlencoded' } if (opts.headers) delete opts.headers['Content-Length'] if (this.isSurge() || this.isLoon()) { if (this.isSurge() && this.isNeedRewrite) { opts.headers = opts.headers || {} Object.assign(opts.headers, { 'X-Surge-Skip-Scripting': false }) } $httpClient.post(opts, (err, resp, body) => { if (!err && resp) { resp.body = body resp.statusCode = resp.status } callback(err, resp, body) }) } else if (this.isQuanX()) { opts.method = 'POST' if (this.isNeedRewrite) { opts.opts = opts.opts || {} Object.assign(opts.opts, { hints: false }) } $task.fetch(opts).then( (resp) => { const { statusCode: status, statusCode, headers, body } = resp callback(null, { status, statusCode, headers, body }, body) }, (err) => callback(err) ) } else if (this.isNode()) { this.initGotEnv(opts) const { url, ..._opts } = opts this.got.post(url, _opts).then( (resp) => { const { statusCode: status, statusCode, headers, body } = resp callback(null, { status, statusCode, headers, body }, body) }, (err) => { const { message: error, response: resp } = err callback(error, resp, resp && resp.body) } ) } } /** * * 示例:$.time('yyyy-MM-dd qq HH:mm:ss.S') * :$.time('yyyyMMddHHmmssS') * y:年 M:月 d:日 q:季 H:时 m:分 s:秒 S:毫秒 * 其中y可选0-4位占位符、S可选0-1位占位符,其余可选0-2位占位符 * @param {*} fmt 格式化参数 * */ time(fmt) { let o = { 'M+': new Date().getMonth() + 1, 'd+': new Date().getDate(), 'H+': new Date().getHours(), 'm+': new Date().getMinutes(), 's+': new Date().getSeconds(), 'q+': Math.floor((new Date().getMonth() + 3) / 3), 'S': new Date().getMilliseconds() } if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (new Date().getFullYear() + '').substr(4 - RegExp.$1.length)) for (let k in o) if (new RegExp('(' + k + ')').test(fmt)) fmt = fmt.replace(RegExp.$1, RegExp.$1.length == 1 ? o[k] : ('00' + o[k]).substr(('' + o[k]).length)) return fmt } /** * 系统通知 * * > 通知参数: 同时支持 QuanX 和 Loon 两种格式, EnvJs根据运行环境自动转换, Surge 环境不支持多媒体通知 * * 示例: * $.msg(title, subt, desc, 'twitter://') * $.msg(title, subt, desc, { 'open-url': 'twitter://', 'media-url': 'https://github.githubassets.com/images/modules/open_graph/github-mark.png' }) * $.msg(title, subt, desc, { 'open-url': 'https://bing.com', 'media-url': 'https://github.githubassets.com/images/modules/open_graph/github-mark.png' }) * * @param {*} title 标题 * @param {*} subt 副标题 * @param {*} desc 通知详情 * @param {*} opts 通知参数 * */ msg(title = name, subt = '', desc = '', opts) { const toEnvOpts = (rawopts) => { if (!rawopts) return rawopts if (typeof rawopts === 'string') { if (this.isLoon()) return rawopts else if (this.isQuanX()) return { 'open-url': rawopts } else if (this.isSurge()) return { url: rawopts } else return undefined } else if (typeof rawopts === 'object') { if (this.isLoon()) { let openUrl = rawopts.openUrl || rawopts.url || rawopts['open-url'] let mediaUrl = rawopts.mediaUrl || rawopts['media-url'] return { openUrl, mediaUrl } } else if (this.isQuanX()) { let openUrl = rawopts['open-url'] || rawopts.url || rawopts.openUrl let mediaUrl = rawopts['media-url'] || rawopts.mediaUrl return { 'open-url': openUrl, 'media-url': mediaUrl } } else if (this.isSurge()) { let openUrl = rawopts.url || rawopts.openUrl || rawopts['open-url'] return { url: openUrl } } } else { return undefined } } if (!this.isMute) { if (this.isSurge() || this.isLoon()) { $notification.post(title, subt, desc, toEnvOpts(opts)) } else if (this.isQuanX()) { $notify(title, subt, desc, toEnvOpts(opts)) } } if (!this.isMuteLog) { let logs = ['', '==============📣系统通知📣=============='] logs.push(title) subt ? logs.push(subt) : '' desc ? logs.push(desc) : '' console.log(logs.join('\n')) this.logs = this.logs.concat(logs) } } log(...logs) { if (logs.length > 0) { this.logs = [...this.logs, ...logs] } console.log(logs.join(this.logSeparator)) } logErr(err, msg) { const isPrintSack = !this.isSurge() && !this.isQuanX() && !this.isLoon() if (!isPrintSack) { this.log('', `❗️${this.name}, 错误!`, err) } else { this.log('', `❗️${this.name}, 错误!`, err.stack) } } wait(time) { return new Promise((resolve) => setTimeout(resolve, time)) } done(val = {}) { const endTime = new Date().getTime() const costTime = (endTime - this.startTime) / 1000 this.log('', `🔔${this.name}, 结束! 🕛 ${costTime} 秒`) this.log() if (this.isSurge() || this.isQuanX() || this.isLoon()) { $done(val) } } })(name, opts) }