huluwaHelper.js 36 KB

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