const config = { "PATH_IMG_V2_PRO": "https://cdn-image.gandMang.com/", "RUN_TYPE": 1, "BOTTOM_LOAD_MORE_THROTTLE": 100, "BASE_API": "https://api.gandMang.com/base", "RESALE_API": "https://api.gandMang.com/market", "RESALE_API_V2": "https://api2.gandMang.com/market/api/v2", "RESALE_API_V3": "https://api2.gandMang.com/market/api/v3", "SALE_API": "https://api.gandMang.com/read/api", "ORDER_API": "https://api.gandMang.com", "WALLET_API": "https://api.gandMang.com/api", "HXT_API": "https://api.gandMang.com/wallet", "PATH_IMG_V2": "https://cdn-image.gandMang.com/", "SAND_RECHARGE_ACTION": "https://cap.sandpay.com.cn/v4/front-electrans/ceas.elec.trans.quick.deposit.apply", "GAME_ElF_API": "https://elf.pentajam.cn/api", "GAME_ELF_URL": "https://elf.pentajam.cn", "CORECENTER_URL": "https://api.gandMang.com/corecenter" }; const lk = new ToolKit(`光芒助手`, `GandMangHelper`); const GandMangConstKey = { // -----通用助手相关的----- Token: 'lkGandMangToken', UserData: 'lkGandMangUserData', PrivCollectionBrief: 'lkGandMangPrivCollectionBrief', PrivCollectionDetail: 'lkGandMangPrivCollectionDetail', CollectionName: 'lkGandMangCollectionName', CastingId: 'lkGandMangCastingId', LimitPrice: 'lkGandMangLimitPrice', TransactionRecordId: 'lkGandMangTransactionRecordId', CollectionCateLst: 'lkGandMangCollectionCateLst', FloatPriceIntaval: 'lkGandMangFloatPriceIntaval', IsCollectionWatchLocked: 'lkIsGandMangCollectionWatchLocked', PrivWalletListInUse: 'lkGandMangPrivWalletListInUse', IsCollectionMaxNumLimit: 'lkGandMangIsCollectionMaxNumLimit', CollectionLockMaxNum: 'lkGandMangCollectionLockMaxNum', // -----合成脚本增加的----- CompositeFuncEnalbleList: 'lkGandMangCompositeFuncEnalbleList', CompositeTaskList0: 'lkGandMangCompositeTaskList0', CompositeTaskList1: 'lkGandMangCompositeTaskList1', CompositeTaskList2: 'lkGandMangCompositeTaskList2', // -----置换分解脚本增加的----- ExchangeTaskList0: 'lkGandMangExchangeTaskList0', ExchangeTaskList1: 'lkGandMangExchangeTaskList1', ExchangeTaskList2: 'lkGandMangExchangeTaskList2', // -----抢首发相关的----- NewCollectionCateLst: 'lkGandMangNewCollectionCateLst', }; let gandMangToken = lk.getVal(GandMangConstKey.Token, `Bearer eyJhbGciOiJIUzUxMiJ9.eyJzdWIiOiLlhYnoipLnlKjmiLdvenVZa2ZXYSIsImV4cCI6MTc1Mzk0MTg4MywiaWF0IjoxNzUxMzQ5ODgzfQ.cQnYrX3ycOkevg_Y-nnRIqSf5CEQzJI_AtEZKMImspcp74pskPzLTeTLtmWpI2DYQCsaz42FiqvLIc86xuOELg`); let gandMangUserAgent = `Mozilla/5.0 (iPhone; CPU iPhone OS 16_0_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.0 Mobile/15E148 Safari/604.1`; const GCommonGandMangHeads = { 'Accept': `*/*`, 'Origin': `https://gandart.art`, 'Accept-Encoding': `gzip, deflate, br`, 'Content-Type': `application/json`, 'Connection': `keep-alive`, 'Host': `gandart.art`, 'User-Agent': gandMangUserAgent, 'Authorization': gandMangToken, 'Accept-Language': `zh-CN,zh-Hans;q=0.9`, 'Referer': `https://gandart.art/` }; let retBody; if (!lk.isExecComm) { if (lk.isRequest()) { checkHandleRequest(); if (retBody) { lk.done({ body: JSON.stringify(retBody) }); } else { lk.done(); } } } function checkHandleRequest() { const url = $request.url; const path = $request.path; lk.log(`请求url=${url}#${$request.method}`); lk.log(`请求body=${lk.getRequestBody()}`); if ($request && $request.method != 'OPTIONS') { switch (path) { case '/nft/activityCollection/getAllActivity': handleAllActivityList(); break; default: if (path.match(/\/nft\/activityCollection\/detail/)) { handleActivityCollectionDetail(); } break; } } } function handleVerifyCodeLogin() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`正在获取Token`); data = JSON.parse(data); if (data.success) { let jwktoken = data.obj.token; lk.setVal(GandMangConstKey.Token, jwktoken); lk.appendNotifyInfo('🎉成功获取光予token,可以关闭相应脚本'); } else { lk.execFail(); lk.appendNotifyInfo('❌获取光予token失败,请稍后再试'); } } catch (e) { lk.execFail(); lk.appendNotifyInfo('❌获取光予token失败'); } lk.msg(''); } function handleUserInfo() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取用户信息`); data = JSON.parse(data); if (data.user) { lk.setVal(GandMangConstKey.UserData, JSON.stringify(data.user)); lk.log('成功获取光予用户信息'); lk.appendNotifyInfo('🎉成功获取光予用户信息'); } else { lk.execFail(); lk.appendNotifyInfo('❌获取光予用户信息失败,请稍后再试'); } } catch (e) { lk.execFail(); lk.appendNotifyInfo('❌获取光予用户信息失败'); } // lk.msg(''); } function handleMyOwnCollection() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取个人拥有的产品简要列表`); data = JSON.parse(data); if (data.rows) { let jsonStr = lk.getVal(GandMangConstKey.PrivCollectionBrief, '[]'); let cacheLst = JSON.parse(jsonStr); let curPage = data.page; if (curPage > data.totalPage) { curPage = data.totalPage; } if (curPage == 1) { //全部清理 cacheLst = []; } else { for (let i = cacheLst.length - 1; i >= 0; i--) { if (cacheLst[i].page == curPage) { cacheLst.splice(i, 1); } } } let pageRows = data.rows; for (let i = 0; i < pageRows.length; i++) { const ele = pageRows[i]; const info = { castingId: ele.castingId, number: ele.number, collectionName: ele.collectionName, page: curPage, } cacheLst.push(info); } jsonStr = JSON.stringify(cacheLst, null, 2); lk.setVal(GandMangConstKey.PrivCollectionBrief, jsonStr); lk.log('成功获取个人产品简要列表'); lk.appendNotifyInfo('🎉成功获取光予个人产品简要列表'); } else { lk.execFail(); lk.appendNotifyInfo('❌获取光予个人产品简要列表失败,请稍后再试'); } } catch (e) { lk.execFail(); lk.appendNotifyInfo('❌获取光予个人产品简要列表失败'); } // lk.msg(''); } function handleMyCollectionDetails() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取个人拥有的产品详情列表`); data = JSON.parse(data); if (data.rows) { let params = {}; let reqBody = lk.getRequestBody(); if (reqBody) { params = lk.parseQueryStr(reqBody); } let jsonStr = lk.getVal(GandMangConstKey.PrivCollectionDetail, '{}'); let cacheDetails = JSON.parse(jsonStr); let pageRows = data.rows; let castingId = params.castingId; if (pageRows[0]) { castingId = pageRows[0].castingId; } let cacheLst = cacheDetails[castingId] || []; let curPage = data.page; if (curPage > data.totalPage) { curPage = data.totalPage; } if (curPage == 1) { //全部清理 cacheLst = []; } else { for (let i = cacheLst.length - 1; i >= 0; i--) { if (cacheLst[i].page == curPage) { cacheLst.splice(i, 1); } } } for (let i = 0; i < pageRows.length; i++) { const ele = pageRows[i]; const info = { id: ele.id, viewSort: ele.viewSort, castingId: ele.castingId, collectionName: ele.collectionName, page: curPage, } cacheLst.push(info); } cacheDetails[castingId] = cacheLst; jsonStr = JSON.stringify(cacheDetails, null, 2); lk.setVal(GandMangConstKey.PrivCollectionDetail, jsonStr); lk.log('成功获取个人产品详情列表'); lk.appendNotifyInfo('🎉成功获取光予个人产品详情列表'); } else { lk.execFail(); lk.appendNotifyInfo('❌获取光予个人产品详情列表失败,请稍后再试'); } } catch (e) { lk.execFail(); lk.appendNotifyInfo('❌获取光予个人产品详情列表发生错误'); } // lk.msg(''); } function handleActivityCollectionDetail() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取活动详情`); let ret = JSON.parse(data); let starDate = new Date(ret.startTime); starDate = new Date(starDate.getTime() - 60 * 60 * 1000); let startTimeStr = lk.formatDate(starDate, 'yyyy-MM-dd HH:mm:ss'); ret.startTime = startTimeStr; ret.stock = ret.total; retBody = ret; } catch (error) { lk.logErr(error); } } function handleFindAllIsOpenResale() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取产品在售列表`); data = JSON.parse(data); if (data.success) { let obj = data.obj; let oldItemlst = JSON.parse(lk.getVal(GandMangConstKey.CollectionCateLst, '[]')); let pageItemlst = obj; let newLst = checkAppendCollectionItemsDiff(oldItemlst, pageItemlst); let jsonStr = JSON.stringify(newLst, null, 2); lk.setVal(GandMangConstKey.CollectionCateLst, jsonStr); lk.log(jsonStr); } } catch (e) { lk.logErr(e); lk.execFail(); } lk.msg(''); } function handleSelectRecord() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取产品收藏列表`); data = JSON.parse(data); if (data.success) { let obj = data.obj; let oldItemlst = JSON.parse(lk.getVal(GandMangConstKey.CollectionCateLst, '[]')); let pageItemlst = obj; let newLst = checkAppendCollectionItemsDiff(oldItemlst, pageItemlst); let jsonStr = JSON.stringify(newLst, null, 2); lk.setVal(GandMangConstKey.CollectionCateLst, jsonStr); lk.log(jsonStr); } } catch (e) { lk.logErr(e); lk.execFail(); } lk.msg(''); } function handleTradingMarket() { try { if ($request.headers.token) { let jwktoken = $request.headers.token; lk.setVal(GandMangConstKey.Token, jwktoken); } let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取最新Token并更新产品列表`); data = JSON.parse(data); if (data.success) { let obj = data.obj; let oldItemlst = JSON.parse(lk.getVal(GandMangConstKey.CollectionCateLst, '[]')); let pageItemlst = obj.list; let newLst = checkAppendCollectionItemsDiff(oldItemlst, pageItemlst); let jsonStr = JSON.stringify(newLst, null, 2); lk.setVal(GandMangConstKey.CollectionCateLst, jsonStr); lk.log(jsonStr); } } catch (e) { lk.logErr(e); lk.execFail(); } lk.msg(''); } function handleQuerySale() { try { if ($request.headers.token) { let jwktoken = $request.headers.token; lk.setVal(GandMangConstKey.Token, jwktoken); } let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取最新Token并更新产品列表`); data = JSON.parse(data); if (data.ok) { let obj = data.data; let oldItemlst = JSON.parse(lk.getVal(GandMangConstKey.CollectionCateLst, '[]')); let pageItemlst = obj.list; let newLst = checkAppendCollectionItemsDiff(oldItemlst, pageItemlst); let jsonStr = JSON.stringify(newLst, null, 2); lk.setVal(GandMangConstKey.CollectionCateLst, jsonStr); lk.log(jsonStr); } } catch (e) { lk.logErr(e); lk.execFail(); } lk.msg(''); } function checkFixLimitPrice(castingId, info, lastCastingId) { // 检查预设捡漏价格 let resalePrice = Number(info.resalePrice); if (lastCastingId == castingId) { let floatInterval = lk.getVal(GandMangConstKey.FloatPriceIntaval, 5); let price = Number(lk.getVal(GandMangConstKey.LimitPrice, 0)); if (resalePrice > 0 && price - floatInterval > resalePrice) { lk.setVal(GandMangConstKey.LimitPrice, info.resalePrice); } } else { if (resalePrice > 0) { resalePrice -= 10; } lk.setVal(GandMangConstKey.LimitPrice, String(resalePrice)); } } function isExistCollectionLocked() { let curCastingId = lk.getVal(GandMangConstKey.CastingId); if (!lk.isEmpty(curCastingId)) { let isLocked = lk.getVal(GandMangConstKey.IsCollectionWatchLocked); isLocked = lk.isEmpty(isLocked) ? true : JSON.parse(isLocked); return isLocked; } return false; } function handleCollectionDetailsByCastingId() { if (isExistCollectionLocked()) { return; } try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取产品详情信息`); data = JSON.parse(data); if (data.success) { let obj = data.obj; let castingId = String(obj.castingId); let lastCastingId = lk.getVal(GandMangConstKey.CastingId); if (castingId != "") { checkFixLimitPrice(castingId, obj, lastCastingId); let collectionName = obj.collectionName; lk.setVal(GandMangConstKey.CastingId, castingId); lk.setVal(GandMangConstKey.CollectionName, collectionName); lk.write(`CollectionDetails#${castingId}`, JSON.stringify(obj)); lk.log(collectionName + " castingId=" + castingId); lk.appendNotifyInfo(`🎉成功获取[${collectionName}(${castingId})]信息`); } else { let reqBody = lk.getRequestBody(); if (reqBody) { let params = lk.parseQueryStr(reqBody); castingId = params.castingId; let jsonStr = lk.read(`CollectionDetails#${castingId}`); let collectionName = `产品${castingId}`; if (jsonStr) { obj = JSON.parse(jsonStr); collectionName = obj.collectionName; checkFixLimitPrice(castingId, obj, lastCastingId); } else { let info = getCollectionMInfoById(castingId); if (info) { collectionName = info.collectionName; checkFixLimitPrice(castingId, info, lastCastingId); } else { lk.setVal(GandMangConstKey.LimitPrice, '0'); } } lk.log(collectionName + " castingId=" + obj.castingId); lk.setVal(GandMangConstKey.CastingId, castingId); lk.setVal(GandMangConstKey.CollectionName, collectionName); lk.appendNotifyInfo(`🎉成功获取[${collectionName}]信息`); } if (castingId == "") { lk.appendNotifyInfo('❌获取产品信息失败'); } } } else { lk.execFail(); lk.appendNotifyInfo('❌获取产品信息失败,请稍后再试'); } } catch (e) { lk.logErr(e); lk.execFail(); lk.appendNotifyInfo('❌获取产品信息失败'); } lk.msg(''); } function handleAllActivityList() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取活动列表`); let ret = JSON.parse(data); if (ret.code == 20000) { let list = ret.data; for (let item of list) { let starDate = new Date(item.startTime); starDate = new Date(starDate.getTime() - 60 * 60 * 1000); let startTimeStr = lk.formatDate(starDate, 'yyyy-MM-dd HH:mm:ss'); item.startTime = startTimeStr; item.stock = item.total; } } retBody = ret; } catch (e) { lk.logErr(e); lk.execFail(); } } function mergeTaskLst(latestLst, newLst, page) { for (let i = latestLst.length - 1; i >= 0; i--) { if (latestLst[i].page == page) { latestLst.splice(i, 1); } } for (let i = 0; i < newLst.length; i++) { newLst[i].page = page; latestLst.push(newLst[i]); } return latestLst; } function getCompositeTaskListByCache(status, pageSize = 10, page = 1) { try { let jsonStr = ''; if (status == 0) { jsonStr = lk.getVal(GandMangConstKey.CompositeTaskList0); } else if (status == 1) { jsonStr = lk.getVal(GandMangConstKey.CompositeTaskList1); } else if (status == 2) { jsonStr = lk.getVal(GandMangConstKey.CompositeTaskList2); } if (!lk.isEmpty(jsonStr)) { return JSON.parse(jsonStr); } } catch (error) { lk.logErr(error); } } function cacheCompositeTaskList(status, taskLst) { // 精简一下内容 let simpleLst = []; for (let i = 0; i < taskLst.length; i++) { let info = taskLst[i]; let item = { id: info.id, compositeLabel: info.compositeLabel, compositeTaskName: info.compositeTaskName, surplus: info.surplus, startTime: info.startTime, endTime: info.endTime, page: info.page, }; simpleLst.push(item); } let jsonStr = JSON.stringify(simpleLst, null, 2); if (status == 0) { lk.setVal(GandMangConstKey.CompositeTaskList0, jsonStr); } else if (status == 1) { lk.setVal(GandMangConstKey.CompositeTaskList1, jsonStr); } else if (status == 2) { lk.setVal(GandMangConstKey.CompositeTaskList2, jsonStr); } } function checkAppendCollectionItemsDiff(cLst, pageLst) { let dic = {}; for (const d of cLst) { dic[d.castingId] = d; } for (const d of pageLst) { dic[d.castingId] = Object.assign(d || {}, dic[d.castingId]); } let newLst = []; for (let k in dic) { let ele = { castingId: dic[k].castingId, collectionName: dic[k].collectionName, resalePrice: dic[k].resalePrice || '0', }; newLst.push(ele); } return newLst; } function getCollectionMInfoById(castingId) { let ret; try { let clst = JSON.parse(lk.getVal(GandMangConstKey.CollectionCateLst, "[]")); for (const d of clst) { if (d.castingId == castingId) { ret = d; break; } } } catch (error) { lk.logErr(error); } return ret; } function getCollectionNameById(castingId) { let name = '未获取'; let info = getCollectionMInfoById(castingId); if (info) { name = info.collectionName; } return name; } function handleExchangeTaskList() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取置换分解任务列表`); let ret = JSON.parse(data); let status; if (ret.rows && ret.rows.length > 0) { let retTaskLst = ret.rows; status = retTaskLst[0].status; if (ret.page == 1) { retTaskLst = mergeTaskLst([], retTaskLst, 1); cacheExchangeTaskList(status, retTaskLst); } else { let latestLst = getExchangeTaskListByCache(status); retTaskLst = mergeTaskLst(latestLst, retTaskLst, ret.page); cacheExchangeTaskList(status, retTaskLst); } } else { let reqBody = lk.getRequestBody(); if (reqBody) { params = lk.parseQueryStr(reqBody); status = params.status; cacheExchangeTaskList(status, []); } } } catch (e) { lk.logErr(e); lk.execFail(); } } function getExchangeTaskListByCache(status, pageSize = 10, page = 1) { try { let jsonStr = ''; if (status == 0) { jsonStr = lk.getVal(GandMangConstKey.ExchangeTaskList0); } else if (status == 1) { jsonStr = lk.getVal(GandMangConstKey.ExchangeTaskList1); } else if (status == 2) { jsonStr = lk.getVal(GandMangConstKey.ExchangeTaskList2); } if (!lk.isEmpty(jsonStr)) { return JSON.parse(jsonStr); } } catch (error) { lk.logErr(error); } } function cacheExchangeTaskList(status, taskLst) { // 精简一下内容 let simpleLst = []; for (let i = 0; i < taskLst.length; i++) { let info = taskLst[i]; let item = { id: info.id, compositeLabel: info.compositeLabel, compositeTaskName: info.compositeTaskName, surplus: info.surplus, startTime: info.startTime, endTime: info.endTime, page: info.page, }; simpleLst.push(item); } let jsonStr = JSON.stringify(simpleLst, null, 2); lk.setVal(GandMangConstKey.ExchangeTaskList1, jsonStr); } function handleFindAll() { try { let data = lk.getResponseBody(); if (!data) { return; } lk.log(`获取首发上新列表列表`); let ret = JSON.parse(data); if (ret.success && ret.obj && ret.obj.length > 0) { let retSeriesLst = ret.obj; cacheNftSeriesLst(retSeriesLst); } else { cacheNftSeriesLst([]); } } catch (e) { lk.logErr(e); lk.execFail(); } } function cacheNftSeriesLst(seriesLst) { // 精简一下内容 let simpleLst = []; for (let i = 0; i < seriesLst.length; i++) { let info = seriesLst[i]; let item = { id: info.id, seriesName: info.seriesName, castingCount: info.castingCount, surplus: info.number, startTime: info.startTime, endTime: info.endTime, }; simpleLst.push(item); } let jsonStr = JSON.stringify(simpleLst, null, 2); lk.setVal(GandMangConstKey.NewCollectionCateLst, jsonStr); } // https://github.com/lowking/Scripts/blob/master/util/ToolKit.min.js //---SyncByPyScript---ToolKit-start function ToolKit(t, s, e) { return new class { constructor(t, s, e) { 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 = e, this.isExecComm = !1, this.isEnableLog = this.getVal(this.prefix + "IsEnableLog" + this.id), this.isEnableLog = !!this.isEmpty(this.isEnableLog) || JSON.parse(this.isEnableLog), this.isNotifyOnlyFail = this.getVal(this.prefix + "NotifyOnlyFail" + this.id), this.isNotifyOnlyFail = !this.isEmpty(this.isNotifyOnlyFail) && JSON.parse(this.isNotifyOnlyFail), this.isEnableTgNotify = this.getVal(this.prefix + "IsEnableTgNotify" + this.id), this.isEnableTgNotify = !this.isEmpty(this.isEnableTgNotify) && JSON.parse(this.isEnableTgNotify), this.tgNotifyUrl = this.getVal(this.prefix + "TgNotifyUrl" + this.id), this.isEnableTgNotify = this.isEnableTgNotify && !this.isEmpty(this.tgNotifyUrl), 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 = this.isNode() ? { request: require("request") } : null, this.execStatus = !0, this.notifyInfo = [], this.log(this.name + ", 开始执行!"), this.initCache(), this.checkRecordRequestBody(), this.execComm() } checkRecordRequestBody() { if (this.isRequest()) { var s = $request.body; if (s) { var e = $request.path; let t = this.id + "#" + e.replace("/", "_"); t = t.replace("?", "#"), this.isQuanX() && $prefs.setValueForKey(s, t), (this.isLoon() || this.isSurge()) && $persistentStore.write(s, t), this.isNode() && this.node.fs.writeFileSync(t + ".json", s, { flag: "w" }, t => console.log(t)) } } } getRequestBody() { var t = $request.path; let s = this.id + "#" + t.replace("/", "_"); if (s = s.replace("?", "#"), this.isSurge() || this.isLoon()) return $persistentStore.read(s); if (this.isQuanX()) return $prefs.valueForKey(s); if (this.isNode()) { t = s + ".json"; if (!this.node.fs.existsSync(t)) return JSON.parse(this.node.fs.readFileSync(t)) } } initCache() { var t, s = this.getPersistKey(); this.isQuanX() && (this.cache = JSON.parse($prefs.valueForKey(s) || "{}")), (this.isLoon() || this.isSurge()) && (this.cache = JSON.parse($persistentStore.read(s) || "{}")), this.isNode() && (this.node.fs.existsSync(t = "root.json") || this.node.fs.writeFileSync(t, JSON.stringify({}), { flag: "wx" }, t => console.log(t)), this.root = {}, this.node.fs.existsSync(t = s + ".json") ? this.cache = JSON.parse(this.node.fs.readFileSync(s + ".json")) : (this.node.fs.writeFileSync(t, JSON.stringify({}), { flag: "wx" }, t => console.log(t)), this.cache = {})) } getPersistKey() { return this.id + "#privateCache" } persistCache() { var t = this.getPersistKey(), s = JSON.stringify(this.cache, null, 2); this.isQuanX() && $prefs.setValueForKey(s, t), (this.isLoon() || this.isSurge()) && $persistentStore.write(s, t), this.isNode() && (this.node.fs.writeFileSync(t + ".json", s, { flag: "w" }, t => console.log(t)), this.node.fs.writeFileSync("root.json", JSON.stringify(this.root, null, 2), { flag: "w" }, t => console.log(t))) } write(t, s) { if (this.log("SET " + s), -1 !== s.indexOf("#")) { if (s = s.substr(1), isSurge || this.isLoon()) return $persistentStore.write(t, s); if (this.isQuanX()) return $prefs.setValueForKey(t, s); this.isNode() && (this.root[s] = t) } else this.cache[s] = t; this.persistCache() } read(t) { return this.log("READ " + t), -1 !== t.indexOf("#") ? (t = t.substr(1), this.isSurge() || this.isLoon() ? $persistentStore.read(t) : this.isQuanX() ? $prefs.valueForKey(t) : this.isNode() ? this.root[t] : void 0) : this.cache[t] } delete(t) { if (this.log("DELETE " + t), -1 !== t.indexOf("#")) { if (t = t.substr(1), this.isSurge() || this.isLoon()) return $persistentStore.write(null, t); if (this.isQuanX()) return $prefs.removeValueForKey(t); this.isNode() && delete this.root[t] } else delete this.cache[t]; this.persistCache() } getRealPath(t) { var s; return this.isNode() ? ((s = process.argv.slice(1, 2)[0].split("/"))[s.length - 1] = t, s.join("/")) : t } getUrlHost(t) { return t.slice(0, t.indexOf("/", 8)) } getUrlPath(t) { var s = t.lastIndexOf("/") === t.length - 1 ? -1 : void 0; return t.slice(t.indexOf("/", 8), s) } async execComm() { if (this.isNode()) { this.comm = process.argv.slice(1); let t = !1; "p" == this.comm[1] && (this.isExecComm = !0, this.log(`开始执行指令【${this.comm[1]}】=> 发送到手机测试脚本!`), this.isEmpty(this.options) || this.isEmpty(this.options.httpApi) ? (this.log("未设置options,使用默认值"), this.isEmpty(this.options) && (this.options = {}), this.options.httpApi = "ffff@10.0.0.9:6166") : /.*?@.*?:[0-9]+/.test(this.options.httpApi) || (t = !0, this.log("❌httpApi格式错误!格式:ffff@3.3.3.18:6166"), this.done()), t || this.callApi(this.comm[2])) } } callApi(t) { let i = this.comm[0], s = (this.log(`获取【${i}】内容传给手机`), ""); this.fs = this.fs || require("fs"), this.path = this.path || require("path"); var e = this.path.resolve(i), r = this.path.resolve(process.cwd(), i), o = this.fs.existsSync(e), h = !o && this.fs.existsSync(r); if (o || h) { h = o ? e : r; try { s = this.fs.readFileSync(h) } catch (t) { s = "" } } else s = ""; o = { url: `http://${this.options.httpApi.split("@")[1]}/v1/scripting/evaluate`, headers: { "X-Key": "" + this.options.httpApi.split("@")[0] }, body: { script_text: "" + s, mock_type: "cron", timeout: !this.isEmpty(t) && 5 < t ? t : 5 }, json: !0 }; this.post(o, (t, s, e) => { this.log(`已将脚本【${i}】发给手机!`), this.done() }) } getCallerFileNameAndLine() { let s; try { throw Error("") } catch (t) { s = t } var t = s.stack.split("\n")[1]; return this.path = this.path || require("path"), `[${t.substring(t.lastIndexOf(this.path.sep) + 1, t.lastIndexOf(":"))}]` } getFunName(t) { t = t.toString(); return t = (t = t.substr("function ".length)).substr(0, t.indexOf("(")) } boxJsJsonBuilder(s, r) { if (this.isNode()) { let i = "/Users/lowking/Desktop/Scripts/lowking.boxjs.json"; if (r && r.hasOwnProperty("target_boxjs_json_path") && (i = r.target_boxjs_json_path), this.fs.existsSync(i)) if (this.isJsonObject(s) && this.isJsonObject(r)) { this.log("using node"); var o = ["settings", "keys"], h = "https://raw.githubusercontent.com/Orz-3"; let e = {}, t = "#lk{script_url}"; if (r && r.hasOwnProperty("script_url") && (t = this.isEmpty(r.script_url) ? "#lk{script_url}" : r.script_url), e.id = "" + this.prefix + this.id, e.name = this.name, e.desc_html = `⚠️使用说明
详情【点我查看】`, e.icons = [h + `/mini/master/Alpha/${this.id.toLocaleLowerCase()}.png`, h + `/mini/master/Color/${this.id.toLocaleLowerCase()}.png`], e.keys = [], e.settings = [{ id: this.prefix + "IsEnableLog" + this.id, name: "开启/关闭日志", val: !0, type: "boolean", desc: "默认开启" }, { id: this.prefix + "NotifyOnlyFail" + this.id, name: "只当执行失败才通知", val: !1, type: "boolean", desc: "默认关闭" }, { id: this.prefix + "IsEnableTgNotify" + this.id, name: "开启/关闭Telegram通知", val: !1, 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=" }], e.author = "#lk{author}", e.repo = "#lk{repo}", e.script = t + "?raw=true", !this.isEmpty(s)) for (var n in o) { var a = o[n]; if (!this.isEmpty(s[a])) { if ("settings" === a) for (let t = 0; t < s[a].length; t++) { var l = s[a][t]; for (let t = 0; t < e.settings.length; t++) { var p = e.settings[t]; l.id === p.id && e.settings.splice(t, 1) } } e[a] = e[a].concat(s[a]) } delete s[a] } if (Object.assign(e, s), this.isNode()) { this.fs = this.fs || require("fs"), this.path = this.path || require("path"); var h = this.path.resolve(this.boxJsJsonFile), c = this.path.resolve(process.cwd(), this.boxJsJsonFile), u = this.fs.existsSync(h), d = !u && this.fs.existsSync(c), f = JSON.stringify(e, null, "\t"), u = (!u && d ? this.fs.writeFileSync(c, f) : this.fs.writeFileSync(h, f), JSON.parse(this.fs.readFileSync(i))); if (u.hasOwnProperty("apps") && Array.isArray(u.apps) && 0 < u.apps.length) { d = u.apps, c = d.indexOf(d.filter(t => t.id == e.id)[0]); 0 <= c ? u.apps[c] = e : u.apps.push(e); let s = JSON.stringify(u, null, 2); if (!this.isEmpty(r)) for (const m in r) { let t = ""; r.hasOwnProperty(m) ? t = r[m] : "author" === m ? t = "@lowking" : "repo" === m && (t = "https://github.com/lowking/Scripts"), s = s.replace(`#lk{${m}}`, t) } for (var g, y = /(?:#lk\{)(.+?)(?=\})/, S = (null !== y.exec(s) && this.log("生成BoxJs还有未配置的参数,请参考https://github.com/lowking/Scripts/blob/master/util/example/ToolKitDemo.js#L17-L18传入参数:\n"), new Set); null !== (g = y.exec(s));)S.add(g[1]), s = s.replace(`#lk{${g[1]}}`, ""); S.forEach(t => { console.log(t + " ") }), this.fs.writeFileSync(i, s) } } } else this.log("构建BoxJsJson传入参数格式错误,请传入json对象") } } isJsonObject(t) { return "object" == typeof t && "[object object]" == Object.prototype.toString.call(t).toLowerCase() && !t.length } appendNotifyInfo(t, s) { 1 == s ? this.notifyInfo = t : this.notifyInfo.push(t) } prependNotifyInfo(t) { this.notifyInfo.splice(0, 0, t) } execFail() { this.execStatus = !1 } isRequest() { return "undefined" != typeof $request } isSurge() { return "undefined" != typeof $httpClient } isQuanX() { return "undefined" != typeof $task } isLoon() { return "undefined" != typeof $loon } isJSBox() { return "undefined" != typeof $app && "undefined" != typeof $http } isStash() { return "undefined" != typeof $environment && $environment["stash-version"] } isNode() { return "function" == typeof require && !this.isJSBox() } async sleep(s) { return new Promise(t => setTimeout(t, s)) } async wait(s) { return new Promise(t => setTimeout(t, s)) } async delay(s) { return new Promise(t => setTimeout(t, s)) } log(t) { this.isEnableLog && console.log("" + this.logSeparator + t) } logErr(t) { this.execStatus = !0, this.isEnableLog && (console.log("" + this.logSeparator + this.name + "执行异常:"), console.log(t), console.log("\n" + t.message)) } msg(t, s, e, i) { if ((this.isRequest() || !this.isNotifyOnlyFail || !this.execStatus) && (this.isEmpty(s) && (s = Array.isArray(this.notifyInfo) ? this.notifyInfo.join("\n") : this.notifyInfo), !this.isEmpty(s))) if (this.isEnableTgNotify) { for (var r in this.log(this.name + "Tg通知开始"), this.tgEscapeCharMapping) this.tgEscapeCharMapping.hasOwnProperty(r) && (s = s.replace(r, this.tgEscapeCharMapping[r])); this.get({ url: encodeURI(this.tgNotifyUrl + "📌" + this.name + "\n" + s) }, (t, s, e) => { this.log("Tg通知完毕") }) } else { var o = {}, h = !this.isEmpty(e), n = !this.isEmpty(i); this.isQuanX() && (h && (o["open-url"] = e), n && (o["media-url"] = i), $notify(this.name, t, s, o)), (this.isSurge() || this.isStash()) && (h && (o.url = e), $notification.post(this.name, t, s, o)), this.isNode() && this.log("⭐️" + this.name + "\n" + t + "\n" + s), this.isJSBox() && $push.schedule({ title: this.name, body: t ? t + "\n" + s : s }) } } pushWxMsg(t, s, e, i = () => { }) { s = { appToken: "AT_rTc93GQYIdMU8XLRnoJaSea8WkfhSzhX", content: s, summary: t, contentType: 1, topicIds: [], uids: ["UID_6P4B00X6Zv8U2oKC0I2R09emxtqq"], url: "", verifyPay: !1 }, e && (s.url = e), t = this.getJsonDoneHeaders(), t.Host = "wxpusher.zjiecode.com", t["Content-Type"] = "application/json;charset=UTF-8", e = { url: "https://wxpusher.zjiecode.com/api/send/message", headers: t, body: JSON.stringify(s) }; this.post(e, i) } getVal(t, s = "") { let e; return (e = this.isSurge() || this.isLoon() || this.isStash() ? $persistentStore.read(t) : this.isQuanX() ? $prefs.valueForKey(t) : this.isNode() ? (this.data = this.loadData(), process.env[t] || this.data[t]) : this.data && this.data[t] || null) || s } setVal(t, s) { return this.isSurge() || this.isLoon() || this.isStash() ? $persistentStore.write(s, t) : this.isQuanX() ? $prefs.setValueForKey(s, t) : this.isNode() ? (this.data = this.loadData(), this.data[t] = s, this.writeData(), !0) : this.data && this.data[t] || null } loadData() { if (!this.isNode()) return {}; this.fs = this.fs || require("fs"), this.path = this.path || require("path"); var t = this.path.resolve(this.dataFile), s = this.path.resolve(process.cwd(), this.dataFile), e = this.fs.existsSync(t), i = !e && this.fs.existsSync(s); if (!e && !i) return {}; i = e ? t : s; try { return JSON.parse(this.fs.readFileSync(i)) } catch (t) { return {} } } writeData() { var t, s, e, i, r; this.isNode() && (this.fs = this.fs || require("fs"), this.path = this.path || require("path"), t = this.path.resolve(this.dataFile), s = this.path.resolve(process.cwd(), this.dataFile), i = !(e = this.fs.existsSync(t)) && this.fs.existsSync(s), r = JSON.stringify(this.data), !e && i ? this.fs.writeFileSync(s, r) : this.fs.writeFileSync(t, r)) } adapterStatus(t) { return t && (t.status ? t.statusCode = t.status : t.statusCode && (t.status = t.statusCode)), t } get(t, i = () => { }) { this.isQuanX() && ((t = "string" == typeof t ? { url: t } : t).method = "GET", $task.fetch(t).then(t => { i(null, this.adapterStatus(t), t.body) }, t => i(t.error, null, null))), (this.isSurge() || this.isLoon() || this.isStash()) && $httpClient.get(t, (t, s, e) => { i(t, this.adapterStatus(s), e) }), this.isNode() && this.node.request(t, (t, s, e) => { i(t, this.adapterStatus(s), e) }), this.isJSBox() && ((t = "string" == typeof t ? { url: t } : t).header = t.headers, t.handler = function (t) { let s = t.error, e = (s = s && JSON.stringify(t.error), t.data); "object" == typeof e && (e = JSON.stringify(t.data)), i(s, this.adapterStatus(t.response), e) }, $http.get(t)) } post(t, i = () => { }) { this.isQuanX() && ((t = "string" == typeof t ? { url: t } : t).method = "POST", $task.fetch(t).then(t => { i(null, this.adapterStatus(t), t.body) }, t => i(t.error, null, null))), (this.isSurge() || this.isLoon() || this.isStash()) && $httpClient.post(t, (t, s, e) => { i(t, this.adapterStatus(s), e) }), this.isNode() && this.node.request.post(t, (t, s, e) => { i(t, this.adapterStatus(s), e) }), this.isJSBox() && ((t = "string" == typeof t ? { url: t } : t).header = t.headers, t.handler = function (t) { let s = t.error, e = (s = s && JSON.stringify(t.error), t.data); "object" == typeof e && (e = JSON.stringify(t.data)), i(s, this.adapterStatus(t.response), e) }, $http.post(t)) } put(t, i = () => { }) { this.isQuanX() && ((t = "string" == typeof t ? { url: t } : t).method = "PUT", $task.fetch(t).then(t => { i(null, this.adapterStatus(t), t.body) }, t => i(t.error, null, null))), (this.isSurge() || this.isLoon() || this.isStash()) && (t.method = "PUT", $httpClient.put(t, (t, s, e) => { i(t, this.adapterStatus(s), e) })), this.isNode() && (t.method = "PUT", this.node.request.put(t, (t, s, e) => { i(t, this.adapterStatus(s), e) })), this.isJSBox() && ((t = "string" == typeof t ? { url: t } : t).header = t.headers, t.handler = function (t) { let s = t.error, e = (s = s && JSON.stringify(t.error), t.data); "object" == typeof e && (e = JSON.stringify(t.data)), i(s, this.adapterStatus(t.response), e) }, $http.post(t)) } costTime() { let t = this.name + "执行完毕!"; this.isNode() && this.isExecComm && (t = `指令【${this.comm[1]}】执行完毕!`); var s = (new Date).getTime() - this.startTime, e = s / 1e3; this.execCount++, this.costTotalMs += s, 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(), (this.isSurge() || this.isQuanX() || this.isLoon() || this.isStash()) && $done(t) } getRequestUrl() { return $request.url } getResponseBody() { if ($response) return $response.body } isGetCookie(t) { return !("OPTIONS" == $request.method || !this.getRequestUrl().match(t)) } isEmpty(t) { return void 0 === t || null == t || "" == t || "null" == t || "undefined" == t || 0 === t.length } randomString(s) { s = s || 32; var e = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz1234567890", i = e.length, r = ""; for (let t = 0; t < s; t++)r += e.charAt(Math.floor(Math.random() * i)); return r } autoComplete(s, t, e, i, r, o, h, n, a, l) { if ((s += "").length < r) for (; s.length < r;)0 == o ? s += i : s = i + s; if (h) { let t = ""; for (var p = 0; p < n; p++)t += l; s = s.substring(0, a) + t + s.substring(n + a) } return this.toDBC(s = t + s + e) } customReplace(t, s, e, i) { try { for (var r in this.isEmpty(e) && (e = "#{"), this.isEmpty(i) && (i = "}"), s) t = t.replace("" + e + r + i, s[r]) } catch (t) { this.logErr(t) } return t } toDBC(t) { for (var s = "", e = 0; e < t.length; e++)32 == t.charCodeAt(e) ? s += String.fromCharCode(12288) : t.charCodeAt(e) < 127 && (s += String.fromCharCode(t.charCodeAt(e) + 65248)); return s } hash(t) { let s = 0, e, i; for (e = 0; e < t.length; e++)i = t.charCodeAt(e), s = (s << 5) - s + i, s |= 0; return String(s) } formatDate(t, s) { var e, i = { "M+": t.getMonth() + 1, "d+": t.getDate(), "H+": t.getHours(), "m+": t.getMinutes(), "s+": t.getSeconds(), "q+": Math.floor((t.getMonth() + 3) / 3), S: t.getMilliseconds() }; for (e in /(y+)/.test(s) && (s = s.replace(RegExp.$1, (t.getFullYear() + "").substr(4 - RegExp.$1.length))), i) new RegExp("(" + e + ")").test(s) && (s = s.replace(RegExp.$1, 1 == RegExp.$1.length ? i[e] : ("00" + i[e]).substr(("" + i[e]).length))); return s } parseDate(n, t) { let a = { y: 0, M: 1, d: 0, H: 0, h: 0, m: 0, s: 0, S: 0 }; (t = t || "yyyy-MM-dd").replace(/([^yMdHmsS]*?)(([yMdHmsS])\3*)([^yMdHmsS]*?)/g, function (t, s, e, i, r, o, h) { return n = n.replace(new RegExp(s + "(\\d{" + e.length + "})" + r), function (t, s) { return a[i] = parseInt(s), "" }), "" }), a.M--; t = new Date(a.y, a.M, a.d, a.H, a.m, a.s); return 0 !== a.S && t.setMilliseconds(a.S), t } objToQueryStr(s, e) { let i = ""; for (const r in s) { let t = s[r]; null != t && "" !== t && ("object" == typeof t ? t = JSON.stringify(t) : e && (t = encodeURIComponent(t)), i += `${r}=${t}&`) } return i = i.substring(0, i.length - 1) } parseQueryStr(t) { var s = {}, e = (t = -1 < t.indexOf("?") ? t.split("?")[1] : t).split("&"); for (let t = 0; t < e.length; t++) { var i = e[t].split("="); s[i[0]] = i[1] } return s } deepClone(t, s) { for (var e in s = s || {}, t) "object" == typeof t[e] ? (s[e] = t[e].constructor === Array ? [] : {}, this.deepClone(t[e], s[e])) : s[e] = t[e]; return s } getBaseDoneHeaders(t = {}) { return Object.assign({ "Access-Control-Allow-Origin": "*", "Access-Control-Allow-Methods": "POST,GET,OPTIONS,PUT,DELETE", "Access-Control-Allow-Headers": "Origin, X-Requested-With, Content-Type, Accept" }, t) } getHtmlDoneHeaders() { return this.getBaseDoneHeaders({ "Content-Type": "text/html;charset=UTF-8" }) } getJsonDoneHeaders() { return this.getBaseDoneHeaders({ "Content-Type": "text/json; charset=utf-8", Connection: "keep-alive" }) } shallowClone(t) { var s, e = {}; for (s in t) t.hasOwnProperty(s) && (e[s] = t[s]); return e } parseCookies(t) { let s = {}; return t && t.split(";").forEach(function (t) { t = t.split("="); s[t.shift().trim()] = decodeURI(t.join("=")) }), s } serializeCookies(t) { var s, e = []; for (s in t) { var i = t[s], i = encodeURIComponent(s) + "=" + encodeURIComponent(i); e.push(i) } return e.join("; ") } parseSetCookies(t) { const r = ["Expires", "Max-Age", "Domain", "Path", "HttpOnly", "SameSite"]; t = t.split(";"); let o = null; const h = {}, n = []; return t.forEach(t => { var s, t = t.trim(); let e = null, i = !0; if (t.includes("=") ? (s = t.split("="), e = s[0].trim(), i = s[1].trim()) : e = t, r.includes(e)) h[o] && (h[o][e] = i, h[o].attribs[e] = i); else { o = e; let s = {}; o.includes(",") && o.split(",").forEach(t => { t = t.trim(); r.includes(t) ? s[t] = !0 : o = t }), h[o] = { name: o, value: i, attribs: s }, n.push(h[o]) } }), n } base64Encode(t) { var s, e, i, r = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; let o, h, n; for (n = t.length, h = 0, o = ""; h < n;) { if (s = 255 & t.charCodeAt(h++), h == n) { o = (o += r.charAt(s >> 2)) + r.charAt((3 & s) << 4) + "=="; break } if (e = t.charCodeAt(h++), h == n) { o = (o = (o += r.charAt(s >> 2)) + r.charAt((3 & s) << 4 | (240 & e) >> 4)) + r.charAt((15 & e) << 2) + "="; break } i = t.charCodeAt(h++), o = (o = (o = (o += r.charAt(s >> 2)) + r.charAt((3 & s) << 4 | (240 & e) >> 4)) + r.charAt((15 & e) << 2 | (192 & i) >> 6)) + r.charAt(63 & i) } return o } base64Decode(t) { var s = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/="; if (/([^\s]+[^0-9a-zA-Z\+\/\=]|[^0-9a-zA-Z\+\/\=]\s+)/.test(t)) throw new Error("Invalid base64 input"); var e, i, r, o, h, n, a = t.replace(/\s/g, ""); let l = "", p = 0; for (; p < a.length;)r = s.indexOf(a.charAt(p++)), e = (15 & (o = s.indexOf(a.charAt(p++)))) << 4 | (h = s.indexOf(a.charAt(p++))) >> 2, i = (3 & h) << 6 | (n = s.indexOf(a.charAt(p++))), l += String.fromCharCode(r << 2 | o >> 4), 64 !== h && (l += String.fromCharCode(e)), 64 !== n && (l += String.fromCharCode(i)); return l = this.utf8Decode(l) } utf8Decode(s) { let t = [], e = 0, i = 0, r = 0; for (s = s.replace(/\r\n/g, "\n"); e < s.length;) { i = 255 & s.charCodeAt(e), r = 0, r = i <= 191 ? (i &= 127, 1) : i <= 223 ? (i &= 31, 2) : i <= 239 ? (i &= 15, 3) : (i &= 7, 4); for (let t = 1; t < r; ++t)i = i << 6 | 63 & s.charCodeAt(t + e); 4 === r ? (i -= 65536, t.push(String.fromCharCode(55296 | i >> 10 & 1023)), t.push(String.fromCharCode(56320 | 1023 & i))) : t.push(String.fromCharCode(i)), e += r } return t.join("") } parseJwt(t) { try { var s = t.split("."), e = s[0].replace(/-/g, "+").replace(/_/g, "/"), i = this.base64Decode(e).replace(/\0/g, ""), r = JSON.parse(i), o = s[1].replace(/-/g, "+").replace(/_/g, "/"), h = this.base64Decode(o).replace(/\0/g, ""); return { header: r, payload: JSON.parse(h), signature: s[2] } } catch (t) { return this.log(t), null } } }(t, s, e) } //---SyncByPyScript---ToolKit-end