ofpayGrab.js 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538
  1. const lk = new ToolKit(`爱购8.8`, `OfpayGrab`);
  2. const OfPayConstKey = {
  3. phone: 'lkOfPayPhone',
  4. uuid: 'lkOfPayUUID',
  5. token: 'lkOfPayAuthorization',
  6. cookie: 'lkOfPayCookie',
  7. marketId: 'lkOfPayMarketId',
  8. eventVisitorId: 'lkOfPayEventVisitorId',
  9. marketItemsData: 'lkOfPayMarketItemsData',
  10. marketBuyList: 'lkOfPayMarketBuyList',
  11. awardDiscountPrice: 'lkOfPayDiscountPrice',
  12. checkBuyRepeatEnable: 'lkOfPayBuyRepeatEnable',
  13. }
  14. let ofpayUserAgent = `Mozilla/5.0 (iPhone; CPU iPhone OS 16_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148 F-OFST elife_moblie_ios fullversion:6.0.2 BSComponentVersion:5.4 WorkStationChannel:0 isBreak:0 ICBCiPhoneBSNew 6.0.2 iphone os wkwebview:true`;
  15. let ofpayUUID = lk.getVal(OfPayConstKey.uuid,'');
  16. // jwt登录身份验证 24小时过期
  17. let ofpayAuthorization = lk.getVal(OfPayConstKey.token, '');
  18. let ofpayCookie = lk.getVal(OfPayConstKey.cookie,'');
  19. let ofpayAccountPhone = lk.getVal(OfPayConstKey.phone);
  20. let ofpayCheckBuyRepeatEnable = lk.getVal(OfPayConstKey.checkBuyRepeatEnable, true);
  21. let appVersion = 602;
  22. const CommonHost = 'market-web.ofpay.com';
  23. const GCommonHeads = {
  24. 'Host': CommonHost,
  25. 'UUID': ofpayUUID,
  26. 'Accept': `*/*`,
  27. 'Sec-Fetch-Site': 'same-origin',
  28. 'Origin': `https://market-web.ofpay.com`,
  29. 'Accept-Language': 'zh-CN,zh-Hans;q=0.9',
  30. 'Accept-Encoding': `gzip, deflate, br`,
  31. 'Sec-Fetch-Mode': 'cors',
  32. 'Content-Type': `application/json; charset=utf-8`,
  33. 'Connection': `keep-alive`,
  34. 'User-Agent': ofpayUserAgent,
  35. 'Referer': `https://www.gandart.com/`,
  36. 'Accept-Language': `zh-CN,zh-Hans;q=0.9`,
  37. 'Authorization': ofpayAuthorization,
  38. 'Cookie': ofpayCookie,
  39. 'Sec-Fetch-Dest': 'empty',
  40. 'Referer': 'https://market-web.ofpay.com/h5/union/standard/interactiveIGoChoose/index',
  41. };
  42. let marketId = lk.getVal(OfPayConstKey.marketId);
  43. let eventVisitorId = lk.getVal(OfPayConstKey.eventVisitorId);
  44. let awardWantDiscountDict = null;
  45. if (!lk.isExecComm) {
  46. if (!lk.isRequest()) {
  47. all();
  48. }
  49. }
  50. async function all() {
  51. let hasNeedSendNotify = true;
  52. if(!checkParamsExists()){
  53. lk.appendNotifyInfo(`❌缺少必要参数,请登录app采集`);
  54. }else{
  55. if(lk.isEmpty(marketId) || lk.isEmpty(eventVisitorId)){
  56. lk.appendNotifyInfo(`❌缺少次要参数,请登录app采集`);
  57. } else {
  58. let cateItems = await getMarketItems(marketId, eventVisitorId);
  59. if(cateItems){
  60. let cateCount = cateItems.length;
  61. let allBuyList = getWillMarketBuyListAll();
  62. // lk.log("#############allBuyList#############");
  63. // lk.log(JSON.stringify(allBuyList));
  64. let allRetList = [];
  65. for (let i = 0; i < cateCount; i++) {
  66. const buyList = allBuyList[i];
  67. if(!buyList){
  68. continue;
  69. }
  70. const activityData = cateItems[i];
  71. const buyRetList = await checkToBuyAll(buyList, activityData);
  72. allRetList = allRetList.concat(buyRetList);
  73. }
  74. let dismsg = '';
  75. for(let i = 0; i < allRetList.length; i++){
  76. const oneRet = allRetList[i];
  77. dismsg += `${oneRet.prizeName}${oneRet.prizeDesc}#¥${oneRet.price}\n`;
  78. }
  79. if(dismsg.length > 0){
  80. lk.appendNotifyInfo(`🎉下单成功:\n${dismsg}`);
  81. }
  82. }
  83. }
  84. }
  85. if (hasNeedSendNotify) {
  86. lk.msg(``);
  87. };
  88. lk.done();
  89. }
  90. function checkParamsExists(){
  91. if(lk.isEmpty(ofpayUUID)){
  92. return false;
  93. }
  94. if(lk.isEmpty(ofpayAuthorization)){
  95. return false;
  96. }
  97. if(lk.isEmpty(ofpayCookie)){
  98. return false;
  99. }
  100. return true;
  101. }
  102. function getWillMarketBuyListAll() {
  103. let key = `${OfPayConstKey.marketBuyList}`;
  104. let defVal = '星巴克|霸王茶姬|百果园|京东E卡|滴滴快车';
  105. let nameArr = [];
  106. let nameStr = lk.getVal(key, defVal);
  107. if (!lk.isEmpty(nameStr)) {
  108. let segments = nameStr.trim().split('|');
  109. nameArr = [];
  110. for (let vstr of segments) {
  111. if(lk.isEmpty(vstr)){
  112. nameArr.push(null);
  113. } else {
  114. const vlist = vstr.trim().split(',');
  115. nameArr.push(vlist);
  116. }
  117. }
  118. } else {
  119. nameArr = [];
  120. }
  121. return nameArr;
  122. }
  123. async function checkToBuyAll(buyList, activityData){
  124. const subActivityId = activityData.subActivityId;
  125. const awardList = activityData.awardList;
  126. let retList = [];
  127. for(let i=0; i < buyList.length; i++){
  128. const oneRet = await checkToBuyOne(subActivityId, buyList[i], awardList);
  129. if(oneRet){
  130. retList.push(oneRet);
  131. }
  132. }
  133. return retList;
  134. }
  135. async function checkToBuyOne(subActivityId, itemName, awardList){
  136. let oneRet = null;
  137. let nowString = lk.formatDate(new Date(), 'yyyyMMdd');
  138. if(ofpayCheckBuyRepeatEnable){
  139. let checkBuyRepeatKey = `lkOfPayBuyItemKey#${itemName}`;
  140. let lastBuySuccDate = lk.getVal(checkBuyRepeatKey, '');
  141. if (nowString == lastBuySuccDate) {
  142. lk.log(`商品[${itemName}]今日已抢购成功过,跳过~~`);
  143. return oneRet;
  144. }
  145. }
  146. for(let i=0; i < awardList.length; i++){
  147. const awardData = awardList[i];
  148. if(awardData.prizeName.indexOf(itemName) > -1){
  149. lk.log(`开始尝试抢购-${awardData.prizeName}${awardData.prizeDesc},价格:${awardData.price},库存:${awardData.remainStock}`);
  150. if(awardData.remainStock > 0){
  151. oneRet = await itemBuy(subActivityId, awardData);
  152. if(oneRet){
  153. // 抢购成功
  154. lk.setVal(checkBuyRepeatKey, nowString);
  155. }
  156. }else{
  157. lk.log(`库存不足,跳过~`);
  158. }
  159. break;
  160. }
  161. }
  162. return oneRet;
  163. }
  164. async function itemBuy(subActivityId, awardData){
  165. const activityId = awardData.activityId;
  166. const prizeName = awardData.prizeName;
  167. let actDataList = await getActivityItems(marketId, activityId, eventVisitorId);
  168. let theActData = await getActivityData(activityId, eventVisitorId);
  169. let desInfo = await getDesDecodeInfo(activityId, eventVisitorId);
  170. let cateActType = 'choose';
  171. if(desInfo){
  172. if(desInfo.code == '5'){
  173. cateActType = 'subChoose';
  174. }
  175. }
  176. for(let i=0; i < actDataList.length; i++){
  177. const actData = actDataList[i];
  178. // 'choose' 'subChoose' 'subPre' 'subShare'
  179. const actType = actData.type;
  180. if(actType == cateActType){
  181. const awardList = actData.awardList;
  182. const newAwardIem = checkPickItem(prizeName, awardList);
  183. if(newAwardIem){
  184. awardData = newAwardIem;
  185. break;
  186. }
  187. }
  188. }
  189. const gameAccount = ofpayAccountPhone;
  190. const thirdInfo = JSON.parse(awardData.thirdInfo);
  191. const awardId = awardData.awardId;
  192. const awardPrice = parseFloat(awardData.price);
  193. const awardFaceValue = parseFloat(thirdInfo.faceValue);
  194. awardData.faceValue = awardFaceValue;
  195. const discountPrice = getAwardExpectedDiscount(awardFaceValue, prizeName);
  196. lk.log(`商品${prizeName}(面值:${awardFaceValue})\n匹配的最终价格:${awardPrice}\n预设折扣价格:${discountPrice}#${awardId}`);
  197. if(awardPrice <= discountPrice){
  198. let payInfo = await getPayInfo(subActivityId, awardId, '', '', gameAccount, eventVisitorId);
  199. let payRet = await pay(subActivityId, eventVisitorId, payInfo.detailId);
  200. if(payRet){
  201. return awardData;
  202. }
  203. return awardData;
  204. } else {
  205. lk.log(`未到预设折扣价格${discountPrice},跳过~`);
  206. }
  207. return null;
  208. }
  209. function getAwardExpectedDiscount(price, prizeName){
  210. if(!awardWantDiscountDict){
  211. awardWantDiscountDict = {};
  212. let defVal = '星巴克#20.80|霸王茶姬#10.80|百果园#10.80|京东E卡#10.80|滴滴快车#10.80';
  213. let discountStr = lk.getVal(OfPayConstKey.awardDiscountPrice,defVal);
  214. if (!lk.isEmpty(discountStr)) {
  215. let segments = discountStr.trim().split('|');
  216. for (let vstr of segments) {
  217. if(!lk.isEmpty(vstr)){
  218. const vlist = vstr.trim().split('#');
  219. const key = vlist[0];
  220. if(vlist[1]){
  221. const price = Number(vlist[1].trim());
  222. awardWantDiscountDict[key] = price;
  223. }
  224. }
  225. }
  226. }
  227. }
  228. if(awardWantDiscountDict[prizeName] != void 0){
  229. return awardWantDiscountDict[prizeName];
  230. }
  231. return price-8.8;
  232. }
  233. function checkPickItem(prizeName, awardList){
  234. for(let i=0; i < awardList.length; i++){
  235. const awardData = awardList[i];
  236. const onePrizeName = awardData.prizeName;
  237. if(onePrizeName.indexOf('忽略') == -1 && onePrizeName.indexOf(prizeName) > -1){
  238. return awardData;
  239. }
  240. }
  241. }
  242. async function getMarketItems(marketId, eventVisitorId){
  243. return new Promise((resolve, _reject) => {
  244. try {
  245. const headers = GCommonHeads;
  246. headers.Host = 'market-web.ofpay.com';
  247. const body = ``;
  248. const url = `https://${CommonHost}/h5/union/api/interactiveIGoChoose/indexConfigRebuild?marketId=${marketId}&eventVisitorId=${eventVisitorId}`;
  249. let options = {
  250. url: url,
  251. headers: headers,
  252. body: body
  253. };
  254. lk.log(`请求市场商品列表数据`);
  255. lk.get(options, async (error, _response, data) => {
  256. let ret;
  257. try {
  258. if (error) {
  259. lk.log(`请求市场商品列表发生错误`);
  260. lk.execFail();
  261. } else {
  262. let info = JSON.parse(data);
  263. if (info.code == 'success') {
  264. lk.log(`请求市场商品列表数据成功`);
  265. ret = info.data;
  266. } else {
  267. lk.log(`请求市场商品列表数据成功,响应:${data}`);
  268. }
  269. }
  270. } catch (e) {
  271. lk.log(`请求市场商品列表发生错误`);
  272. lk.logErr(e);
  273. lk.execFail();
  274. } finally {
  275. resolve(ret);
  276. }
  277. });
  278. } catch (e) {
  279. lk.log(`请求市场商品列表发生错误`);
  280. lk.logErr(e);
  281. resolve();
  282. }
  283. });
  284. }
  285. async function getActivityItems(marketId, activityId, eventVisitorId){
  286. return new Promise((resolve, _reject) => {
  287. try {
  288. const headers = GCommonHeads;
  289. headers.Host = 'market-web.ofpay.com';
  290. const body = ``;
  291. const url = `https://${CommonHost}/h5/union/interactiveIGoChoose/marketIndexRebuild?marketId=${marketId}&activityId=${activityId}&eventVisitorId=${eventVisitorId}`;
  292. let options = {
  293. url: url,
  294. headers: headers,
  295. body: body
  296. };
  297. lk.log(`请求活动商品列表数据`);
  298. lk.get(options, async (error, _response, data) => {
  299. let ret;
  300. try {
  301. if (error) {
  302. lk.log(`请求活动商品列表发生错误`);
  303. lk.execFail();
  304. } else {
  305. let info = JSON.parse(data);
  306. if (info.code == 'success') {
  307. lk.log(`请求活动商品列表数据成功`);
  308. ret = info.data;
  309. } else {
  310. lk.log(`请求活动商品列表数据失败,响应:${data}`);
  311. }
  312. }
  313. } catch (e) {
  314. lk.log(`请求活动商品列表发生错误`);
  315. lk.logErr(e);
  316. lk.execFail();
  317. } finally {
  318. resolve(ret);
  319. }
  320. });
  321. } catch (e) {
  322. lk.log(`请求活动商品列表发生错误`);
  323. lk.logErr(e);
  324. resolve();
  325. }
  326. });
  327. }
  328. async function getActivityData(activityId, eventVisitorId){
  329. return new Promise((resolve, _reject) => {
  330. try {
  331. const headers = GCommonHeads;
  332. headers.Host = 'market-web.ofpay.com';
  333. const body = ``;
  334. const url = `https://${CommonHost}/h5/api/mobile/activity/data?activityNo=${activityId}&eventVisitorId=${eventVisitorId}`;
  335. let options = {
  336. url: url,
  337. headers: headers,
  338. body: body
  339. };
  340. lk.log(`请求活动状态数据`);
  341. lk.get(options, async (error, _response, data) => {
  342. let ret;
  343. try {
  344. if (error) {
  345. lk.log(`请求活动状态发生错误`);
  346. lk.execFail();
  347. } else {
  348. let info = JSON.parse(data);
  349. if (info.code == 'success') {
  350. lk.log(`请求活动状态数据成功`);
  351. ret = info.data;
  352. } else {
  353. lk.log(`请求活动状态数据失败,响应:${data}`);
  354. }
  355. }
  356. } catch (e) {
  357. lk.log(`请求活动状态发生错误`);
  358. lk.logErr(e);
  359. lk.execFail();
  360. } finally {
  361. resolve(ret);
  362. }
  363. });
  364. } catch (e) {
  365. lk.log(`请求活动状态发生错误`);
  366. lk.logErr(e);
  367. resolve();
  368. }
  369. });
  370. }
  371. async function getDesDecodeInfo(activityId, eventVisitorId){
  372. return new Promise((resolve, _reject) => {
  373. try {
  374. const headers = GCommonHeads;
  375. headers.Host = 'market-web.ofpay.com';
  376. const body = ``;
  377. const url = `https://${CommonHost}/h5/union/api/interactiveIGoChoose/getDesDecodeInfo?activityNo=${activityId}&eventVisitorId=${eventVisitorId}`;
  378. let options = {
  379. url: url,
  380. headers: headers,
  381. body: body
  382. };
  383. lk.log(`请求活动描述数据`);
  384. lk.get(options, async (error, _response, data) => {
  385. let ret;
  386. try {
  387. if (error) {
  388. lk.log(`请求活动描述发生错误`);
  389. lk.execFail();
  390. } else {
  391. let info = JSON.parse(data);
  392. lk.log(`请求活动描述数据成功`);
  393. lk.log(data);
  394. ret = info;
  395. }
  396. } catch (e) {
  397. lk.log(`请求活动描述发生错误`);
  398. lk.logErr(e);
  399. lk.execFail();
  400. } finally {
  401. resolve(ret);
  402. }
  403. });
  404. } catch (e) {
  405. lk.log(`请求活动描述发生错误`);
  406. lk.logErr(e);
  407. resolve();
  408. }
  409. });
  410. }
  411. async function getPayInfo(subActivityId, awardId, goodsId, invitationCode, gameAccount, eventVisitorId){
  412. return new Promise((resolve, _reject) => {
  413. try {
  414. const headers = GCommonHeads;
  415. headers.Host = 'market-web.ofpay.com';
  416. const body = ``;
  417. const url = `https://${CommonHost}/h5/union/api/draw/interactiveIGoChoose/${subActivityId}?awardId=${awardId}&goodsId=${goodsId}&invitationCode=${invitationCode}&gameAccount=${gameAccount}&eventVisitorId=${eventVisitorId}`;
  418. let options = {
  419. url: url,
  420. headers: headers,
  421. body: body
  422. };
  423. lk.log(`请求商品预支付数据`);
  424. lk.get(options, async (error, _response, data) => {
  425. let ret;
  426. try {
  427. if (error) {
  428. lk.log(`请求商品预支付数据发生错误`);
  429. lk.execFail();
  430. } else {
  431. let info = JSON.parse(data);
  432. if (info.pay) {
  433. lk.log(`请求商品预支付数据成功`);
  434. lk.log(data);
  435. ret = info;
  436. } else {
  437. lk.log(`请求商品预支付数据失败,响应:${data}`);
  438. }
  439. }
  440. } catch (e) {
  441. lk.log(`请求商品预支付数据发生错误`);
  442. lk.logErr(e);
  443. lk.execFail();
  444. } finally {
  445. resolve(ret);
  446. }
  447. });
  448. } catch (e) {
  449. lk.log(`请求商品预支付数据发生错误`);
  450. lk.logErr(e);
  451. resolve();
  452. }
  453. });
  454. }
  455. async function pay(subActivityId, eventVisitorId, detailId){
  456. return new Promise((resolve, _reject) => {
  457. try {
  458. const headers = GCommonHeads;
  459. headers.Host = 'market-web.ofpay.com';
  460. headers['Content-Type'] = 'application/json; charset=utf-8';
  461. let postData = {
  462. detailId: detailId,
  463. rechargeAccount: ofpayAccountPhone,
  464. account: ofpayAccountPhone,
  465. appVersion: appVersion,
  466. };
  467. const body = JSON.stringify(postData);
  468. const url = `https://${CommonHost}/h5/api/mobile/activity/pay/${subActivityId}?eventVisitorId=${eventVisitorId}`;
  469. let options = {
  470. url: url,
  471. headers: headers,
  472. body: body
  473. };
  474. lk.log(`请求下单信息`);
  475. lk.post(options, async (error, _response, data) => {
  476. let ret;
  477. try {
  478. if (error) {
  479. lk.log(`请求下单发生错误`);
  480. lk.execFail();
  481. } else {
  482. let info = JSON.parse(data);
  483. if (info.code == 'success') {
  484. lk.log(`请求下单成功`);
  485. lk.log(data);
  486. ret = info;
  487. } else {
  488. lk.log(`请求下单失败,响应:${data}`);
  489. }
  490. }
  491. } catch (e) {
  492. lk.log(`请求下单发生错误`);
  493. lk.logErr(e);
  494. lk.execFail();
  495. } finally {
  496. resolve(ret);
  497. }
  498. });
  499. } catch (e) {
  500. lk.log(`请求下单发生错误`);
  501. lk.logErr(e);
  502. resolve();
  503. }
  504. });
  505. }
  506. //---SyncByPyScript---ToolKit-start
  507. 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="[email protected]:6166"):/.*?@.*?:[0-9]+/.test(this.options.httpApi)||(t=!0,this.log("❌httpApi格式错误!格式:[email protected]: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=`⚠️使用说明</br>详情【<a href='${t}?raw=true'><font class='red--text'>点我查看</font></a>】`,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),u=this.path.resolve(process.cwd(),this.boxJsJsonFile),c=this.fs.existsSync(h),f=!c&&this.fs.existsSync(u),d=JSON.stringify(e,null,"\t"),c=(!c&&f?this.fs.writeFileSync(u,d):this.fs.writeFileSync(h,d),JSON.parse(this.fs.readFileSync(i)));if(c.hasOwnProperty("apps")&&Array.isArray(c.apps)&&0<c.apps.length){f=c.apps,u=f.indexOf(f.filter(t=>t.id==e.id)[0]);0<=u?c.apps[u]=e:c.apps.push(e);let s=JSON.stringify(c,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}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"})}}(t,s,e)}
  508. //---SyncByPyScript---ToolKit-end