gandartGrab.js 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564
  1. const config = {
  2. "PATH_IMG_V2_PRO": "https://cdn-image.gandart.com/",
  3. "RUN_TYPE": 1,
  4. "BOTTOM_LOAD_MORE_THROTTLE": 100,
  5. "BASE_API": "https://api.gandart.com/base",
  6. "RESALE_API": "https://api.gandart.com/market",
  7. "RESALE_API_V2": "https://api2.gandart.com/market/api/v2",
  8. "RESALE_API_V3": "https://api2.gandart.com/market/api/v3",
  9. "SALE_API": "https://api.gandart.com/read/api",
  10. "ORDER_API": "https://api.gandart.com",
  11. "WALLET_API": "https://api.gandart.com/api",
  12. "HXT_API": "https://api.gandart.com/wallet",
  13. "PATH_IMG_V2": "https://cdn-image.gandart.com/",
  14. "SAND_RECHARGE_ACTION": "https://cap.sandpay.com.cn/v4/front-electrans/ceas.elec.trans.quick.deposit.apply",
  15. "GAME_ElF_API": "https://elf.pentajam.cn/api",
  16. "GAME_ELF_URL": "https://elf.pentajam.cn",
  17. "CORECENTER_URL": "https://api.gandart.com/corecenter"
  18. };
  19. const lk = new ToolKit(`光予首发手`, `GandartGrabber`, `gandartGrab.js`);
  20. const GandartConstKey = {
  21. // -----通用助手相关的-----
  22. Token: 'lkGandartToken',
  23. UserData: 'lkGandartUserData',
  24. PrivCollectionBrief: 'lkGandartPrivCollectionBrief',
  25. PrivCollectionDetail: 'lkGandartPrivCollectionDetail',
  26. CollectionName: 'lkGandartCollectionName',
  27. CastingId: 'lkGandartCastingId',
  28. LimitPrice: 'lkGandartLimitPrice',
  29. TransactionRecordId: 'lkGandartTransactionRecordId',
  30. CollectionCateLst: 'lkGandartCollectionCateLst',
  31. FloatPriceIntaval: 'lkGandartFloatPriceIntaval',
  32. IsCollectionWatchLocked: 'lkIsGandartCollectionWatchLocked',
  33. PrivWalletListInUse: 'lkGandartPrivWalletListInUse',
  34. IsCollectionMaxNumLimit: 'lkGandartIsCollectionMaxNumLimit',
  35. CollectionLockMaxNum: 'lkGandartCollectionLockMaxNum',
  36. PasswordPlainText: 'lkGandartPasswordPlainText',
  37. PrivPayWalletList: 'lkGandartPrivPayWalletList',
  38. // -----抢首发相关的-----
  39. NewCastingId: 'lkGandartNewCollectionCastingId',
  40. };
  41. let gandartToken = lk.getVal(GandartConstKey.Token, '');
  42. let gandartUserAgent = `Mozilla/5.0 (iPhone; CPU iPhone OS 16_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.3 Mobile/15E148 Safari/604.1`;
  43. const GCommonGandartHeads = {
  44. 'Accept': `application/json, text/plain, */*`,
  45. 'Origin': `https://www.gandart.com`,
  46. 'Accept-Encoding': `gzip, deflate, br`,
  47. 'Content-Type': `application/x-www-form-urlencoded;charset=UTF-8`,
  48. 'Connection': `keep-alive`,
  49. 'Host': `api2.gandart.com`,
  50. 'User-Agent': gandartUserAgent,
  51. 'Referer': `https://www.gandart.com/`,
  52. 'Accept-Language': `zh-CN,zh-Hans;q=0.9`,
  53. 'token': gandartToken
  54. };
  55. let gandartWalletList = 'A,B,C';
  56. if (!lk.isExecComm) {
  57. if (!lk.isRequest()) {
  58. all();
  59. }
  60. }
  61. function getCollectionMInfoById(castingId) {
  62. let ret;
  63. try {
  64. let clst = JSON.parse(lk.getVal(GandartConstKey.CollectionCateLst, "[]"));
  65. for (const d of clst) {
  66. if (d.castingId == castingId) {
  67. ret = d;
  68. break;
  69. }
  70. }
  71. } catch (error) {
  72. lk.logErr(error);
  73. }
  74. return ret;
  75. }
  76. function getCollectionNameById(castingId) {
  77. let name = '未获取';
  78. let info = getCollectionMInfoById(castingId);
  79. if (info) {
  80. name = info.collectionName;
  81. }
  82. return name;
  83. }
  84. function isCollectionMaxNumLimitEnable() {
  85. let isEnabled = lk.getVal(GandartConstKey.IsCollectionMaxNumLimit);
  86. isEnabled = lk.isEmpty(isEnabled) ? false : JSON.parse(isEnabled);
  87. return isEnabled;
  88. }
  89. function getCollectionLockMaxNum() {
  90. let num = lk.getVal(GandartConstKey.CollectionLockMaxNum, 6);
  91. return Number(num);
  92. }
  93. function getMyPasswordPlainText() {
  94. let str = lk.getVal(GandartConstKey.PasswordPlainText);
  95. let ret = '';
  96. if (!lk.isEmpty(str)) {
  97. ret = str;
  98. }
  99. return ret;
  100. }
  101. function getMyPayWalletList() {
  102. let str = lk.getVal(GandartConstKey.PrivPayWalletList);
  103. let ret = 'A,B,C';
  104. if (!lk.isEmpty(str)) {
  105. ret = str;
  106. }
  107. return ret;
  108. }
  109. async function all() {
  110. let hasNeedSendNotify = true;
  111. if (gandartToken == '') {
  112. lk.execFail();
  113. lk.appendNotifyInfo(`⚠️请先获取光予token`);
  114. } else {
  115. let isPass = true;
  116. if (isCollectionMaxNumLimitEnable()) {
  117. let maxLockNum = getCollectionLockMaxNum();
  118. let orderLstInfo = await getNftOrderList(0);
  119. if (orderLstInfo.total >= maxLockNum) {
  120. lk.log('当前未支付订单达到预设的限制单量');
  121. lk.prependNotifyInfo(`⚠️当前未支付订单达到限制${maxLockNum}单`);
  122. isPass = false;
  123. hasNeedSendNotify = true;
  124. }
  125. }
  126. if (isPass) {
  127. let castingItem = await getNewCollectionCastingData();
  128. if (castingItem) {
  129. let userInfo = await getUser();
  130. let stockNum = await getStock(castingItem.id, castingItem.collectionName);
  131. if (stockNum > 0) {
  132. gandartWalletList = getMyPayWalletList();
  133. let myWalletList = await getMyWalletListDetail();
  134. let usedWalletInfo = myWalletList[0];
  135. for (let i = myWalletList.length - 1; i >= 0; i--) {
  136. let walletInfo = myWalletList[i];
  137. if (checkIsInPrivWalletList(walletInfo) && walletInfo.availableMoney >= castingItem.price) {
  138. usedWalletInfo = walletInfo;
  139. break;
  140. }
  141. }
  142. if (usedWalletInfo) {
  143. await generate();
  144. await submitOrder();
  145. let maxQueryCnt = 10;
  146. let startNum = 0;
  147. let isCreated = false;
  148. while (startNum < maxQueryCnt) {
  149. let ret = await getIsCreateOrder();
  150. if (ret && ret.success) {
  151. if (Number(ret.code) == 1) {
  152. isCreated = true;
  153. break;
  154. } else {
  155. lk.log(ret.msg);
  156. }
  157. }
  158. startNum++;
  159. }
  160. if (isCreated) {
  161. let orderInfo = await getByOrderNo();
  162. await getPublicKey();
  163. await confirmPassword();
  164. await kftPay();
  165. }
  166. } else {
  167. lk.log('无匹配钱包进行支付');
  168. }
  169. } else {
  170. lk.log('产品库存不足');
  171. }
  172. hasNeedSendNotify = true;
  173. } else {
  174. lk.log('暂无匹配的首发产品发售');
  175. hasNeedSendNotify = false;
  176. }
  177. }
  178. }
  179. if (hasNeedSendNotify) {
  180. lk.msg(``);
  181. }
  182. lk.done();
  183. }
  184. function checkIsInPrivWalletList(walletInfo) {
  185. // if (gandartWalletList.indexOf(walletInfo.channel) > -1) {
  186. // return true;
  187. // }
  188. // return false;
  189. return true;
  190. }
  191. async function getNewCollectionCastingData() {
  192. let newCastingId = lk.getVal(GandartConstKey.NewCastingId, 0);
  193. newCastingId = Number(newCastingId);
  194. if (newCastingId == 0) {
  195. lk.log(`没有指定首发产品铸造ID,进行自动匹配合适的首发产品`);
  196. }
  197. let nftSeriers = await findAllFirstNftSeriers();
  198. let tNftCollection;
  199. let allMateCollections = [];
  200. if (nftSeriers) {
  201. let nowDate = new Date();
  202. for (let i = 0; i < nftSeriers.length; i++) {
  203. let nftSerier = nftSeriers[i];
  204. let startDate = new Date(nftSerier.startDate);
  205. if (nowDate.getTime() >= startDate.getTime() - 1000 && nftSerier.status == 1) {
  206. let castingList = nftSerier.castingList;
  207. for (let cItem of castingList) {
  208. if (cItem.id == newCastingId) {
  209. tNftCollection = cItem;
  210. break;
  211. }
  212. allMateCollections.push(cItem);
  213. }
  214. }
  215. }
  216. } else {
  217. newCastingId = 0;
  218. }
  219. if (tNftCollection) {
  220. return tNftCollection;
  221. }
  222. return allMateCollections[0];
  223. }
  224. async function getNftOrderList(status, orderType = 0, page = 1, pageSize = 6) {
  225. return new Promise((resolve, _reject) => {
  226. try {
  227. const headers = GCommonGandartHeads;
  228. headers.Host = "api.gandart.com";
  229. // status 0:进行中 1:已完成 2:已取消
  230. let body = `orderNo=&status=${status}&orderType=${orderType}&page=${page}&pageSize=${pageSize}`;
  231. let url = {
  232. url: `${config.BASE_API}/order/nftorder/list`,
  233. headers: headers,
  234. body: body
  235. };
  236. lk.log(`请求个人订单列表:${body}`);
  237. lk.post(url, async (error, _response, data) => {
  238. let rows;
  239. let total = 0;
  240. try {
  241. if (error) {
  242. lk.log(`获取订单列表数据发生错误`);
  243. lk.execFail();
  244. // lk.appendNotifyInfo(`❌获取订单列表失败,请稍后再试`);
  245. } else {
  246. let info = JSON.parse(data);
  247. rows = info.rows;
  248. total = info.total || info.recordsTotal;
  249. }
  250. } catch (e) {
  251. lk.log(`获取订单列表数据发生错误`);
  252. lk.logErr(e);
  253. lk.execFail();
  254. } finally {
  255. resolve({ total: total, rows: rows });
  256. }
  257. });
  258. } catch (e) {
  259. lk.log(`获取订单列表数据发生错误`);
  260. lk.logErr(e);
  261. resolve({ total: 0 });
  262. }
  263. });
  264. }
  265. async function findAllFirstNftSeriers(page = 1, pageSize = 5) {
  266. return new Promise((resolve, _reject) => {
  267. try {
  268. const headers = GCommonGandartHeads;
  269. headers.Host = "api.gandart.com";
  270. let body = `page=${page}&pageSize=${pageSize}`;
  271. let url = {
  272. url: `${config.BASE_API}/casting/nftseriers/findAll`,
  273. headers: headers,
  274. body: body
  275. };
  276. lk.log(`请求首发产品列表:${body}`);
  277. lk.post(url, async (error, _response, data) => {
  278. let retItemLst = [];
  279. try {
  280. if (error) {
  281. lk.log(`获取首发列表数据发生错误`);
  282. lk.execFail();
  283. lk.appendNotifyInfo(`❌获取首发列表失败,请稍后再试`);
  284. } else {
  285. let ret = JSON.parse(data);
  286. if (ret.success) {
  287. retItemLst = ret.obj;
  288. } else {
  289. lk.log(`获取首发列表失败,响应:${data}`);
  290. }
  291. }
  292. } catch (e) {
  293. lk.log(`获取首发列表数据发生错误`);
  294. lk.logErr(e);
  295. lk.execFail();
  296. } finally {
  297. resolve(retItemLst);
  298. }
  299. });
  300. } catch (e) {
  301. lk.log(`获取首发列表数据发生错误`);
  302. lk.logErr(e);
  303. resolve();
  304. }
  305. });
  306. }
  307. async function getUser() {
  308. return new Promise((resolve, _reject) => {
  309. try {
  310. const headers = GCommonGandartHeads;
  311. headers.Host = "api.gandart.com";
  312. let body = ``;
  313. let url = {
  314. url: `${config.BASE_API}/user/getUser`,
  315. headers: headers,
  316. body: body
  317. };
  318. collectionName = collectionName || getCollectionNameById(castingId);
  319. lk.log(`请求用户信息:${body}`);
  320. lk.post(url, async (error, _response, data) => {
  321. let userInfo;
  322. try {
  323. if (error) {
  324. lk.log(`获取用户信息发生错误`);
  325. lk.execFail();
  326. } else {
  327. let ret = JSON.parse(data);
  328. if (ret.success) {
  329. lk.log(`获取用户信息成功`);
  330. let obj = ret.obj;
  331. userInfo = obj.user;
  332. } else {
  333. lk.log(`获取用户信息失败,响应:${data}`);
  334. }
  335. }
  336. } catch (e) {
  337. lk.log(`获取用户信息发生错误`);
  338. lk.logErr(e);
  339. lk.execFail();
  340. } finally {
  341. resolve(userInfo);
  342. }
  343. });
  344. } catch (e) {
  345. lk.log(`获取用户信息发生错误`);
  346. lk.logErr(e);
  347. resolve();
  348. }
  349. });
  350. }
  351. async function getStock(castingId, collectionName) {
  352. return new Promise((resolve, _reject) => {
  353. try {
  354. const headers = GCommonGandartHeads;
  355. headers.Host = "api2.gandart.com";
  356. let body = `ids=${castingId}`;
  357. let url = {
  358. url: `${config.BASE_API}/casting/nftseries/getStock`,
  359. headers: headers,
  360. body: body
  361. };
  362. collectionName = collectionName || getCollectionNameById(castingId);
  363. lk.log(`请求产品[${collectionName}(${castingId})]库存:${body}`);
  364. lk.post(url, async (error, _response, data) => {
  365. let stockNum = 0;
  366. try {
  367. if (error) {
  368. lk.log(`获取产品库存发生错误`);
  369. lk.execFail();
  370. lk.appendNotifyInfo(`❌获取产品库存失败,请稍后再试`);
  371. } else {
  372. let ret = JSON.parse(data);
  373. if (ret.success) {
  374. lk.log(`获取产品库存成功,响应:${data}`);
  375. let obj = ret.obj;
  376. stockNum = Number(obj[castingId] || 0);
  377. } else {
  378. lk.log(`获取产品库存失败,响应:${data}`);
  379. }
  380. }
  381. } catch (e) {
  382. lk.log(`获取产品库存发生错误`);
  383. lk.logErr(e);
  384. lk.execFail();
  385. } finally {
  386. resolve(stockNum);
  387. }
  388. });
  389. } catch (e) {
  390. lk.log(`获取产品库存发生错误`);
  391. lk.logErr(e);
  392. resolve(0);
  393. }
  394. });
  395. }
  396. async function getMyWalletListDetail() {
  397. return new Promise((resolve, _reject) => {
  398. try {
  399. const headers = GCommonGandartHeads;
  400. headers.Host = "api.gandart.com";
  401. const body = ``;
  402. let url = {
  403. url: `${config.WALLET_API}/v2/wallet/list?`,
  404. headers: headers,
  405. body: body
  406. };
  407. lk.log(`请求个人钱包列表数据`);
  408. lk.get(url, async (error, _response, data) => {
  409. let ret;
  410. try {
  411. if (error) {
  412. lk.log(`请求个人钱包列表数据发生错误`);
  413. lk.execFail();
  414. } else {
  415. let info = JSON.parse(data);
  416. if (info.success) {
  417. lk.log(`请求个人钱包列表数据成功`);
  418. ret = info.obj;
  419. } else {
  420. lk.log(`请求个人钱包列表数据失败,响应:${data}`);
  421. }
  422. }
  423. } catch (e) {
  424. lk.log(`请求个人钱包列表数据发生错误`);
  425. lk.logErr(e);
  426. lk.execFail();
  427. } finally {
  428. resolve(ret);
  429. }
  430. });
  431. } catch (e) {
  432. lk.log(`请求个人钱包列表数据发生错误`);
  433. lk.logErr(e);
  434. resolve();
  435. }
  436. });
  437. }
  438. function getPayUrlPath(walletName) {
  439. let path = `${config.BASE_API}/v2/resaleManage/resale/pay`;
  440. switch (walletName) {
  441. case 'A':
  442. // TEST
  443. path = `${config.WALLET_API}/v2/wallet/hxt/pay/hengXinTongPay`;
  444. // RELEASE
  445. path = `${config.HXT_API}/v2/wallet/hxt/pay/hengXinTongPay`;
  446. break;
  447. case 'B':
  448. path = `${config.WALLET_API}/v2/wallet/sand/pay/createOrder`;
  449. break;
  450. case 'C':
  451. path = `${config.WALLET_API}/v2/wallet/kft/pay/kftPay`;
  452. break;
  453. default:
  454. break;
  455. }
  456. return path;
  457. }
  458. function checkToBuyCollection(item) {
  459. return new Promise((resolve, _reject) => {
  460. try {
  461. const id = item.id;
  462. const viewSort = item.viewSort || 0;
  463. const headers = GCommonGandartHeads;
  464. headers.Host = "api.gandart.com";
  465. const body = `transactionRecordId=${id}`;
  466. let url = {
  467. url: `${config.BASE_API}/v2/resaleManage/resale/buy/v2`,
  468. headers: headers,
  469. body: body
  470. };
  471. lk.log(`请求购买产品[${item.collectionName}#${viewSort}]=${body}`);
  472. const collectionName = item.collectionName;
  473. lk.post(url, async (error, _response, data) => {
  474. let retItem;
  475. try {
  476. if (error) {
  477. lk.log(`无法下单产品[${collectionName}#${viewSort}]`);
  478. lk.execFail();
  479. lk.appendNotifyInfo(`❌购买失败,请稍后再试`);
  480. } else {
  481. lk.log(data);
  482. let retBuyInfo = JSON.parse(data);
  483. if (retBuyInfo.success) {
  484. let obj = retBuyInfo.obj;
  485. let orderNum = obj.orderNum;
  486. if (orderNum && orderNum.length > 0) {
  487. retItem = retBuyInfo;
  488. lk.log(`成功下单产品[${obj.collectionName}#${viewSort}],价格:${item.resalePrice},订单号:${obj.orderNum}`);
  489. lk.appendNotifyInfo(`🎉成功下单产品[${obj.collectionName}#${viewSort}],价格:${item.resalePrice},订单号:${obj.orderNum}`);
  490. }
  491. } else {
  492. lk.log(`产品[${collectionName}#${viewSort}]下单失败`);
  493. }
  494. }
  495. } catch (e) {
  496. lk.log(`下单产品[${collectionName}#${viewSort}]发生错误`);
  497. lk.logErr(e);
  498. lk.execFail();
  499. } finally {
  500. resolve(retItem);
  501. }
  502. });
  503. } catch (e) {
  504. lk.log(`下单产品[${item.collectionName}#${item.viewSort}]发生错误`);
  505. lk.logErr(e);
  506. resolve();
  507. }
  508. });
  509. }
  510. function getSign() {
  511. const now = new Date().getTime();
  512. let s1 = now - 20000;
  513. let s2 = now;
  514. let s3 = now + 20000;
  515. let s4 = now + 40000;
  516. const sssfffeee = 'e9'; //localStorage.getItem('sssfffeee');
  517. const aggaaa = '9'; //localStorage.getItem('aggaaa');
  518. const yyhhhasdww = '0a';
  519. let arr = [sssfffeee, yyhhhasdww, aggaaa, '29', 'e', 'c', 3];
  520. let crypto = createCrypto();
  521. let s5 = crypto.md5(s3 + arr.join(''));
  522. return {
  523. 's1': s1,
  524. 's2': s2,
  525. 's3': s3,
  526. 's4': s4,
  527. 's5': s5
  528. };
  529. }
  530. // https://github.com/lowking/Scripts/blob/master/util/ToolKit.min.js
  531. //ToolKit-start
  532. function ToolKit(t, s, i) { return new class { constructor(t, s, i) { this.tgEscapeCharMapping = { "&": "&", "#": "#" }, this.userAgent = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_6) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/12.0.2 Safari/605.1.15", this.prefix = "lk", this.name = t, this.id = s, this.data = null, this.dataFile = this.getRealPath("" + this.prefix + this.id + ".dat"), this.boxJsJsonFile = this.getRealPath("" + this.prefix + this.id + ".boxjs.json"), this.options = i, this.isExecComm = !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() { var t, s; this.isRequest() && (t = $request.body) && (s = $request.path, s = this.id + "#" + s.replace("/", "_"), this.isQuanX() && $prefs.setValueForKey(t, s), (this.isLoon() || this.isSurge()) && $persistentStore.write(t, s), this.isNode()) && this.node.fs.writeFileSync(s + ".json", t, { flag: "w" }, t => console.log(t)) } getRequestBody() { var t = $request.path, t = this.id + "#" + t.replace("/", "_"); if (this.isSurge() || this.isLoon()) return $persistentStore.read(t); if (this.isQuanX()) return $prefs.valueForKey(t); if (this.isNode()) { t = t + ".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 e = this.comm[0], s = (this.log(`获取【${e}】内容传给手机`), ""); this.fs = this.fs || require("fs"), this.path = this.path || require("path"); var i = this.path.resolve(e), r = this.path.resolve(process.cwd(), e), o = this.fs.existsSync(i), h = !o && this.fs.existsSync(r); if (o || h) { h = o ? i : 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, i) => { this.log(`已将脚本【${e}】发给手机!`), 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 e = "/Users/lowking/Desktop/Scripts/lowking.boxjs.json"; if (r && r.hasOwnProperty("target_boxjs_json_path") && (e = r.target_boxjs_json_path), this.fs.existsSync(e)) if (this.isJsonObject(s) && this.isJsonObject(r)) { this.log("using node"); var o = ["settings", "keys"], h = "https://raw.githubusercontent.com/Orz-3"; let i = {}, t = "#lk{script_url}"; if (r && r.hasOwnProperty("script_url") && (t = this.isEmpty(r.script_url) ? "#lk{script_url}" : r.script_url), i.id = "" + this.prefix + this.id, i.name = this.name, i.desc_html = `⚠️使用说明</br>详情【<a href='${t}?raw=true'><font class='red--text'>点我查看</font></a>】`, i.icons = [h + `/mini/master/Alpha/${this.id.toLocaleLowerCase()}.png`, h + `/mini/master/Color/${this.id.toLocaleLowerCase()}.png`], i.keys = [], i.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=" }], i.author = "#lk{author}", i.repo = "#lk{repo}", i.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 < i.settings.length; t++) { var p = i.settings[t]; l.id === p.id && i.settings.splice(t, 1) } } i[a] = i[a].concat(s[a]) } delete s[a] } if (Object.assign(i, 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), f = this.fs.existsSync(h), c = !f && this.fs.existsSync(u), d = JSON.stringify(i, null, "\t"), f = (!f && c ? this.fs.writeFileSync(u, d) : this.fs.writeFileSync(h, d), JSON.parse(this.fs.readFileSync(e))); if (f.hasOwnProperty("apps") && Array.isArray(f.apps) && 0 < f.apps.length) { c = f.apps, u = c.indexOf(c.filter(t => t.id == i.id)[0]); 0 <= u ? f.apps[u] = i : f.apps.push(i); let s = JSON.stringify(f, 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(e, 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() } sleep(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, i, e) { 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, i) => { this.log("Tg通知完毕") }) } else { var o = {}, h = !this.isEmpty(i), n = !this.isEmpty(e); this.isQuanX() && (h && (o["open-url"] = i), n && (o["media-url"] = e), $notify(this.name, t, s, o)), (this.isSurge() || this.isStash()) && (h && (o.url = i), $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 }) } } getVal(t, s = "") { let i; return (i = 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), i = this.fs.existsSync(t), e = !i && this.fs.existsSync(s); if (!i && !e) return {}; e = i ? t : s; try { return JSON.parse(this.fs.readFileSync(e)) } catch (t) { return {} } } writeData() { var t, s, i, e, 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), e = !(i = this.fs.existsSync(t)) && this.fs.existsSync(s), r = JSON.stringify(this.data), !i && e ? 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, e = () => { }) { this.isQuanX() && ((t = "string" == typeof t ? { url: t } : t).method = "GET", $task.fetch(t).then(t => { e(null, this.adapterStatus(t), t.body) }, t => e(t.error, null, null))), (this.isSurge() || this.isLoon() || this.isStash()) && $httpClient.get(t, (t, s, i) => { e(t, this.adapterStatus(s), i) }), this.isNode() && this.node.request(t, (t, s, i) => { e(t, this.adapterStatus(s), i) }), this.isJSBox() && ((t = "string" == typeof t ? { url: t } : t).header = t.headers, t.handler = function (t) { let s = t.error, i = (s = s && JSON.stringify(t.error), t.data); "object" == typeof i && (i = JSON.stringify(t.data)), e(s, this.adapterStatus(t.response), i) }, $http.get(t)) } post(t, e = () => { }) { this.isQuanX() && ((t = "string" == typeof t ? { url: t } : t).method = "POST", $task.fetch(t).then(t => { e(null, this.adapterStatus(t), t.body) }, t => e(t.error, null, null))), (this.isSurge() || this.isLoon() || this.isStash()) && $httpClient.post(t, (t, s, i) => { e(t, this.adapterStatus(s), i) }), this.isNode() && this.node.request.post(t, (t, s, i) => { e(t, this.adapterStatus(s), i) }), this.isJSBox() && ((t = "string" == typeof t ? { url: t } : t).header = t.headers, t.handler = function (t) { let s = t.error, i = (s = s && JSON.stringify(t.error), t.data); "object" == typeof i && (i = JSON.stringify(t.data)), e(s, this.adapterStatus(t.response), i) }, $http.post(t)) } put(t, e = () => { }) { this.isQuanX() && ((t = "string" == typeof t ? { url: t } : t).method = "PUT", $task.fetch(t).then(t => { e(null, this.adapterStatus(t), t.body) }, t => e(t.error, null, null))), (this.isSurge() || this.isLoon() || this.isStash()) && (t.method = "PUT", $httpClient.put(t, (t, s, i) => { e(t, this.adapterStatus(s), i) })), this.isNode() && (t.method = "PUT", this.node.request.put(t, (t, s, i) => { e(t, this.adapterStatus(s), i) })), this.isJSBox() && ((t = "string" == typeof t ? { url: t } : t).header = t.headers, t.handler = function (t) { let s = t.error, i = (s = s && JSON.stringify(t.error), t.data); "object" == typeof i && (i = JSON.stringify(t.data)), e(s, this.adapterStatus(t.response), i) }, $http.post(t)) } costTime() { let t = this.name + "执行完毕!"; this.isNode() && this.isExecComm && (t = `指令【${this.comm[1]}】执行完毕!`); var s = (new Date).getTime() - this.startTime, i = s / 1e3; this.execCount++, this.costTotalMs += s, this.log(`${t}耗时【${i}】秒\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 i = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz1234567890", e = i.length, r = ""; for (let t = 0; t < s; t++)r += i.charAt(Math.floor(Math.random() * e)); return r } autoComplete(s, t, i, e, r, o, h, n, a, l) { if ((s += "").length < r) for (; s.length < r;)0 == o ? s += e : s = e + 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 + i) } customReplace(t, s, i, e) { try { for (var r in this.isEmpty(i) && (i = "#{"), this.isEmpty(e) && (e = "}"), s) t = t.replace("" + i + r + e, s[r]) } catch (t) { this.logErr(t) } return t } toDBC(t) { for (var s = "", i = 0; i < t.length; i++)32 == t.charCodeAt(i) ? s += String.fromCharCode(12288) : t.charCodeAt(i) < 127 && (s += String.fromCharCode(t.charCodeAt(i) + 65248)); return s } hash(t) { let s = 0, i, e; for (i = 0; i < t.length; i++)e = t.charCodeAt(i), s = (s << 5) - s + e, s |= 0; return String(s) } formatDate(t, s) { var i, e = { "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 (i in /(y+)/.test(s) && (s = s.replace(RegExp.$1, (t.getFullYear() + "").substr(4 - RegExp.$1.length))), e) new RegExp("(" + i + ")").test(s) && (s = s.replace(RegExp.$1, 1 == RegExp.$1.length ? e[i] : ("00" + e[i]).substr(("" + e[i]).length))); return s } objToQueryStr(s, i) { let e = ""; for (const r in s) { let t = s[r]; null != t && "" !== t && ("object" == typeof t ? t = JSON.stringify(t) : i && (t = encodeURIComponent(t)), e += `${r}=${t}&`) } return e = e.substring(0, e.length - 1) } parseQueryStr(t) { var s = {}, i = (t = -1 < t.indexOf("?") ? t.split("?")[1] : t).split("&"); for (let t = 0; t < i.length; t++) { var e = i[t].split("="); s[e[0]] = e[1] } return s } deepClone(t, s) { for (var i in s = s || {}, t) "object" == typeof t[i] ? (s[i] = t[i].constructor === Array ? [] : {}, this.deepClone(t[i], s[i])) : s[i] = t[i]; return s } }(t, s, i) }
  533. //ToolKit-end
  534. //Crypto-start
  535. function createCrypto() { function o(t, n) { var r = c(t[0], o = t[1], u = t[2], e = t[3], n[0], 7, -680876936), e = c(e, r, o, u, n[1], 12, -389564586), u = c(u, e, r, o, n[2], 17, 606105819), o = c(o, u, e, r, n[3], 22, -1044525330); r = c(r, o, u, e, n[4], 7, -176418897), e = c(e, r, o, u, n[5], 12, 1200080426), u = c(u, e, r, o, n[6], 17, -1473231341), o = c(o, u, e, r, n[7], 22, -45705983), r = c(r, o, u, e, n[8], 7, 1770035416), e = c(e, r, o, u, n[9], 12, -1958414417), u = c(u, e, r, o, n[10], 17, -42063), o = c(o, u, e, r, n[11], 22, -1990404162), r = c(r, o, u, e, n[12], 7, 1804603682), e = c(e, r, o, u, n[13], 12, -40341101), u = c(u, e, r, o, n[14], 17, -1502002290), r = i(r, o = c(o, u, e, r, n[15], 22, 1236535329), u, e, n[1], 5, -165796510), e = i(e, r, o, u, n[6], 9, -1069501632), u = i(u, e, r, o, n[11], 14, 643717713), o = i(o, u, e, r, n[0], 20, -373897302), r = i(r, o, u, e, n[5], 5, -701558691), e = i(e, r, o, u, n[10], 9, 38016083), u = i(u, e, r, o, n[15], 14, -660478335), o = i(o, u, e, r, n[4], 20, -405537848), r = i(r, o, u, e, n[9], 5, 568446438), e = i(e, r, o, u, n[14], 9, -1019803690), u = i(u, e, r, o, n[3], 14, -187363961), o = i(o, u, e, r, n[8], 20, 1163531501), r = i(r, o, u, e, n[13], 5, -1444681467), e = i(e, r, o, u, n[2], 9, -51403784), u = i(u, e, r, o, n[7], 14, 1735328473), r = a(r, o = i(o, u, e, r, n[12], 20, -1926607734), u, e, n[5], 4, -378558), e = a(e, r, o, u, n[8], 11, -2022574463), u = a(u, e, r, o, n[11], 16, 1839030562), o = a(o, u, e, r, n[14], 23, -35309556), r = a(r, o, u, e, n[1], 4, -1530992060), e = a(e, r, o, u, n[4], 11, 1272893353), u = a(u, e, r, o, n[7], 16, -155497632), o = a(o, u, e, r, n[10], 23, -1094730640), r = a(r, o, u, e, n[13], 4, 681279174), e = a(e, r, o, u, n[0], 11, -358537222), u = a(u, e, r, o, n[3], 16, -722521979), o = a(o, u, e, r, n[6], 23, 76029189), r = a(r, o, u, e, n[9], 4, -640364487), e = a(e, r, o, u, n[12], 11, -421815835), u = a(u, e, r, o, n[15], 16, 530742520), r = l(r, o = a(o, u, e, r, n[2], 23, -995338651), u, e, n[0], 6, -198630844), e = l(e, r, o, u, n[7], 10, 1126891415), u = l(u, e, r, o, n[14], 15, -1416354905), o = l(o, u, e, r, n[5], 21, -57434055), r = l(r, o, u, e, n[12], 6, 1700485571), e = l(e, r, o, u, n[3], 10, -1894986606), u = l(u, e, r, o, n[10], 15, -1051523), o = l(o, u, e, r, n[1], 21, -2054922799), r = l(r, o, u, e, n[8], 6, 1873313359), e = l(e, r, o, u, n[15], 10, -30611744), u = l(u, e, r, o, n[6], 15, -1560198380), o = l(o, u, e, r, n[13], 21, 1309151649), r = l(r, o, u, e, n[4], 6, -145523070), e = l(e, r, o, u, n[11], 10, -1120210379), u = l(u, e, r, o, n[2], 15, 718787259), o = l(o, u, e, r, n[9], 21, -343485551), t[0] = h(r, t[0]), t[1] = h(o, t[1]), t[2] = h(u, t[2]), t[3] = h(e, t[3]) } function f(t, n, r, e, u, o) { return n = h(h(n, t), h(e, o)), h(n << u | n >>> 32 - u, r) } function c(t, n, r, e, u, o, c) { return f(n & r | ~n & e, t, n, u, o, c) } function i(t, n, r, e, u, o, c) { return f(n & e | r & ~e, t, n, u, o, c) } function a(t, n, r, e, u, o, c) { return f(n ^ r ^ e, t, n, u, o, c) } function l(t, n, r, e, u, o, c) { return f(r ^ (n | ~e), t, n, u, o, c) } function n(t) { /[\x80-\xFF]/.test(t) && (t = unescape(encodeURI(t))), txt = ""; let n = t.length, r = [1732584193, -271733879, -1732584194, 271733878], e; for (e = 64; e <= t.length; e += 64)o(r, function (t) { let n = [], r; for (r = 0; r < 64; r += 4)n[r >> 2] = t.charCodeAt(r) + (t.charCodeAt(r + 1) << 8) + (t.charCodeAt(r + 2) << 16) + (t.charCodeAt(r + 3) << 24); return n }(t.substring(e - 64, e))); t = t.substring(e - 64); var u = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]; for (e = 0; e < t.length; e++)u[e >> 2] |= t.charCodeAt(e) << (e % 4 << 3); if (u[e >> 2] |= 128 << (e % 4 << 3), 55 < e) for (o(r, u), e = 0; e < 16; e++)u[e] = 0; return u[14] = 8 * n, o(r, u), r } let e = "0123456789abcdef".split(""); function r(n) { for (let t = 0; t < n.length; t++)n[t] = function (t) { let n = "", r = 0; for (; r < 4; r++)n += e[t >> 8 * r + 4 & 15] + e[t >> 8 * r & 15]; return n }(n[t]); return n.join("") } function h(t, n) { return t + n & 4294967295 } return { md5: function (t) { return r(n(t)) } } }
  536. //Crypto-end