/* Jump游戏价格监控-lowking-v1.1.4 ⚠️只测试过surge没有其他app自行测试 ************************ Surge 4.2.0+ 脚本配置(其他APP自行转换配置): ************************ [Script] # > Jump游戏价格监控 Jump游戏价格监控cookie = requires-body=0,type=http-request,pattern=https:\/\/switch\.jumpvg\.com\/jump\/app\/conf,script-path=https://raw.githubusercontent.com/lowking/Scripts/master/jump/jumpPrice.js Jump游戏价格监控 = type=cron,cronexp="0 10 0 * * ?",wake-system=1,script-path=https://raw.githubusercontent.com/lowking/Scripts/master/jump/jumpPrice.js [MITM] hostname = %APPEND% switch.jumpvg.com */ const lk = new ToolKit(`Jump游戏价格监控`, `JumpPrice`, {"httpApi": "ffff@10.0.0.19:6166"}) const domain = "https://switch.jumpvg.com" const jumpHeaderKey = 'jumpHeaderKey' const jumpDifferenceLowestPercentKey = 'jumpDifferenceLowestPercentp' const countryKey = 'jumpCountry' let header = lk.getVal(jumpHeaderKey) let differenceLowestPercent = Number(lk.getVal(jumpDifferenceLowestPercentKey, 0.15)) let country = `,${lk.getVal(countryKey, "Steam国区,日本,美国")},` if(!lk.isExecComm) { if (lk.isRequest()) { getCookie() lk.done() } else { lk.boxJsJsonBuilder({ "icons": [ "https://raw.githubusercontent.com/lowking/Scripts/master/doc/icon/jump.png", "https://raw.githubusercontent.com/lowking/Scripts/master/doc/icon/jump.png" ], "settings": [ { "id": jumpDifferenceLowestPercentKey, "name": "过滤 史低折扣-当前折扣 <= 该值的游戏", "val": 0.15, "type": "number", "desc": "写小数,默认:0.15,如:0.8-0.7<=0.15。避免一直等史低,错过差一点史低的价格" }, { "id": countryKey, "name": "监控区服", "val": "Steam国区,日本,美国", "type": "text", "desc": "要监控哪些区服,可以先运行一次看看日志。默认值:Steam国区,日本,美国" }, ], "keys": [jumpHeaderKey, jumpDifferenceLowestPercentKey, countryKey], "script_timeout": 5 }, { "script_url": "https://github.com/lowking/Scripts/blob/master/jump/jumpPrice.js", "author": "@lowking", "repo": "https://github.com/lowking/Scripts", }) all().catch((err) => { lk.logErr(err) lk.execFail() lk.msg(``, err) }).finally(() => { lk.done() }) } } function getCookie() { if (lk.isGetCookie(/\/jump\/app\/conf/)) { lk.log(`开始获取cookie`) if ($request.headers) { lk.setVal(jumpHeaderKey, JSON.stringify($request.headers)) lk.appendNotifyInfo('🎉成功获取cookie,可以关闭相应脚本') } else { lk.appendNotifyInfo("❌获取cookie失败") } } lk.msg(``) lk.done() } async function all() { if (!header) { throw "⚠️请先打开jump app获取cookie" } let headers = JSON.parse(header) await getUserInfo(headers).then(([userInfo, t]) => { if (!userInfo?.data?.userId) { throw `❌${userInfo?.msg || t + "失败"},请重新获取token` } return userInfo }).then(async (userInfo) => { return await getGamePlatforms(userInfo.data.userId, headers).then(([platforms, t]) => { if (!(platforms?.code == 0 && platforms.data.length > 0)) { throw `❌${userInfo?.msg || t + "失败"}` } return { platforms: platforms.data, userId: userInfo.data.userId } }) }).then(async ({platforms, userId}) => { for (const platform of platforms) { await dealPlatform(platform, userId, headers) } }) } function dealAllPrice(game, prices, platform) { const gameId = game.gameId const discountEndTime = prices[0].discountEndTime || "unknown" let gameNotifyKey = `jumpPriceNotify-${gameId}` let isNotify = lk.getVal(gameNotifyKey, "") != discountEndTime let info = `${platform?.platformAlias} 🎮${game?.title} ${(prices[0].price / 100).toFixed(2)}¥` let matchCount = 0 let isLastDay = false prices = prices.filter(price => { let hasIncludeJump = price.country.toLowerCase().indexOf("jump") != -1 let watching = country == ",," || country.indexOf(`,${price.country},`) != -1 lk.log(`${price.country}: ${price.leftTime}, ${hasIncludeJump}, ${watching}`) return price.leftTime && !hasIncludeJump && watching }) if (prices.length == 0) { return } for (const price of prices) { let priceCNY = (price.price / 100).toFixed(2) let priceDiscountCNY = (price.priceDiscount / 100).toFixed(2) let lowestPriceCNY = (price.lowestPrice / 100).toFixed(2) let discountPercent = (price.price - price.priceDiscount) / price.price let lowestPercent = (price.price - price.lowestPrice) / price.price if (!price.lowestPrice) { lowestPriceCNY = priceDiscountCNY lowestPercent = discountPercent } if (!isLastDay && price.leftTime.trim().indexOf("1天") == 0) { isLastDay = true } if (lowestPercent - discountPercent <= differenceLowestPercent) { matchCount++ info = `${info}\n┏${price.country} ${price.leftTime ? price.leftTime : ""}\n┣目前${priceDiscountCNY}¥(-${(discountPercent * 100).toFixed(0)}%)\n┗史低${lowestPriceCNY}¥(-${(lowestPercent * 100).toFixed(0)}%)` } } lk.log(`info: ${info}\nisNotify: ${isNotify}\nmatchCount: ${matchCount}\nisLastDay: ${isLastDay}\ndiscountEndTime: ${discountEndTime}`) // 不同活动结束时间并且符合价格条件,或者符合条件价格并且是活动最后一天才通知 if (isNotify && matchCount || isLastDay && matchCount) { lk.setVal(gameNotifyKey, discountEndTime) lk.msg(``, info) } } async function dealGames(games, platform, headers) { games = games?.data.filter(game => game?.discountOff != 0) for (const game of games) { await allPrice({...game, ...platform.moduleId}, headers).then((prices) => { dealAllPrice(game, prices, platform) }) } } async function dealPlatform(platform, userId, headers) { if (platform?.gameNum > 0 && platform?.moduleId > 0) { await getGames(userId, platform.moduleId, platform?.platformAlias, headers).then(async (games) => { await dealGames(games, platform, headers) }) } } async function getUserInfo(headers) { return new Promise((resolve, _reject) => { const t = '获取用户信息' lk.log(t) lk.get({ url: `${domain}/jump/mine/userinfo`, headers }, async (error, _response, data) => { try { if (error) { lk.execFail() lk.log(error) lk.appendNotifyInfo(`❌${t}失败,请稍后再试`) } else { data = JSON.parse(data) } } catch (e) { lk.logErr(e) lk.log(`返回数据:${data}`) lk.execFail() throw `❌${t}错误,请稍后再试` } finally { resolve([data, t]) } }) }) } async function getGamePlatforms(userId, headers) { return new Promise((resolve, _reject) => { const t = '获取游戏平台列表' lk.log(t) lk.get({ url: `${domain}/jump/favorite/count?userId=${userId}&version=3`, headers }, async (error, _response, data) => { try { if (error) { lk.execFail() lk.log(error) lk.appendNotifyInfo(`❌${t}失败,请稍后再试`) } else { data = JSON.parse(data) } } catch (e) { lk.logErr(e) lk.log(`返回数据:${data}`) lk.execFail() throw `❌${t}错误,请稍后再试` } finally { resolve([data, t]) } }) }) } function getGames(userId, moduleId, platformAlias, headers) { return new Promise((resolve, _reject) => { const t = `获取${platformAlias}游戏列表` lk.log(t) let url = { url: `${domain}/jump/favorite/appList`, headers, body: JSON.stringify({ "userId": userId, "offset": 0, "priceHigh": -1, "discount": 0, "subModuleId": 1, "moduleId": moduleId, "lowestPrice": 0, "limit": 100, "orderBy": 1 }) } lk.post(url, (error, _response, data) => { try { if (error) { lk.execFail() lk.log(error) lk.appendNotifyInfo(`❌${t}失败,请稍后再试`) } else { data = JSON.parse(data) } } catch (e) { lk.logErr(e) lk.log(`返回数据:${data}`) lk.execFail() throw `❌${t}错误,请稍后再试` } finally { resolve(data) } }) }) } async function gameDetail(game, headers) { return new Promise((resolve, _reject) => { const t = `获取[${game?.title}]游戏详情` lk.log(t) lk.post({ url: `${domain}/jump/game/detail?clickFrom=-1&gameId=${game.gameIdNew}&id=${game.gameId}&path=&platform=4&version=3`, headers, }, async (error, _response, data) => { try { if (error) { lk.execFail() lk.log(error) lk.appendNotifyInfo(`❌${t}失败,请稍后再试`) } else { data = JSON.parse(data) } } catch (e) { lk.logErr(e) lk.log(`返回数据:${data}`) lk.execFail() throw `❌${t}错误,请稍后再试` } finally { resolve(data?.data) } }) }) } function allPrice(game, headers) { return new Promise((resolve, _reject) => { const t = `获取[${game?.title}]游戏所有价格-${game.gameId}-${game.moduleId}` lk.log(t) lk.post({ url: `${domain}/jump/price/getAllPriceByGame?id=${game.gameId}&platform=${game.moduleId}`, headers, }, (error, _response, data) => { try { if (error) { lk.execFail() lk.log(error) lk.appendNotifyInfo(`❌${t}失败,请稍后再试`) } else { data = JSON.parse(data) } } catch (e) { lk.logErr(e) lk.log(`返回数据:${data}`) lk.execFail() throw `❌${t}错误,请稍后再试` } finally { resolve(data?.data?.prices) } }) }) } //ToolKit-start function ToolKit(t,s,i){return new class{constructor(t,s,i){this.tgEscapeCharMapping={"&":"&","#":"#"};this.userAgent=`Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_6) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/12.0.2 Safari/605.1.15`;this.prefix=`lk`;this.name=t;this.id=s;this.data=null;this.dataFile=this.getRealPath(`${this.prefix}${this.id}.dat`);this.boxJsJsonFile=this.getRealPath(`${this.prefix}${this.id}.boxjs.json`);this.options=i;this.isExecComm=false;this.isEnableLog=this.getVal(`${this.prefix}IsEnableLog${this.id}`);this.isEnableLog=this.isEmpty(this.isEnableLog)?true:JSON.parse(this.isEnableLog);this.isNotifyOnlyFail=this.getVal(`${this.prefix}NotifyOnlyFail${this.id}`);this.isNotifyOnlyFail=this.isEmpty(this.isNotifyOnlyFail)?false:JSON.parse(this.isNotifyOnlyFail);this.isEnableTgNotify=this.getVal(`${this.prefix}IsEnableTgNotify${this.id}`);this.isEnableTgNotify=this.isEmpty(this.isEnableTgNotify)?false:JSON.parse(this.isEnableTgNotify);this.tgNotifyUrl=this.getVal(`${this.prefix}TgNotifyUrl${this.id}`);this.isEnableTgNotify=this.isEnableTgNotify?!this.isEmpty(this.tgNotifyUrl):this.isEnableTgNotify;this.costTotalStringKey=`${this.prefix}CostTotalString${this.id}`;this.costTotalString=this.getVal(this.costTotalStringKey);this.costTotalString=this.isEmpty(this.costTotalString)?`0,0`:this.costTotalString.replace('"',"");this.costTotalMs=this.costTotalString.split(",")[0];this.execCount=this.costTotalString.split(",")[1];this.costTotalMs=this.isEmpty(this.costTotalMs)?0:parseInt(this.costTotalMs);this.execCount=this.isEmpty(this.execCount)?0:parseInt(this.execCount);this.logSeparator="\n██";this.now=new Date;this.startTime=this.now.getTime();this.node=(()=>{if(this.isNode()){const t=require("request");return{request:t}}else{return null}})();this.execStatus=true;this.notifyInfo=[];this.boxjsCurSessionKey="chavy_boxjs_cur_sessions";this.boxjsSessionsKey="chavy_boxjs_sessions";this.log(`${this.name}, 开始执行!`);this.execComm()}getRealPath(t){if(this.isNode()){let s=process.argv.slice(1,2)[0].split("/");s[s.length-1]=t;return s.join("/")}return t}async execComm(){if(!this.isNode()){return}this.comm=process.argv.slice(1);if(this.comm[1]!="p"){return}let t=false;this.isExecComm=true;this.log(`开始执行指令【${this.comm[1]}】=> 发送到其他终端测试脚本!`);if(this.isEmpty(this.options)||this.isEmpty(this.options.httpApi)){this.log(`未设置options,使用默认值`);if(this.isEmpty(this.options)){this.options={}}this.options.httpApi=`ffff@10.0.0.19:6166`}else{if(!/.*?@.*?:[0-9]+/.test(this.options.httpApi)){t=true;this.log(`❌httpApi格式错误!格式:ffff@3.3.3.18:6166`);this.done()}}if(!t){this.callApi(this.comm[2])}}callApi(t){let s=this.comm[0];let i=this.options.httpApi.split("@")[1];this.log(`获取【${s}】内容传给【${i}】`);let e="";this.fs=this.fs?this.fs:require("fs");this.path=this.path?this.path:require("path");const o=this.path.resolve(s);const h=this.path.resolve(process.cwd(),s);const r=this.fs.existsSync(o);const n=!r&&this.fs.existsSync(h);if(r||n){const t=r?o:h;try{e=this.fs.readFileSync(t)}catch(t){e=""}}else{e=""}let a={url:`http://${i}/v1/scripting/evaluate`,headers:{"X-Key":`${this.options.httpApi.split("@")[0]}`},body:{script_text:`${e}`,mock_type:"cron",timeout:!this.isEmpty(t)&&t>5?t:5},json:true};this.post(a,(t,e,o)=>{this.log(`已将脚本【${s}】发给【${i}】`);this.done()})}boxJsJsonBuilder(t,s){if(!this.isNode()){return}if(!this.isJsonObject(t)||!this.isJsonObject(s)){this.log("构建BoxJsJson传入参数格式错误,请传入json对象");return}let i="/Users/lowking/Desktop/Scripts/lowking.boxjs.json";if(s&&s.hasOwnProperty("target_boxjs_json_path")){i=s["target_boxjs_json_path"]}if(!this.fs.existsSync(i)){return}this.log("using node");let e=["settings","keys"];const o="https://raw.githubusercontent.com/Orz-3";let h={};let r="#lk{script_url}";if(s&&s.hasOwnProperty("script_url")){r=this.isEmpty(s["script_url"])?"#lk{script_url}":s["script_url"]}h.id=`${this.prefix}${this.id}`;h.name=this.name;h.desc_html=`⚠️使用说明
详情【点我查看】`;h.icons=[`${o}/mini/master/Alpha/${this.id.toLocaleLowerCase()}.png`,`${o}/mini/master/Color/${this.id.toLocaleLowerCase()}.png`];h.keys=[];h.settings=[{id:`${this.prefix}IsEnableLog${this.id}`,name:"开启/关闭日志",val:true,type:"boolean",desc:"默认开启"},{id:`${this.prefix}NotifyOnlyFail${this.id}`,name:"只当执行失败才通知",val:false,type:"boolean",desc:"默认关闭"},{id:`${this.prefix}IsEnableTgNotify${this.id}`,name:"开启/关闭Telegram通知",val:false,type:"boolean",desc:"默认关闭"},{id:`${this.prefix}TgNotifyUrl${this.id}`,name:"Telegram通知地址",val:"",type:"text",desc:"Tg的通知地址,如:https://api.telegram.org/bot-token/sendMessage?chat_id=-100140&parse_mode=Markdown&text="}];h.author="#lk{author}";h.repo="#lk{repo}";h.script=`${r}?raw=true`;if(!this.isEmpty(t)){for(let s of e){if(this.isEmpty(t[s])){break}if(s==="settings"){for(let i=0;i0)){return}let c=u.apps;let g=c.indexOf(c.filter(t=>{return t.id==h.id})[0]);if(g>=0){u.apps[g]=h}else{u.apps.push(h)}let d=JSON.stringify(u,null,2);if(!this.isEmpty(s)){for(const t in s){let i=s[t];if(!i){switch(t){case"author":i="@lowking";case"repo":i="https://github.com/lowking/Scripts";default:continue}}d=d.replace(`#lk{${t}}`,i)}}const y=/(?:#lk\{)(.+?)(?=\})/;let S=y.exec(d);if(S!==null){this.log(`生成BoxJs还有未配置的参数,请参考https://github.com/lowking/Scripts/blob/master/util/example/ToolKitDemo.js#L17-L19传入参数:`)}let m=new Set;while((S=y.exec(d))!==null){m.add(S[1]);d=d.replace(`#lk{${S[1]}}`,``)}m.forEach(t=>{console.log(`${t} `)});this.fs.writeFileSync(i,d)}isJsonObject(t){return typeof t=="object"&&Object.prototype.toString.call(t).toLowerCase()=="[object object]"&&!t.length}appendNotifyInfo(t,s){if(s==1){this.notifyInfo=t}else{this.notifyInfo.push(t)}}prependNotifyInfo(t){this.notifyInfo.splice(0,0,t)}execFail(){this.execStatus=false}isRequest(){return typeof $request!="undefined"}isSurge(){return typeof $httpClient!="undefined"}isQuanX(){return typeof $task!="undefined"}isLoon(){return typeof $loon!="undefined"}isJSBox(){return typeof $app!="undefined"&&typeof $http!="undefined"}isStash(){return"undefined"!==typeof $environment&&$environment["stash-version"]}isNode(){return typeof require=="function"&&!this.isJSBox()}sleep(t){return new Promise(s=>setTimeout(s,t))}log(t){if(this.isEnableLog)console.log(`${this.logSeparator}${t}`)}logErr(t){this.execStatus=true;if(this.isEnableLog){console.log(`${this.logSeparator}${this.name}执行异常:`);console.log(t);if(!t.message){return}console.log(`\n${t.message}`)}}msg(t,s,i,e,o){if(!this.isRequest()&&this.isNotifyOnlyFail&&this.execStatus){return}if(this.isEmpty(s)){if(Array.isArray(this.notifyInfo)){s=this.notifyInfo.join("\n")}else{s=this.notifyInfo}}if(this.isEmpty(s)){return}if(this.isEnableTgNotify){this.log(`${this.name}Tg通知开始`);for(let t in this.tgEscapeCharMapping){if(!this.tgEscapeCharMapping.hasOwnProperty(t)){continue}s=s.replace(t,this.tgEscapeCharMapping[t])}this.get({url:encodeURI(`${this.tgNotifyUrl}📌${this.name}\n${s}`)},(t,s,i)=>{this.log(`Tg通知完毕`)})}else{let h={};const r=!this.isEmpty(i);const n=!this.isEmpty(e);const a=!this.isEmpty(o);if(this.isSurge()||this.isLoon()||this.isStash()){if(r){h["url"]=i;h["action"]="open-url"}if(a){h["text"]=o;h["action"]="clipboard"}if(n)h["media-url"]=e;$notification.post(this.name,t,s,h)}else if(this.isQuanX()){if(r)h["open-url"]=i;if(n)h["media-url"]=e;$notify(this.name,t,s,h)}else if(this.isNode()){this.log("⭐️"+this.name+"\n"+t+"\n"+s)}else if(this.isJSBox()){$push.schedule({title:this.name,body:t?t+"\n"+s:s})}}}getVal(t,s=""){let i;if(this.isSurge()||this.isLoon()||this.isStash()){i=$persistentStore.read(t)}else if(this.isQuanX()){i=$prefs.valueForKey(t)}else if(this.isNode()){this.data=this.loadData();i=process.env[t]||this.data[t]}else{i=this.data&&this.data[t]||null}return!i?s:i}updateBoxjsSessions(t,s){if(t==this.boxjsSessionsKey){return}const i=`${this.prefix}${this.id}`;let e=JSON.parse(this.getVal(this.boxjsCurSessionKey,"{}"));if(!e.hasOwnProperty(i)){return}let o=e[i];let h=JSON.parse(this.getVal(this.boxjsSessionsKey,"[]"));if(h.length==0){return}let r=[];h.forEach(t=>{if(t.id==o){r=t.datas}});if(r.length==0){return}let n=false;r.forEach(i=>{if(i.key==t){i.val=s;n=true}});if(!n){r.push({key:t,val:s})}h.forEach(t=>{if(t.id==o){t.datas=r}});this.setVal(this.boxjsSessionsKey,JSON.stringify(h))}setVal(t,s){if(this.isSurge()||this.isLoon()||this.isStash()){this.updateBoxjsSessions(t,s);return $persistentStore.write(s,t)}else if(this.isQuanX()){this.updateBoxjsSessions(t,s);return $prefs.setValueForKey(s,t)}else if(this.isNode()){this.data=this.loadData();this.data[t]=s;this.writeData();return true}else{return this.data&&this.data[t]||null}}loadData(){if(!this.isNode()){return{}}this.fs=this.fs?this.fs:require("fs");this.path=this.path?this.path:require("path");const t=this.path.resolve(this.dataFile);const s=this.path.resolve(process.cwd(),this.dataFile);const i=this.fs.existsSync(t);const e=!i&&this.fs.existsSync(s);if(i||e){const e=i?t:s;try{return JSON.parse(this.fs.readFileSync(e))}catch(t){return{}}}else{return{}}}writeData(){if(!this.isNode()){return}this.fs=this.fs?this.fs:require("fs");this.path=this.path?this.path:require("path");const t=this.path.resolve(this.dataFile);const s=this.path.resolve(process.cwd(),this.dataFile);const i=this.fs.existsSync(t);const e=!i&&this.fs.existsSync(s);const o=JSON.stringify(this.data);if(i){this.fs.writeFileSync(t,o)}else if(e){this.fs.writeFileSync(s,o)}else{this.fs.writeFileSync(t,o)}}adapterStatus(t){if(t){if(t.status){t["statusCode"]=t.status}else if(t.statusCode){t["status"]=t.statusCode}}return t}get(t,s=(()=>{})){if(this.isSurge()||this.isLoon()||this.isStash()){$httpClient.get(t,(t,i,e)=>{s(t,this.adapterStatus(i),e)})}else if(this.isQuanX()){if(typeof t=="string")t={url:t};t["method"]="GET";$task.fetch(t).then(t=>{s(null,this.adapterStatus(t),t.body)},t=>s(t.error,null,null))}else if(this.isNode()){this.node.request(t,(t,i,e)=>{s(t,this.adapterStatus(i),e)})}else if(this.isJSBox()){if(typeof t=="string")t={url:t};t["header"]=t["headers"];t["handler"]=function(t){let i=t.error;if(i)i=JSON.stringify(t.error);let e=t.data;if(typeof e=="object")e=JSON.stringify(t.data);s(i,this.adapterStatus(t.response),e)};$http.get(t)}}post(t,s=(()=>{})){if(this.isSurge()||this.isLoon()||this.isStash()){$httpClient.post(t,(t,i,e)=>{s(t,this.adapterStatus(i),e)})}else if(this.isQuanX()){if(typeof t=="string")t={url:t};t["method"]="POST";$task.fetch(t).then(t=>{s(null,this.adapterStatus(t),t.body)},t=>s(t.error,null,null))}else if(this.isNode()){this.node.request.post(t,(t,i,e)=>{s(t,this.adapterStatus(i),e)})}else if(this.isJSBox()){if(typeof t=="string")t={url:t};t["header"]=t["headers"];t["handler"]=function(t){let i=t.error;if(i)i=JSON.stringify(t.error);let e=t.data;if(typeof e=="object")e=JSON.stringify(t.data);s(i,this.adapterStatus(t.response),e)};$http.post(t)}}put(t,s=(()=>{})){if(this.isSurge()||this.isLoon()||this.isStash()){t.method="PUT";$httpClient.put(t,(t,i,e)=>{s(t,this.adapterStatus(i),e)})}else if(this.isQuanX()){if(typeof t=="string")t={url:t};t["method"]="PUT";$task.fetch(t).then(t=>{s(null,this.adapterStatus(t),t.body)},t=>s(t.error,null,null))}else if(this.isNode()){t.method="PUT";this.node.request.put(t,(t,i,e)=>{s(t,this.adapterStatus(i),e)})}else if(this.isJSBox()){if(typeof t=="string")t={url:t};t["header"]=t["headers"];t["handler"]=function(t){let i=t.error;if(i)i=JSON.stringify(t.error);let e=t.data;if(typeof e=="object")e=JSON.stringify(t.data);s(i,this.adapterStatus(t.response),e)};$http.post(t)}}costTime(){let t=`${this.name}执行完毕!`;if(this.isNode()&&this.isExecComm){t=`指令【${this.comm[1]}】执行完毕!`}const s=(new Date).getTime();const i=s-this.startTime;const e=i/1e3;this.execCount++;this.costTotalMs+=i;this.log(`${t}耗时【${e}】秒\n总共执行【${this.execCount}】次,平均耗时【${(this.costTotalMs/this.execCount/1e3).toFixed(4)}】秒`);this.setVal(this.costTotalStringKey,JSON.stringify(`${this.costTotalMs},${this.execCount}`))}done(t={}){this.costTime();if(this.isSurge()||this.isQuanX()||this.isLoon()||this.isStash()){$done(t)}}getRequestUrl(){return $request.url}getResponseBody(){return $response.body}isGetCookie(t){return!!($request.method!="OPTIONS"&&this.getRequestUrl().match(t))}isEmpty(t){return typeof t=="undefined"||t==null||t==""||t=="null"||t=="undefined"||t.length===0}randomString(t,s="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz1234567890"){t=t||32;let i=s.length;let e="";for(let o=0;o