huluwaHelper.js 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590
  1. /******************************
  2. > 葫芦娃预约 v1.06
  3. cron: 30 8 * * *
  4. const $ = new Env("葫芦娃预约");
  5. 自行抓包把token(一般在请求头里)填到变量中, 多账号用换行隔开(可自定义)
  6. 环境变量 XLTH_COOKIE 新联惠购
  7. 环境变量 GLYP_COOKIE 贵旅优品
  8. 环境变量 KGLG_COOKIE 空港乐购
  9. 环境变量 HLQG_COOKIE 航旅黔购
  10. 环境变量 ZHCS_COOKIE 遵行出山
  11. 环境变量 GYQP_COOKIE 贵盐黔品
  12. 环境变量 LLSC_COOKIE 乐旅商城
  13. 环境变量 YLQX_COOKIE 驿路黔寻
  14. *******************************
  15. [rewrite_local]
  16. ^http[s]?:\/\/gw.huiqunchina.com url script-request-body https://git.jojo21.top/shawenguan/Quantumult-X/raw/master/Scripts/huluwa/huluwaHelper.js
  17. ^http[s]?:\/\/gw.huiqunchina.com url script-response-body https://git.jojo21.top/shawenguan/Quantumult-X/raw/master/Scripts/huluwa/huluwaHelper.js
  18. [MITM]
  19. hostname=gw.huiqunchina.com
  20. ********************************/
  21. const lk = new ToolKit(`葫芦娃预约助手`, `huluwaHelper`);
  22. const SPLIT = "\n"; // 分割符(可自定义)
  23. const axios = require('axios');
  24. const crypto = require('crypto');
  25. const XLTH_APPID = 'wxded2e7e6d60ac09d'; // 新联惠购
  26. const GLYP_APPID = 'wx61549642d715f361'; // 贵旅优品
  27. const KGLG_APPID = 'wx613ba8ea6a002aa8'; // 空港乐购
  28. const HLQG_APPID = 'wx936aa5357931e226'; // 航旅黔购
  29. const ZHCS_APPID = 'wx624149b74233c99a'; // 遵航出山
  30. const GYQP_APPID = 'wx5508e31ffe9366b8'; // 贵盐黔品
  31. const LLSC_APPID = 'wx821fb4d8604ed4d6'; // 乐旅商城
  32. const YLQX_APPID = 'wxee0ce83ab4b26f9c'; // 驿路黔寻
  33. const APPID_TOKEN_STORE_KEYS = {
  34. ['wxded2e7e6d60ac09d'] : 'XLTH_COOKIE',
  35. ['wx61549642d715f361'] : 'GLYP_COOKIE',
  36. ['wx613ba8ea6a002aa8'] : 'KGLG_COOKIE',
  37. ['wx936aa5357931e226'] : 'HLQG_COOKIE',
  38. ['wx624149b74233c99a'] : 'ZHCS_COOKIE',
  39. ['wx5508e31ffe9366b8'] : 'GYQP_COOKIE',
  40. ['wx821fb4d8604ed4d6'] : 'LLSC_COOKIE',
  41. ['wxee0ce83ab4b26f9c'] : 'YLQX_COOKIE',
  42. };
  43. const REVER_APPID_TOKEN_STORE_KEYS = {};
  44. for(let key in APPID_TOKEN_STORE_KEYS){
  45. let value = APPID_TOKEN_STORE_KEYS[key];
  46. REVER_APPID_TOKEN_STORE_KEYS[value] = value;
  47. }
  48. const HOST = 'https://gw.huiqunchina.com';
  49. const AK = '00670fb03584fbf44dd6b136e534f495';
  50. const SK = '0d65f24dbe2bc1ede3c3ceeb96ef71bb';
  51. let sendMessage = [];
  52. let retBody;
  53. if (!lk.isExecComm) {
  54. if (lk.isRequest()) {
  55. checkHandleRequest();
  56. lk.msg('');
  57. if (retBody) {
  58. lk.done({ body: JSON.stringify(retBody) });
  59. } else {
  60. lk.done();
  61. }
  62. }else{
  63. main();
  64. }
  65. }
  66. function checkHandleRequest() {
  67. const url = $request.url;
  68. const path = $request.path;
  69. lk.log(`请求url=${url}#${$request.method}`);
  70. lk.log(`请求body=${lk.getRequestBody()}`);
  71. if ($request && $request.method != 'OPTIONS') {
  72. if(path.match(/\/front-manager\/api\/login\/wxMiniLogin/)){
  73. handleWxMiniLogin();
  74. } else if(path.match(/\/front-manager\/api\/get\/channelId/)){
  75. handleChannelId();
  76. } else if(path.match(/\/front-manager\/api\/customer\/queryById\/token/)){
  77. handleUserInfo();
  78. }
  79. }
  80. }
  81. function handleWxMiniLogin(){
  82. let reqParams = {};
  83. let reqBody = lk.getRequestBody();
  84. if (reqBody) {
  85. reqParams = JSON.parse(reqBody);
  86. }
  87. let rspBody = lk.getResponseBody();
  88. if (!rspBody) {
  89. return;
  90. }
  91. let appId = reqParams.appId;
  92. let rspParams = JSON.parse(rspBody);
  93. if(rspParams.code == '10000'){
  94. let rspData = rspParams.data;
  95. let token = rspData.token;
  96. let key = `huluwa#${appId}`;
  97. lk.setVal(key, token);
  98. key = APPID_TOKEN_STORE_KEYS[appId];
  99. if(key){
  100. key = `huluwa#${key}`;
  101. lk.setVal(key, token);
  102. }
  103. }
  104. }
  105. function handleChannelId(){
  106. let headers = $request.headers;
  107. if(!headers['X-access-token']){
  108. return;
  109. }
  110. let reqParams = {};
  111. let reqBody = lk.getRequestBody();
  112. if (reqBody) {
  113. reqParams = JSON.parse(reqBody);
  114. }
  115. let appId = reqParams.appId;
  116. let token = headers['X-access-token'];
  117. let key = `huluwa#${appId}`;
  118. lk.setVal(key, token);
  119. key = APPID_TOKEN_STORE_KEYS[appId];
  120. if(key){
  121. key = `huluwa#${key}`;
  122. lk.setVal(key, token);
  123. }
  124. let rspParams = JSON.parse(rspBody);
  125. if(rspParams.code == '10000'){
  126. let channelId = rspParams.data;
  127. }
  128. }
  129. function handleUserInfo(){
  130. }
  131. function getCookieByAppId(appId){
  132. let key = `huluwa#${appId}`;
  133. let value = lk.getVal(key);
  134. if(!lk.isEmpty(value)){
  135. return value;
  136. }
  137. key = APPID_TOKEN_STORE_KEYS[appId];
  138. if(key){
  139. key = `huluwa#${key}`;
  140. return lk.getVal(key);
  141. }
  142. }
  143. async function main() {
  144. const XLTH_COOKIE_ARR = getCookieByAppId(XLTH_APPID); // 新联惠购
  145. const GLYP_COOKIE_ARR = getCookieByAppId(GLYP_APPID); // 贵旅优品
  146. const KGLG_COOKIE_ARR = getCookieByAppId(KGLG_APPID); // 空港乐购
  147. const HLQG_COOKIE_ARR = getCookieByAppId(HLQG_APPID); // 航旅黔购
  148. const ZHCS_COOKIE_ARR = getCookieByAppId(ZHCS_APPID); // 遵行出山
  149. const GYQP_COOKIE_ARR = getCookieByAppId(GYQP_APPID); // 贵盐黔品
  150. const LLSC_COOKIE_ARR = getCookieByAppId(LLSC_APPID); // 乐旅商城
  151. const YLQX_COOKIE_ARR = getCookieByAppId(YLQX_APPID); // 驿路黔寻
  152. if (XLTH_COOKIE_ARR) {
  153. lk.log('新联惠购预约开始');
  154. sendMessage.push('新联惠购预约开始');
  155. for (let [index, item] of XLTH_COOKIE_ARR.split(SPLIT).entries()) {
  156. lk.log(`----第${index + 1}个号----`);
  157. sendMessage.push(`----第${index + 1}个号----`);
  158. await autoSubmit(XLTH_APPID, item);
  159. await delay(1000);
  160. }
  161. lk.log('新联惠购预约结束\n');
  162. sendMessage.push('新联惠购预约结束\n');
  163. }
  164. if (GLYP_COOKIE_ARR) {
  165. lk.log('贵旅优品预约开始');
  166. sendMessage.push('贵旅优品预约开始');
  167. for (let [index, item] of GLYP_COOKIE_ARR.split(SPLIT).entries()) {
  168. lk.log(`----第${index + 1}个号----`);
  169. sendMessage.push(`----第${index + 1}个号----`);
  170. await autoSubmit(GLYP_APPID, item);
  171. await delay(1000);
  172. }
  173. lk.log('贵旅优品预约结束\n');
  174. sendMessage.push('贵旅优品预约结束\n');
  175. }
  176. if (KGLG_COOKIE_ARR) {
  177. lk.log('空港乐购预约开始');
  178. sendMessage.push('新联惠购预约开始');
  179. for (let [index, item] of KGLG_COOKIE_ARR.split(SPLIT).entries()) {
  180. lk.log(`----第${index + 1}个号----`);
  181. sendMessage.push(`----第${index + 1}个号----`);
  182. await autoSubmit(KGLG_APPID, item);
  183. await delay(1000);
  184. }
  185. lk.log('空港乐购预约结束\n');
  186. sendMessage.push('空港乐购预约结束\n');
  187. }
  188. if (HLQG_COOKIE_ARR) {
  189. lk.log('航旅黔购预约开始');
  190. sendMessage.push('新联惠购预约开始');
  191. for (let [index, item] of HLQG_COOKIE_ARR.split(SPLIT).entries()) {
  192. lk.log(`----第${index + 1}个号----`);
  193. sendMessage.push(`----第${index + 1}个号----`);
  194. await autoSubmit(HLQG_APPID, item);
  195. await delay(1000);
  196. }
  197. lk.log('航旅黔购预约结束\n');
  198. sendMessage.push('航旅黔购预约结束\n');
  199. }
  200. if (ZHCS_COOKIE_ARR) {
  201. lk.log('遵行出山预约开始');
  202. sendMessage.push('新联惠购预约开始');
  203. for (let [index, item] of ZHCS_COOKIE_ARR.split(SPLIT).entries()) {
  204. lk.log(`----第${index + 1}个号----`);
  205. sendMessage.push(`----第${index + 1}个号----`);
  206. await autoSubmit(ZHCS_APPID, item);
  207. await delay(1000);
  208. }
  209. lk.log('遵行出山预约结束\n');
  210. sendMessage.push('遵行出山预约结束\n');
  211. }
  212. if (GYQP_COOKIE_ARR) {
  213. lk.log('贵盐黔品预约开始');
  214. sendMessage.push('贵盐黔品预约开始');
  215. for (let [index, item] of GYQP_COOKIE_ARR.split(SPLIT).entries()) {
  216. lk.log(`----第${index + 1}个号----`);
  217. sendMessage.push(`----第${index + 1}个号----`);
  218. await autoSubmit(GYQP_APPID, item);
  219. await delay(1000);
  220. }
  221. lk.log('贵盐黔品预约结束\n');
  222. sendMessage.push('贵盐黔品预约结束\n');
  223. }
  224. if (LLSC_COOKIE_ARR) {
  225. lk.log('乐旅商城预约开始');
  226. sendMessage.push('乐旅商城预约开始');
  227. for (let [index, item] of LLSC_COOKIE_ARR.split(SPLIT).entries()) {
  228. lk.log(`----第${index + 1}个号----`);
  229. sendMessage.push(`----第${index + 1}个号----`);
  230. await autoSubmit(LLSC_APPID, item);
  231. await delay(1000);
  232. }
  233. lk.log('乐旅商城预约结束\n');
  234. sendMessage.push('乐旅商城预约结束\n');
  235. }
  236. if (YLQX_COOKIE_ARR) {
  237. lk.log('驿路黔寻预约开始');
  238. sendMessage.push('驿路黔寻预约开始');
  239. for (let [index, item] of YLQX_COOKIE_ARR.split(SPLIT).entries()) {
  240. lk.log(`----第${index + 1}个号----`);
  241. sendMessage.push(`----第${index + 1}个号----`);
  242. await autoSubmit(YLQX_APPID, item);
  243. await delay(1000);
  244. }
  245. lk.log('驿路黔寻预约结束\n');
  246. sendMessage.push('驿路黔寻预约结束\n');
  247. }
  248. lk.appendNotifyInfo(sendMessage.join('\n'));
  249. lk.msg('');
  250. lk.done();
  251. }
  252. function delay(time) {
  253. return new Promise(resolve => setTimeout(resolve, time));
  254. }
  255. function calculateDigest(body, sk) {
  256. const CryptoJS = createCryptoJS();
  257. const signature = CryptoJS.HmacSHA256(body, sk);
  258. return CryptoJS.enc.Base64.stringify(signature);
  259. }
  260. function calculateSignature(method, url, ak, sk, date) {
  261. const strToSign = `${method.toUpperCase()}\n${url}\n\n${ak}\n${date}\n`;
  262. const signature = CryptoJS.HmacSHA256(strToSign, sk);
  263. return CryptoJS.enc.Base64.stringify(signature);
  264. }
  265. function buildHeader(method, url, body) {
  266. const nowDate = new Date();
  267. const date = lk.formatDate(nowDate, 'ddd, DD MMM YYYY HH:mm:ss [GMT]');
  268. const signature = calculateSignature(method, url, AK, SK, date);
  269. const digest = calculateDigest(body, SK);
  270. const headers = {
  271. 'Content-Type': 'application/json',
  272. 'X-HMAC-SIGNATURE': signature,
  273. 'X-HMAC-ACCESS-KEY': AK,
  274. 'X-HMAC-ALGORITHM': 'hmac-sha256',
  275. 'X-HMAC-DIGEST': digest,
  276. 'X-HMAC-Date': date,
  277. 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/98.0.4758.102 Safari/537.36 MicroMessenger/7.0.20.1781(0x6700143B) NetType/WIFI MiniProgramEnv/Windows WindowsWechat/WMPF XWEB/6945'
  278. };
  279. return headers;
  280. }
  281. async function getUserInfo(appId, token) {
  282. const url = '/front-manager/api/customer/queryById/token';
  283. const method = 'post';
  284. const data = {appId};
  285. const headers = buildHeader(method, url, JSON.stringify(data));
  286. headers['X-access-token'] = token;
  287. return new Promise((resolve, _reject) => {
  288. try {
  289. const body = JSON.stringify(data);
  290. const url = HOST + url;
  291. let options = {
  292. url: url,
  293. headers: headers,
  294. body: body
  295. };
  296. lk.post(options, async (error, response, data) => {
  297. let resData;
  298. try {
  299. if (error) {
  300. lk.execFail();
  301. } else {
  302. resData = JSON.parse(data);
  303. }
  304. } catch (e) {
  305. lk.logErr(e);
  306. lk.execFail();
  307. } finally {
  308. resolve(resData);
  309. }
  310. });
  311. } catch (e) {
  312. lk.logErr(e);
  313. resolve({code:'-1', message: e.message});
  314. }
  315. });
  316. }
  317. async function getChannelActivity(id, token) {
  318. const url = '/front-manager/api/customer/promotion/channelActivity';
  319. const method = 'post';
  320. const data = {id};
  321. const headers = buildHeader(method, url, JSON.stringify(data));
  322. headers['X-access-token'] = token;
  323. return new Promise((resolve, _reject) => {
  324. try {
  325. const body = JSON.stringify(data);
  326. const url = HOST + url;
  327. let options = {
  328. url: url,
  329. headers: headers,
  330. body: body
  331. };
  332. lk.post(options, async (error, response, data) => {
  333. let resData;
  334. try {
  335. if (error) {
  336. lk.execFail();
  337. } else {
  338. resData = JSON.parse(data);
  339. }
  340. } catch (e) {
  341. lk.logErr(e);
  342. lk.execFail();
  343. } finally {
  344. resolve(resData);
  345. }
  346. });
  347. } catch (e) {
  348. lk.logErr(e);
  349. resolve({code:'-1', message: e.message});
  350. }
  351. });
  352. }
  353. async function getChannelInfoId(appId) {
  354. const url = '/front-manager/api/get/getChannelInfoId';
  355. const method = 'post';
  356. const data = {appId};
  357. const headers = buildHeader(method, url, JSON.stringify(data));
  358. headers['X-access-token'] = token;
  359. return new Promise((resolve, _reject) => {
  360. try {
  361. const body = JSON.stringify(data);
  362. const url = HOST + url;
  363. let options = {
  364. url: url,
  365. headers: headers,
  366. body: body
  367. };
  368. lk.post(options, async (error, response, data) => {
  369. let resData;
  370. try {
  371. if (error) {
  372. lk.execFail();
  373. } else {
  374. resData = JSON.parse(data);
  375. }
  376. } catch (e) {
  377. lk.logErr(e);
  378. lk.execFail();
  379. } finally {
  380. resolve(resData);
  381. }
  382. });
  383. } catch (e) {
  384. lk.logErr(e);
  385. resolve({code:'-1', message: e.message});
  386. }
  387. });
  388. }
  389. async function appoint(activityId, channelId, token) {
  390. const url = '/front-manager/api/customer/promotion/appoint';
  391. const method = 'post';
  392. const data = {activityId, channelId};
  393. const headers = buildHeader(method, url, JSON.stringify(data));
  394. headers['X-access-token'] = token;
  395. return new Promise((resolve, _reject) => {
  396. try {
  397. const body = JSON.stringify(data);
  398. const url = HOST + url;
  399. let options = {
  400. url: url,
  401. headers: headers,
  402. body: body
  403. };
  404. lk.post(options, async (error, response, data) => {
  405. let resData;
  406. try {
  407. if (error) {
  408. lk.execFail();
  409. } else {
  410. resData = JSON.parse(data);
  411. }
  412. } catch (e) {
  413. lk.logErr(e);
  414. lk.execFail();
  415. } finally {
  416. resolve(resData);
  417. }
  418. });
  419. } catch (e) {
  420. lk.logErr(e);
  421. resolve({code:'-1', message: e.message});
  422. }
  423. });
  424. }
  425. async function checkCustomerInQianggou(activityId, channelId, token) {
  426. const url = '/front-manager/api/customer/promotion/checkCustomerInQianggou';
  427. const method = 'post';
  428. const data = {activityId, channelId};
  429. const headers = buildHeader(method, url, JSON.stringify(data));
  430. headers['X-access-token'] = token;
  431. return new Promise((resolve, _reject) => {
  432. try {
  433. const body = JSON.stringify(data);
  434. const url = HOST + url;
  435. let options = {
  436. url: url,
  437. headers: headers,
  438. body: body
  439. };
  440. lk.post(options, async (error, response, data) => {
  441. let resData;
  442. try {
  443. if (error) {
  444. lk.execFail();
  445. } else {
  446. resData = JSON.parse(data);
  447. }
  448. } catch (e) {
  449. lk.logErr(e);
  450. lk.execFail();
  451. } finally {
  452. resolve(resData);
  453. }
  454. });
  455. } catch (e) {
  456. lk.logErr(e);
  457. resolve({code:'-1', message: e.message});
  458. }
  459. });
  460. }
  461. async function autoSubmit(appId, token) {
  462. let channelId = '';
  463. let channelName = '';
  464. if (appId === XLTH_APPID) {
  465. channelId = '8';
  466. channelName = '新联惠购';
  467. }
  468. if (appId === GLYP_APPID) {
  469. channelId = '7';
  470. channelName = '贵旅优品';
  471. }
  472. if (appId === KGLG_APPID) {
  473. channelId = '2';
  474. channelName = '空港乐购';
  475. }
  476. if (appId === HLQG_APPID) {
  477. channelId = '6';
  478. channelName = '航旅黔购';
  479. }
  480. if (appId === ZHCS_APPID) {
  481. channelId = '5';
  482. channelName = '遵行出山';
  483. }
  484. if (appId === GYQP_APPID) {
  485. channelId = '3';
  486. channelName = '贵盐黔品';
  487. }
  488. if (appId === LLSC_APPID) {
  489. channelId = '1';
  490. channelName = '乐旅商城';
  491. }
  492. if (appId === YLQX_APPID) {
  493. channelId = '9';
  494. channelName = '驿路黔寻';
  495. }
  496. try {
  497. const res1 = await getUserInfo(appId, token);
  498. if (res1.code != '10000') {
  499. lk.log(res1.message);
  500. sendMessage.push(res1.message);
  501. return;
  502. }
  503. const realName = res1.data.realName;
  504. const phone = res1.data.phone;
  505. lk.log(`当前用户[${phone}]`);
  506. sendMessage.push(`当前用户[${phone}]`);
  507. const res2 = await getChannelActivity(channelId, token);
  508. if (res2.code != '10000') {
  509. lk.log(res2.message);
  510. sendMessage.push(res2.message);
  511. return;
  512. }
  513. const activityId = res2.data.id;
  514. const activityName = res2.data.name;
  515. lk.log(`活动名称[${activityName}]`);
  516. sendMessage.push(`活动名称[${activityName}]`);
  517. const res3 = await checkCustomerInQianggou(activityId, channelId, token);
  518. if (res3.code != '10000') {
  519. lk.log(res3.message);
  520. sendMessage.push(res3.message);
  521. return;
  522. }
  523. const data = res3.data;
  524. let message = '用户已经预约成功';
  525. if (data == false) {
  526. const res4 = await appoint(activityId, channelId, token);
  527. this.sendMessage = res4.message;
  528. }
  529. lk.log(`预约结果[${message}]`);
  530. sendMessage.push(`预约结果[${message}]`);
  531. } catch (err) {
  532. lk.log(`运行异常[${err.message}]`);
  533. sendMessage.push(`运行异常[${err.message}]`);
  534. }
  535. }
  536. //---SyncByPyScript---ToolKit-start
  537. 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=>lk.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=>lk.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=>lk.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=>lk.log(t)),this.node.fs.writeFileSync("root.json",JSON.stringify(this.root,null,2),{flag:"w"},t=>lk.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),d=!c&&this.fs.existsSync(u),f=JSON.stringify(e,null,"\t"),c=(!c&&d?this.fs.writeFileSync(u,f):this.fs.writeFileSync(h,f),JSON.parse(this.fs.readFileSync(i)));if(c.hasOwnProperty("apps")&&Array.isArray(c.apps)&&0<c.apps.length){d=c.apps,u=d.indexOf(d.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=>{lk.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&&lk.log(""+this.logSeparator+t)}logErr(t){this.execStatus=!0,this.isEnableLog&&(lk.log(""+this.logSeparator+this.name+"执行异常:"),lk.log(t),lk.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"})}}(t,s,e)}
  538. //---SyncByPyScript---ToolKit-end