wubianHelper.js 81 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401
  1. /*
  2. 无边星链助手
  3. [Script]
  4. # > 无边星链助手
  5. ^https?:\/\/api\.wubian\.pro url script-request-body https://git.jojo21.top/shawenguan/Quantumult-X/raw/master/Scripts/wubian/wubianHelper.js
  6. ^https?:\/\/api\.wubian\.pro url script-response-body https://git.jojo21.top/shawenguan/Quantumult-X/raw/master/Scripts/wubian/wubianHelper.js
  7. [MITM]
  8. hostname = api.wubian.pro
  9. */
  10. const scriptName = `无边星链助手`;
  11. const magicJS = MagicJS(scriptName, "INFO");
  12. const WuBianConstKey = {
  13. // 基础数据
  14. UserInfo: 'WubianUserInfo',
  15. Token: 'WubianProToken',
  16. BrowserProxyUrl: 'WubianBrowserProxyUrl',
  17. TenantId: 'WubianTenantId',
  18. MarketAlbumListData: 'WubianMarketAlbumListData',
  19. AllMarketGroupListData: 'WubianAllMarketGroupListData',
  20. MyAllArtModifyCountData: 'WubianMyAllArtModifyCountData',
  21. MyAllArtRealCountData: 'WubianMyAllArtRealCountData',
  22. MyArtCountCaptureEnabled: 'WubianMyArtCountCaptureEnabled',
  23. MyArtCountModifyMode: 'WubianMyArtCountModifyMode',
  24. ArtIdNameMapData: 'WubianArtIdNameMapData',
  25. DataMaskingEnabled: 'WubianDataMaskingEnabled',
  26. VerListMaskkingDataMap: 'WubianVerListMaskkingDataMap',
  27. AllUserSimpleInfo: 'WubianAllUserSimpleInfo',
  28. ScoreRankDaily: 'WubianScoreRankDaily',
  29. QueryOthersCollectUserName: 'WubianQueryOthersCollectUserName',
  30. QueryOthersCollectHashId: 'WubianQueryOthersCollectHashId',
  31. ActivityListData: 'WubianActivityListData',
  32. ActivityGoodsList: 'WubianActivityGoodsList',
  33. ActivityTaskCaptureEnabled: 'WubianActivityTaskCaptureEnabled',
  34. ActivityTagName: 'WubianActivityActivityTagName',
  35. ActivityIndexName: 'WubianActivityIndexName',
  36. ActivityId: 'WubianActivityId',
  37. SynthesisConcurrentMode: 'WubianSynthesisConcurrentMode',
  38. SynthesisRunsPerSecond: 'WubianSynthesisRunsPerSecond',
  39. SynthesisAmount: 'WubianSynthesisTargetAmount',
  40. SynthesisActivityDisabled: 'WubianSynthesisActivityDisabled',
  41. FirstOrderCaptureEnabled: 'WubianFirstOrderCaptureEnabled',
  42. FirstArtListData: 'WubianFirstArtListData',
  43. FirstGoodsId: 'WubianFirstGoodsId',
  44. FirstGoodsName: 'WubianFirstGoodsName',
  45. CreateFirstOrderData: 'WubianCreateFirstOrderData',
  46. FirstGrabConcurrentMode: 'WubianFirstGrabConcurrentMode',
  47. FirstGrabRunsPerSecond: 'WubianFirstGrabRunsPerSecond',
  48. ConsignmentCaptureEnabled: 'WubianConsignmentCaptureEnabled',
  49. MyCollectListData: 'WubianMyCollectListData',
  50. ConsignmentArtName: 'WubianConsignmentArtName',
  51. ConsignmentArtId: 'WubianConsignmentArtId',
  52. ConsignmentArtPrice: 'WubianConsignmentArtPrice',
  53. ConsignmentArtAmount: 'WubianConsignmentArtAmount',
  54. FastBuyCaptureEnabled: 'WubianFastBuyCaptureEnabled',
  55. FastBuyQuickModeEnabled: 'WubianFastBuyQuickModeEnabled',
  56. FastBuyArtName: 'WubianFastBuyArtName',
  57. FastBuyArtId: 'WubianFastBuyArtId',
  58. FastBuyArtPrice: 'WubianFastBuyArtPrice',
  59. FastBuyArtAmount: 'WubianFastBuyArtAmount',
  60. FastBuyArtInfo: 'WubianFastBuyArtInfo',
  61. };
  62. const gUserAgent = `Mozilla/5.0 (iPhone; CPU iPhone OS 16_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148 Html5Plus/1.0 (Immersed/20) uni-app`;
  63. const gHost = `api.wubian.pro`;
  64. let gToken = magicJS.data.read(WuBianConstKey.Token, '');
  65. const gCommonHeaders = {
  66. 'Accept': `*/*`,
  67. 'Accept-Encoding': `gzip, deflate, br`,
  68. 'Connection': `keep-alive`,
  69. 'Content-Type': `application/json`,
  70. 'Cookie': `token=${gToken}`,
  71. 'Host': gHost,
  72. 'User-Agent': gUserAgent,
  73. 'CLIENT-TYPE': `APP`,
  74. 'token': gToken,
  75. 'Accept-Language': `zh-CN,zh-Hans;q=0.9`
  76. };
  77. let gRetBody;
  78. async function Main() {
  79. if (magicJS.isStrictRequest) {
  80. magicJS.checkRecordRequestBody();
  81. }
  82. if (magicJS.isRequest) {
  83. checkHandleRequest();
  84. } else {
  85. updateHeaders();
  86. }
  87. magicJS.notification.msg('');
  88. if (gRetBody) {
  89. magicJS.done({
  90. body: JSON.stringify(gRetBody)
  91. });
  92. } else {
  93. magicJS.done();
  94. }
  95. }
  96. function checkHandleRequest() {
  97. const url = $request.url;
  98. const path = $request.path;
  99. magicJS.logger.info(`请求url=${url}#${$request.method}`);
  100. magicJS.logger.info(`请求body=${magicJS.getRequestBody()}`);
  101. if ($request && $request.method != 'OPTIONS') {
  102. switch (path) {
  103. case '/vmf/app/login/mobileLogin':
  104. handleMobileLogin();
  105. break;
  106. case '/vmf/app/user/collect/list':
  107. // 我的-产品品类列表
  108. handleUserCollectList();
  109. break;
  110. case '/vmf/app/user/collect/verList':
  111. // 我的-某一个产品的列表详情
  112. handleVerList();
  113. break;
  114. case '/vmf/app/user/getBatchGiveArtIds':
  115. break;
  116. case '/vmf/app/market/goodsInfo':
  117. // 我的-某产品详情
  118. handleGoodsInfo();
  119. break;
  120. case 'vmf/app/market/confirmGoodsInfo':
  121. handleConfirmGoodsInfo();
  122. break;
  123. case '/vmf/app/common/getInitData':
  124. handleInitData();
  125. break;
  126. case '/vmf/app/index/topBannerList':
  127. break;
  128. case '/vmf/app/index/bannerArticleList':
  129. break;
  130. case '/vmf/app/index/artList':
  131. // 首发产品
  132. handleFirstArtList();
  133. break;
  134. case '/vmf/app/index/firstArtInfo':
  135. // 首发产品详情,市场产品详情
  136. handleFirstArtInfo();
  137. break;
  138. case '/vmf/app/market/confirmArtInfo':
  139. // 首发产品购买信息确认页面
  140. break;
  141. case '/vmf/app/order/createFirstOrder':
  142. // 创建发售订单
  143. break;
  144. case '/vmf/app/index/littleMinerInfo':
  145. break;
  146. case '/vmf/app/article/category':
  147. break;
  148. case '/vmf/app/article/listByCategory':
  149. // 公告列表
  150. handleNoticeList();
  151. break;
  152. case '/vmf/app/article/info':
  153. // 公告详情
  154. handleNoticeInfo();
  155. break;
  156. case '/vmf/app/article/selectArticleByType':
  157. // 产品相关的历史公告
  158. break;
  159. case '/vmf/app/index/recommend':
  160. // 推荐
  161. break;
  162. case '/vmf/app/index/firstArtInfo':
  163. // 产品信息
  164. break;
  165. case '/vmf/app/user/batchRole':
  166. break;
  167. case '/vmf/app/market/albumList':
  168. // 市场分区专辑列表
  169. handleMarketAlbumList();
  170. break;
  171. case '/vmf/app/market/groupList':
  172. // 市场产品寄售列表信息
  173. hadnleMarketGroupList();
  174. break;
  175. case '/vmf/app/market/saleRecord':
  176. // 市场最近成交记录
  177. break;
  178. case '/vmf/app/market/artInfo':
  179. // 市场寄售时产品详情
  180. break;
  181. case '/vmf/app/market/verList':
  182. // 市场某产品寄售列表
  183. break;
  184. case '/vmf/app/order/entrustArtInfo':
  185. // 市场某产品委托限价范围
  186. break;
  187. case '/vmf/app/order/createEntrustOrder':
  188. // 创建委托订单
  189. break;
  190. case '/vmf/app/order/cancelEntrustOrder':
  191. // 取消委托订单
  192. break;
  193. case '/vmf/app/pay/cashier':
  194. // 支付收银台钱包选择列表
  195. break;
  196. case '/vmf/app/pay/getUrl':
  197. // 获取支付地址
  198. break;
  199. case '/vmf/app/pay/payResult':
  200. break;
  201. case '/vmf/app/synthesis/activityList':
  202. handleActivityList();
  203. break;
  204. case '/vmf/app/synthesis/convertGoodsList':
  205. handleCvtGoodsList();
  206. break;
  207. case '/vmf/app/synthesis/createConvertOrder':
  208. break;
  209. case '/vmf/app/synthesis/compositeGoodsList':
  210. handleCompositeGoodsList();
  211. break;
  212. case '/vmf/app/synthesis/createCompositeOrder':
  213. break;
  214. case 'vmf/app/order/list':
  215. // 订单列表
  216. handleOrderList();
  217. break;
  218. case '/vmf/app/order/info':
  219. // 订单详情
  220. handleOrderInfo();
  221. break;
  222. default:
  223. if (path.indexOf('/vmf/app/order/createFirstOrder') != -1) {
  224. handleCreateFirstOrder();
  225. }
  226. break;
  227. }
  228. }
  229. }
  230. function handleHeaders() {
  231. if ($request && $request.method != 'OPTIONS') {
  232. let headers = $request.headers;
  233. let newToken = headers['token'];
  234. let oldToken = magicJS.data.read(WuBianConstKey.Token, null);
  235. if (newToken) {
  236. if (newToken != oldToken) {
  237. magicJS.logger.info(`当前Token刷新到最新`);
  238. magicJS.notification.appendNotifyInfo(`🎉当前Token刷新到最新`);
  239. } else {
  240. magicJS.logger.info(`当前Token已经是最新无需更新`);
  241. }
  242. magicJS.data.write(WuBianConstKey.Token, newToken);
  243. } else {
  244. let newCookieStr = headers['Cookie'];
  245. if (newCookieStr) {
  246. let cookieDict = magicJS.parseCookies(newCookieStr);
  247. if (cookieDict['token'] != void 0) {
  248. newToken = ookieDict['token'];
  249. magicJS.data.write(WuBianConstKey.Token, newToken);
  250. magicJS.logger.info(`当前Token刷新到最新`);
  251. }
  252. }
  253. }
  254. }
  255. }
  256. function updateHeaders() {
  257. gToken = magicJS.data.read(WuBianConstKey.Token, '');
  258. gCommonHeaders['token'] = gToken;
  259. gCommonHeaders['cookie'] = `token=${gToken}`;
  260. }
  261. function getResponsePlainData() {
  262. let data = magicJS.getResponseBody();
  263. if (!data) {
  264. return;
  265. }
  266. try {
  267. return JSON.parse(data);
  268. } catch (err) {
  269. magicJS.logger.info(data);
  270. magicJS.logger.error(err);
  271. }
  272. }
  273. function getRequestPlainData() {
  274. let data = magicJS.getRequestBody();
  275. let reqData = null;
  276. if (data) {
  277. reqData = JSON.parse(data);
  278. } else {
  279. reqData = magicJS.parseQueryStr($request.url);
  280. }
  281. return reqData;
  282. }
  283. function getQueryPlainData() {
  284. let params = magicJS.parseQueryStr($request.url);
  285. return params;
  286. }
  287. function compareCookieStr(cookieStr1, cookieStr2) {
  288. let cookieDict1 = magicJS.parseCookies(cookieStr1);
  289. let cookieDict2 = magicJS.parseCookies(cookieStr2);
  290. let skipKeys = [];
  291. let retCode = 0;
  292. for (let key in cookieDict1) {
  293. if (!skipKeys.includes(key)) {
  294. if (cookieDict1[key] != void 0 && cookieDict2[key] != void 0 && cookieDict1[key] != cookieDict2[key]) {
  295. magicJS.logger.info(`${key}`);
  296. retCode = -1;
  297. break;
  298. }
  299. }
  300. }
  301. return retCode;
  302. }
  303. function appendCookieStr(newCookieStr) {
  304. let cookieStr = magicJS.data.read(WuBianConstKey.Cookie, null);
  305. let cookieDict = magicJS.parseCookies(cookieStr);
  306. let newCookieDict = magicJS.parseCookies(newCookieStr);
  307. for (let key in newCookieDict) {
  308. cookieDict[key] = newCookieDict[key];
  309. }
  310. cookieStr = magicJS.serializeCookies(cookieDict);
  311. magicJS.data.write(WuBianConstKey.Cookie, cookieStr);
  312. }
  313. /**
  314. * 登录后获取用户信息,然后持久化保存在本地
  315. * @returns
  316. */
  317. function handleMobileLogin() {
  318. let rspData = getResponsePlainData();
  319. if (!rspData) {
  320. return;
  321. }
  322. if (rspData.code != 200) {
  323. return;
  324. }
  325. /*
  326. // 结果示例:
  327. let result = {
  328. "code": 200,
  329. "data": {
  330. "rankInfoHide": 0,
  331. "hashId": "80n19034n1",
  332. "cancellation": 0,
  333. "mobile": "134*****115",
  334. "isAuth": 1,
  335. "isSecurityPassword": 1,
  336. "tokenExpireTime": 1728625959,
  337. "avatar": "/assets/img/avatar01.png",
  338. "token": "bccafa59-d04c-48c9-82f6-08829155b2c6",
  339. "address": "0x3dc7d16739d0f169977f589788007c33a0e8996d",
  340. "locksold": 0,
  341. "gameUserId": 0,
  342. "nickname": "WBXL-9115",
  343. "email": "",
  344. "score": 0,
  345. "lockown": 0
  346. },
  347. "msg": "成功",
  348. "fail": false
  349. }
  350. */
  351. magicJS.data.write(WuBianConstKey.UserInfo, rspData.data);
  352. if (isDataMaskingEnabled()) {
  353. rspData.data.nickname = '测试专用001';
  354. rspData.data.address = '0x0000000000000000000000000000000000000001';
  355. rspData.data.avatar = '/uploads/20240930/c8a5a0d2aa889057e2d79334b70ffd9d.png';
  356. gRetBody = rspData;
  357. }
  358. let token = rspData.data.token;
  359. let expireDate = new Date(rspData.data.tokenExpireTime * 1000);
  360. magicJS.data.write(WuBianConstKey.Token, token);
  361. magicJS.notification.appendNotifyInfo(`🎉当前Token刷新到最新,过期时间:${magicJS.formatDate(expireDate, 'yyyy-MM-dd HH:mm:ss')}`);
  362. }
  363. function handleUserCollectList() {
  364. let rspData = getResponsePlainData();
  365. if (!rspData) {
  366. return;
  367. }
  368. if (rspData.code != 200) {
  369. return;
  370. }
  371. handleHeaders();
  372. let reqData = getRequestPlainData();
  373. if (reqData.state == 1) {//1=收藏中 2=转让中 3=已转让
  374. let artIdNameMapData = magicJS.data.read(WuBianConstKey.ArtIdNameMapData, {});
  375. let realCntDict = getMyAllArtRealCountData();
  376. let modCntDict = getAllArtModifyCountDict();
  377. let list = rspData.data.list;
  378. for (let i = list.length - 1; i >= 0; i--) {
  379. const artInfo = list[i];
  380. artIdNameMapData[artInfo.artId] = artInfo.title;
  381. realCntDict[artInfo.title] = artInfo.total;
  382. }
  383. magicJS.data.write(WuBianConstKey.ArtIdNameMapData, artIdNameMapData);
  384. if (isMyArtCountCaptureEnabled()) {
  385. for (let i = list.length - 1; i >= 0; i--) {
  386. const artInfo = list[i];
  387. realCntDict[artInfo.title] = artInfo.total;
  388. }
  389. setAllArtRealCountDict(realCntDict);
  390. }
  391. if (isMyArtCountModifyMode()) {
  392. for (let i = list.length - 1; i >= 0; i--) {
  393. const artInfo = list[i];
  394. checkArticleModify(artInfo, modCntDict);
  395. if (artInfo.total == 0) {
  396. list.splice(i, 1);
  397. modCntDict[artInfo.title] = 0;
  398. }
  399. }
  400. gRetBody = rspData;
  401. setAllArtModifyCountDict(modCntDict);
  402. }
  403. }
  404. }
  405. function handleVerList() {
  406. let rspData = getResponsePlainData();
  407. if (!rspData) {
  408. return;
  409. }
  410. if (rspData.code != 200) {
  411. return;
  412. }
  413. if (isMyArtCountModifyMode()) {
  414. let reqData = getRequestPlainData();
  415. let artHashId = reqData.artHashId;
  416. let count = getArtModifyCountByHashId(artHashId);
  417. if (count != void 0) {
  418. rspData.data.total = count;
  419. gRetBody = rspData;
  420. }
  421. if (isDataMaskingEnabled()) {
  422. let page = reqData.page;
  423. let verListMap = magicJS.data.read(WuBianConstKey.VerListMaskkingDataMap, {});
  424. if (page == 1) {
  425. verListMap = {};
  426. }
  427. let verList = rspData.data.list;
  428. for (let i = 0; i < verList.length; i++) {
  429. let verInfo = verList[i];
  430. let maskStr = desensitizeNumber(verInfo.ver);
  431. if (!verListMap[maskStr]) {
  432. verListMap[maskStr] = [];
  433. }
  434. verListMap[maskStr].push(verInfo.ver);
  435. verInfo.ver = maskStr;
  436. }
  437. magicJS.data.write(WuBianConstKey.VerListMaskkingDataMap, verListMap);
  438. gRetBody = rspData;
  439. }
  440. }
  441. }
  442. function desensitizeNumber(number, maskChar = '*', leastPadCount = 3) {
  443. // 将数字转换为字符串
  444. let str = number.toString();
  445. // 保留最后一位,其他位用脱敏字符替换
  446. let repCount = Math.max(leastPadCount, str.length - 1);
  447. return maskChar.repeat(repCount) + str[str.length - 1];
  448. }
  449. function getVerNumByMaskStr(maskStr) {
  450. let verListMap = magicJS.data.read(WuBianConstKey.VerListMaskkingDataMap, {});
  451. let arr = verListMap[maskStr];
  452. if (arr && arr.length > 0) {
  453. return getRandomElement(arr);
  454. }
  455. return maskStr;
  456. }
  457. function getRandomElement(arr) {
  458. // 生成一个随机索引
  459. const randomIndex = Math.floor(Math.random() * arr.length);
  460. // 返回随机索引处的元素
  461. return arr[randomIndex];
  462. }
  463. function getArtNameByHashId(hashId) {
  464. let artIdNameMapData = magicJS.data.read(WuBianConstKey.ArtIdNameMapData, {});
  465. return artIdNameMapData[hashId];
  466. }
  467. function getAllArtModifyCountDict() {
  468. let dict = {};
  469. let str = magicJS.data.read(WuBianConstKey.MyAllArtModifyCountData, '');
  470. if (str && str.trim().length > 0) {
  471. let lineArr = str.split('\n');
  472. for (let i = 0; i < lineArr.length; i++) {
  473. let kvArr = lineArr[i].trim().split('=');
  474. let name = kvArr[0].trim();
  475. if (kvArr[1]) {
  476. let count = parseInt(kvArr[1].trim());
  477. dict[name] = count;
  478. }
  479. }
  480. }
  481. return dict;
  482. }
  483. function setAllArtModifyCountDict(dict) {
  484. let str = '';
  485. for (let name in dict) {
  486. str += `${name}=${dict[name]}\n`;
  487. }
  488. magicJS.data.write(WuBianConstKey.MyAllArtModifyCountData, str);
  489. }
  490. function getArtModifyCountByName(name) {
  491. let dict = getAllArtModifyCountDict();
  492. return dict[name];
  493. }
  494. function getArtModifyCountByHashId(hashId) {
  495. let name = getArtNameByHashId(hashId);
  496. if (!name) {
  497. return null;
  498. }
  499. return getArtModifyCountByName(name);
  500. }
  501. function checkArticleModify(artInfo, modifyCntDict) {
  502. let title = artInfo.title;
  503. if (modifyCntDict[title] != void 0) {
  504. artInfo.total = modifyCntDict[title];
  505. }
  506. }
  507. function getMyAllArtRealCountData() {
  508. let dict = {};
  509. let str = magicJS.data.read(WuBianConstKey.MyAllArtRealCountData, '');
  510. if (str && str.trim().length > 0) {
  511. let lineArr = str.split('\n');
  512. for (let i = 0; i < lineArr.length; i++) {
  513. let kvArr = lineArr[i].trim().split('=');
  514. let name = kvArr[0].trim();
  515. if (kvArr[1]) {
  516. let count = parseInt(kvArr[1].trim());
  517. dict[name] = count;
  518. }
  519. }
  520. }
  521. return dict;
  522. }
  523. function setAllArtRealCountDict(dict) {
  524. let str = '';
  525. for (let name in dict) {
  526. str += `${name}=${dict[name]}\n`;
  527. }
  528. magicJS.data.write(WuBianConstKey.MyAllArtRealCountData, str);
  529. }
  530. function handleInitData() {
  531. let rspData = getResponsePlainData();
  532. if (!rspData) {
  533. return;
  534. }
  535. if (rspData.code != 200) {
  536. return;
  537. }
  538. handleHeaders();
  539. }
  540. function handleNoticeList() {
  541. let rspData = getResponsePlainData();
  542. if (!rspData) {
  543. return;
  544. }
  545. if (rspData.code != 200) {
  546. return;
  547. }
  548. let reqData = getRequestPlainData();
  549. let tenantId = reqData.tenantId;
  550. magicJS.data.write(WuBianConstKey.TenantId, tenantId);
  551. if (reqData.page == 1) {
  552. let oneNotice = {
  553. "hashId": "468xab6e",
  554. "category": "系统公告",
  555. "icon": "/uploads/icon/系统公告.png",
  556. "title": "【神秘空间】应有尽有等你来探索!",
  557. "image": "",
  558. "createdTime": magicJS.formatDate(new Date(), 'yyyy-MM-dd HH:mm:ss'),
  559. "type": 0,
  560. "categoryType": 6,
  561. "url": ""
  562. };
  563. rspData.data.list.splice(1, 0, oneNotice);
  564. gRetBody = rspData;
  565. }
  566. }
  567. function handleNoticeInfo() {
  568. let rspData = getResponsePlainData();
  569. if (!rspData) {
  570. return;
  571. }
  572. let reqData = getRequestPlainData();
  573. let hashId = reqData.hashId;
  574. if (hashId == '468xab6e') {
  575. let marketAlbumList = magicJS.data.read(WuBianConstKey.MarketAlbumListData, []);
  576. let marketGroupLstDict = magicJS.data.read(WuBianConstKey.AllMarketGroupListData, {});
  577. let baseLstDict = [];
  578. let relationLstDict = [];
  579. let tempId = 1;
  580. for (let artId in marketGroupLstDict) {
  581. const artInfo = marketGroupLstDict[artId];
  582. const groupName = artInfo.groupName;
  583. const artInfoA = {
  584. sortId: artInfo.sortId,
  585. hash_id: artInfo.artId,
  586. id: tempId,
  587. image: artInfo.image,
  588. title: artInfo.title,
  589. event: "art",
  590. type: { syn: "合成兑换", act: "空投", art: "藏品", pri: "抽奖", article: "公告" },
  591. };
  592. if (!baseLstDict[groupName]) {
  593. baseLstDict[groupName] = [];
  594. }
  595. baseLstDict[groupName].push(artInfoA);
  596. tempId += 1;
  597. const artInfoB = {
  598. sortId: artInfo.sortId,
  599. jumpType: "",
  600. url: "",
  601. timeList: [],
  602. title: artInfo.onSale == 1 ? artInfo.title : artInfo.title + '(停售)',
  603. image: artInfo.image,
  604. activityType: "",
  605. authorName: artInfo.brandName,
  606. type: "art",
  607. typeName: "",
  608. categoryType: "",
  609. relationId: artInfo.artId,
  610. };
  611. if (!relationLstDict[groupName]) {
  612. relationLstDict[groupName] = [];
  613. }
  614. relationLstDict[groupName].push(artInfoB);
  615. }
  616. let artLstText = ``;
  617. let relationAllList = [];
  618. for (let t = 0; t < marketAlbumList.length; t++) {
  619. const albumInfo = marketAlbumList[t];
  620. const albumName = albumInfo.name;
  621. if (albumName.indexOf('推荐') >= 0) {
  622. continue;
  623. }
  624. artLstText += `<p><strong><span style="color:#E53333;">&nbsp; &nbsp; <span style="color:#000000;">${albumName}:</span></span></strong></p>`;
  625. let artLstSubText = ``;
  626. let baseList = baseLstDict[albumName] || [];
  627. let relationList = relationLstDict[albumName] || [];
  628. baseList.sort((a, b) => a.sortId - b.sortId);
  629. relationList.sort((a, b) => a.sortId - b.sortId);
  630. relationAllList = relationAllList.concat(relationList);
  631. for (let i = 0; i < baseList.length; ++i) {
  632. let element = baseList[i];
  633. let dataStr = JSON.stringify(element);
  634. dataStr = dataStr.replace(/"/g, "&quot;");
  635. let tmpStr = `<input ` +
  636. `style="align-content:center;width:80%;height:35px;background:#18bc9c;color:#FFFFFF;" readonly="readonly" ` +
  637. `disabled="disabled" ` +
  638. `name="relation['art'][]" ` +
  639. `value="[显示]藏品:${element.title}" ` +
  640. `data-event="art" ` +
  641. `data-id="${element.hash_id}" ` +
  642. `data-show="1" ` +
  643. `data="${dataStr}" />`;
  644. artLstSubText += tmpStr;
  645. };
  646. artLstText += `<p><strong><span style="color:#E53333;">${artLstSubText}<br /></span></strong></p>`;
  647. }
  648. // magicJS.logger.info(artLstText);
  649. let dateStr = magicJS.formatDate(new Date(), 'yyyy年M月d日');
  650. let content = `
  651. <p><br /></p>
  652. <p style="white-space:normal;"><strong>以下过往开放过的藏品:</strong></p>
  653. <p><br /></p>
  654. ${artLstText}
  655. <p><strong><span style="color:#E53333;"><br /></span></strong></p>
  656. <p><strong><span style="color:#E53333;"><br /></span></strong></p>
  657. <p><strong><span style="color:#E53333;">&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</span></strong></p>
  658. <p style="white-space:normal;text-align:right;"><strong><b>无边星链运营团队</b></strong></p>
  659. <p style="white-space:normal;text-align:right;"><strong><b><br /></b></strong></p>
  660. <p style="white-space:normal;text-align:right;"><strong><b>${dateStr}</b></strong></p>
  661. <p><br /></p>
  662. `;
  663. dateStr = magicJS.formatDate(new Date(), 'yyyy-MM-dd HH:mm:ss');
  664. rspData.data.createdTime = dateStr;
  665. rspData.data.content = content;
  666. rspData.data.relationList = relationAllList;
  667. rspData.data.title = '欢迎来到神秘空间';
  668. gRetBody = rspData;
  669. }
  670. }
  671. function handleMarketAlbumList() {
  672. let rspData = getResponsePlainData();
  673. if (!rspData) {
  674. return;
  675. }
  676. if (rspData.code != 200) {
  677. return;
  678. }
  679. if (rspData.data && rspData.data.length > 0) {
  680. magicJS.data.write(WuBianConstKey.MarketAlbumListData, rspData.data);
  681. }
  682. }
  683. function hadnleMarketGroupList() {
  684. let rspData = getResponsePlainData();
  685. if (!rspData) {
  686. return;
  687. }
  688. if (rspData.code != 200) {
  689. return;
  690. }
  691. let reqData = getRequestPlainData();
  692. let dataDict = magicJS.data.read(WuBianConstKey.AllMarketGroupListData, {});
  693. let sortId = (reqData.page - 1) * 20 + 1;
  694. let list = rspData.data?.list || [];
  695. let groupName = '';
  696. let nowTime = Date.now();
  697. let artIdNameMapData = magicJS.data.read(WuBianConstKey.ArtIdNameMapData, {});
  698. for (let i = 0; i < list.length; ++i) {
  699. let item = list[i];
  700. dataDict[item.artId] = item;
  701. item.sortId = sortId;
  702. item.onSale = 1;
  703. item.updateTime = nowTime;
  704. sortId += 1;
  705. groupName = item.groupName;
  706. artIdNameMapData[item.artId] = item.title;
  707. }
  708. magicJS.data.write(WuBianConstKey.ArtIdNameMapData, artIdNameMapData);
  709. for (let artId in dataDict) {
  710. let item = dataDict[artId];
  711. if (item.updateTime == void 0) {
  712. item.updateTime = 0;
  713. }
  714. if (item.sortId == void 0) {
  715. item.sortId = 9999;
  716. }
  717. if (item.updateTime < nowTime - 2 * 24 * 3600 * 1000) {
  718. item.onSale = 0;
  719. item.sortId = 9999;
  720. } else {
  721. item.onSale = 1;
  722. }
  723. }
  724. magicJS.data.write(WuBianConstKey.AllMarketGroupListData, dataDict);
  725. }
  726. function handleActivityList() {
  727. let rspData = getResponsePlainData();
  728. if (!rspData) {
  729. return;
  730. }
  731. if (rspData.code != 200) {
  732. return;
  733. }
  734. let reqData = getRequestPlainData();
  735. // 1=合成活动,2=置换活动,3=分解活动,4=兑换活动
  736. let activityType = reqData.activityType || 0;
  737. let page = reqData.page;
  738. let key = `${WuBianConstKey.ActivityListData}#${activityType}`;
  739. let list = rspData.data.list;
  740. let dataDict = magicJS.data.read(key, {});
  741. let nowTime = Math.floor(Date.now() / 1000);
  742. for (let item of list) {
  743. // name
  744. // startTime
  745. // endTime
  746. // hashId
  747. // activityType
  748. // type 0=合成 1=兑换或置换
  749. // status -1=未开始 1=进行中 2=已结束
  750. const hashId = item.hashId;
  751. dataDict[hashId] = item;
  752. // item.startTime = Math.floor(Date.now() / 1000 - 3600);
  753. // item.endTime = Math.floor(Date.now() / 1000 + 3600);
  754. if (nowTime < item.endTime + 24 * 3600) {
  755. if (item.status == -1) {
  756. item.name = `${item.name}(未开始)`;
  757. } else if (item.status == 2) {
  758. item.name = `${item.name}(已结束)`;
  759. item.endTime = item.endTime + 24 * 3600;
  760. item.status = 1;
  761. }
  762. }
  763. }
  764. magicJS.data.write(key, dataDict);
  765. gRetBody = rspData;
  766. }
  767. function isActivityTaskCaptureEnabled() {
  768. let isEnabled = magicJS.data.read(WuBianConstKey.ActivityTaskCaptureEnabled, true);
  769. return isEnabled;
  770. }
  771. function isFirstOrderCaptureEnabled() {
  772. let isEnabled = magicJS.data.read(WuBianConstKey.FirstOrderCaptureEnabled, true);
  773. return isEnabled;
  774. }
  775. function isConsignmentCaptureEnabled() {
  776. let isEnabled = magicJS.data.read(WuBianConstKey.ConsignmentCaptureEnabled, true);
  777. return isEnabled;
  778. }
  779. function isFastBuyCaptureEnabled() {
  780. let isEnabled = magicJS.data.read(WuBianConstKey.FastBuyCaptureEnabled, true);
  781. return isEnabled;
  782. }
  783. function setActivityDisabled(activityId, bDisabled = true) {
  784. let key = `${WuBianConstKey.SynthesisActivityDisabled}#${activityId}`;
  785. magicJS.data.write(key, bDisabled);
  786. }
  787. function isMyArtCountCaptureEnabled() {
  788. let isEnabled = magicJS.data.read(WuBianConstKey.MyArtCountCaptureEnabled, true);
  789. return isEnabled;
  790. }
  791. function isMyArtCountModifyMode() {
  792. let isEnabled = magicJS.data.read(WuBianConstKey.MyArtCountModifyMode, false);
  793. return isEnabled;
  794. }
  795. function isDataMaskingEnabled() {
  796. let isEnabled = magicJS.data.read(WuBianConstKey.DataMaskingEnabled, true);
  797. return isEnabled;
  798. }
  799. function handleCvtGoodsList() {
  800. let rspData = getResponsePlainData();
  801. if (!rspData) {
  802. return;
  803. }
  804. if (rspData.code != 200) {
  805. return;
  806. }
  807. let reqData = getRequestPlainData();
  808. let activityId = reqData.activityId || 'default';
  809. if (isActivityTaskCaptureEnabled()) {
  810. let activityInfo = rspData.data.activityInfo;
  811. activityId = activityInfo.activityId;
  812. magicJS.data.write(WuBianConstKey.ActivityId, activityId);
  813. magicJS.data.write(WuBianConstKey.ActivityIndexName, activityInfo.name);
  814. magicJS.data.write(WuBianConstKey.SynthesisAmount, 10);
  815. activityInfo.type = 1;
  816. setActivityDisabled(activityId, false);
  817. magicJS.notification.appendNotifyInfo(`🎉[${activityInfo.name}]兑换数据采集成功!`);
  818. }
  819. let key = `${WuBianConstKey.ActivityGoodsList}#${activityId}`;
  820. magicJS.data.write(key, rspData.data);
  821. }
  822. function handleCompositeGoodsList() {
  823. let rspData = getResponsePlainData();
  824. if (!rspData) {
  825. return;
  826. }
  827. if (rspData.code != 200) {
  828. return;
  829. }
  830. let reqData = getRequestPlainData();
  831. let activityId = reqData.id || 'default';
  832. if (isActivityTaskCaptureEnabled()) {
  833. let activityInfo = rspData.data.activityInfo;
  834. activityId = activityInfo.activityId;
  835. magicJS.data.write(WuBianConstKey.ActivityId, activityId);
  836. magicJS.data.write(WuBianConstKey.ActivityIndexName, activityInfo.name);
  837. magicJS.data.write(WuBianConstKey.SynthesisAmount, 10);
  838. activityInfo.type = 0;
  839. setActivityDisabled(activityId, false);
  840. magicJS.notification.appendNotifyInfo(`🎉[${activityInfo.name}]合成数据采集成功!`);
  841. }
  842. let key = `${WuBianConstKey.ActivityGoodsList}#${activityId}`;
  843. magicJS.data.write(key, rspData.data);
  844. }
  845. function handleFirstArtList() {
  846. let rspData = getResponsePlainData();
  847. if (!rspData) {
  848. return;
  849. }
  850. if (rspData.code != 200) {
  851. return;
  852. }
  853. // 结果示例
  854. /*
  855. let result = {
  856. "code": 200,
  857. "data": {
  858. "total": 1,
  859. "next": 0,
  860. "list": [
  861. {
  862. "category": 1,
  863. "coverImage": "",
  864. "refreshTime": 1725881400,
  865. "count": 10000,
  866. "type": 1,
  867. "title": "星链计划",
  868. "avatar": "/uploads/20240830/c18be0bb7951859b97a7a2afba9d33a2.jpg",
  869. "image": "/uploads/20240909/0c5d20baa4c00a3390b131b84bf8d592.jpg",
  870. "tagList": [
  871. ],
  872. "price": 138,
  873. "attachFileType": 1,
  874. "artId": "3oe8bev8",
  875. "nickname": "无边星链",
  876. "status": "1" // 1=进行中 2=已结束
  877. }
  878. ]
  879. },
  880. "msg": "成功",
  881. "fail": false
  882. }
  883. */
  884. // let result = {
  885. // "code": 200,
  886. // "data": {
  887. // "total": 1,
  888. // "next": 0,
  889. // "list": [
  890. // {
  891. // "category": 1,
  892. // "coverImage": "",
  893. // "refreshTime": 1725881400,
  894. // "count": 10000,
  895. // "type": 1,
  896. // "title": "星链计划",
  897. // "avatar": "/uploads/20240830/c18be0bb7951859b97a7a2afba9d33a2.jpg",
  898. // "image": "/uploads/20240909/0c5d20baa4c00a3390b131b84bf8d592.jpg",
  899. // "tagList": [
  900. // ],
  901. // "price": 138,
  902. // "attachFileType": 1,
  903. // "artId": "3oe8bev8",
  904. // "nickname": "无边星链",
  905. // "status": "1"
  906. // }
  907. // ]
  908. // },
  909. // "msg": "成功",
  910. // "fail": false
  911. // }
  912. // rspData = result;
  913. let artDataDict = magicJS.data.read(WuBianConstKey.FirstArtListData, {});
  914. let reqData = getRequestPlainData();
  915. if (reqData && reqData.page == 1) {
  916. artDataDict = {};
  917. }
  918. let artList = rspData.data.list;
  919. for (let item of artList) {
  920. const artId = item.artId;
  921. artDataDict[artId] = item;
  922. }
  923. magicJS.data.write(WuBianConstKey.FirstArtListData, artDataDict);
  924. gRetBody = rspData;
  925. }
  926. function getFirstArtData(artId) {
  927. let artDataDict = magicJS.data.read(WuBianConstKey.FirstArtListData, {});
  928. return artDataDict[artId];
  929. }
  930. function getFirstArtName(artId) {
  931. let data = getFirstArtData(artId);
  932. if (data) {
  933. return data.title;
  934. }
  935. return `${artId}`;
  936. }
  937. function checkIsValidFirstArt(artId) {
  938. let artData = getFirstArtData(artId);
  939. if (artData) {
  940. if (artData.status == 1) {
  941. return true;
  942. }
  943. }
  944. return false;
  945. }
  946. function handleGoodsInfo() {
  947. let rspData = getResponsePlainData();
  948. if (!rspData) {
  949. return;
  950. }
  951. if (rspData.code != 200) {
  952. return;
  953. }
  954. /* 结果示例:
  955. let result = {
  956. "code": 200,
  957. "data": {
  958. "giveInfo": "",
  959. "resellInfo": "",
  960. "author": {
  961. "avatar": "/uploads/20240830/c18be0bb7951859b97a7a2afba9d33a2.jpg",
  962. "address": "0x96d1e7792e13c655b311e68e71d56698346786ce",
  963. "isAuthor": 1,
  964. "id": "657v3oean1",
  965. "bio": "",
  966. "isFocus": "",
  967. "nickName": "无边星链"
  968. },
  969. "owner": {
  970. "avatar": "/assets/img/avatar01.png",
  971. "address": "0x3dc7d16739d0f169977f589788007c33a0e8996d",
  972. "isAuthor": 0,
  973. "id": "80n19034n1",
  974. "bio": "",
  975. "isFocus": "",
  976. "nickName": "******5"
  977. },
  978. "button": [
  979. {
  980. "key": "NONE",
  981. "disable": 1,
  982. "highlight": ""
  983. }
  984. ],
  985. "isBox": 0,
  986. "paymentInfo": {
  987. "walletType": "",
  988. "isBuy": 0,
  989. "price": 48,
  990. "serviceRate": "0.07",
  991. "saleMinPrice": "",
  992. "saleMaxPrice": "",
  993. "mainOrderSn": "",
  994. "beginSaleTime": "",
  995. "walletId": ""
  996. },
  997. "info": {
  998. "lock": false,
  999. "tokenId": "182894170004127",
  1000. "status": 2,
  1001. "title": "星链计划",
  1002. "image": "/uploads/20240909/0c5d20baa4c00a3390b131b84bf8d592.jpg",
  1003. "count": 10000,
  1004. "images": "",
  1005. "contractAddress": "0x3a7ddb3129c52e8304ba1124a6d5c5f6622d8038",
  1006. "type": 1,
  1007. "brandName": "无边星链",
  1008. "id": "e3nl20k41n4d",
  1009. "ver": 4127,
  1010. "attachFileType": 1,
  1011. "circulationNum": 7500,
  1012. "groupName": "限价区",
  1013. "json": "{\"entrustEnable\":\"1\"}",
  1014. "artId": "3oe8bev8",
  1015. "isLike": 0,
  1016. "attachFile": "",
  1017. "content": "在浩渺的宇宙中,人类的探索从未停止。【星链计划】便是对人类太空探索梦想的独特诠释。这件藏品以精湛的工艺呈现出星链在太空中的壮丽景象。神秘深邃的背景中,无数颗卫星如同璀璨的明珠,以特定的轨道排列,仿佛一条闪耀着科技之光的银河。每一颗卫星都刻画得细致入微,金属质感的外壳散发着冷冽的光泽,彰显着现代科技的强大魅力。星链计划不仅代表着在宇宙领域的重大突破,更是人类向太空进军的勇敢尝试,也是对人类无限创造力的致敬。它不仅具有极高的艺术价值,更是历史与未来的交汇点,让你在欣赏它的同时,仿佛能感受到宇宙的浩瀚与人类的伟大。",
  1018. "description": ""
  1019. }
  1020. },
  1021. "msg": "成功",
  1022. "fail": false
  1023. }
  1024. */
  1025. let artInfo = rspData.data.info;
  1026. let paymentInfo = rspData.data.paymentInfo;
  1027. if (isConsignmentCaptureEnabled()) {
  1028. let oldArtId = magicJS.data.read(WuBianConstKey.ConsignmentArtId, null);
  1029. if (oldArtId != artInfo.artId) {
  1030. if (typeof paymentInfo.saleMaxPrice == 'number') {
  1031. magicJS.data.write(WuBianConstKey.ConsignmentArtPrice, paymentInfo.saleMaxPrice);
  1032. } else {
  1033. if (paymentInfo.price > 0) {
  1034. magicJS.data.write(WuBianConstKey.ConsignmentArtPrice, paymentInfo.price);
  1035. } else {
  1036. magicJS.data.write(WuBianConstKey.ConsignmentArtPrice, 9999);
  1037. }
  1038. }
  1039. }
  1040. magicJS.data.write(WuBianConstKey.ConsignmentArtId, artInfo.artId);
  1041. magicJS.data.write(WuBianConstKey.ConsignmentArtName, artInfo.title);
  1042. magicJS.notification.appendNotifyInfo(`🎉[${artInfo.title}]一键寄售数据采集成功!`);
  1043. }
  1044. let authorInfo = rspData.data.author;
  1045. if (typeof paymentInfo.saleMinPrice == 'string' && paymentInfo.saleMinPrice.length == 0) {
  1046. paymentInfo.saleMinPrice = 1;
  1047. }
  1048. if (typeof paymentInfo.saleMaxPrice == 'string' && paymentInfo.saleMaxPrice.length == 0) {
  1049. paymentInfo.saleMaxPrice = 99999;
  1050. }
  1051. let buttons = rspData.data.button;
  1052. if (buttons[0].disable == 1) {
  1053. // let mapList = [
  1054. // ["DESTROY", "已销毁"],
  1055. // ["LOCKING", "锁定中"],
  1056. // ["OPENED", "已开启"],
  1057. // ["SOLD", "已售出"],
  1058. // ["PAY", "立即购买"],
  1059. // ["DRIVE", "车辆试驾"],
  1060. // ["AVATAR", "设为头像"],
  1061. // ["CLOSE_SALE", "取消寄售"],
  1062. // ["SALE", "平台寄售"],
  1063. // ["GIVE", "转赠好友"],
  1064. // ["OPEN_BOX", "打开宝箱"],
  1065. // ["SYNTHESISING", "合成中"],
  1066. // ["NONE", "暂未开放"],
  1067. // ["CONVERT", "兑换"],
  1068. // ["SHARE", "分享"],
  1069. // ["RESELL", "转售"],
  1070. // ];
  1071. let frtBtnKey = buttons[0]?.key;
  1072. if (frtBtnKey == void 0 || frtBtnKey == 'NONE') {
  1073. buttons[0].disable = 0;
  1074. buttons[0].key = 'SALE';
  1075. buttons[1] = {
  1076. disable: 0,
  1077. key: 'GIVE',
  1078. };
  1079. authorInfo.nickName = `${authorInfo.nickName}#改`;
  1080. }
  1081. }
  1082. let owner = rspData.data.owner;
  1083. if (owner) {
  1084. let userDataCache = magicJS.data.read(WuBianConstKey.AllUserSimpleInfo, {});
  1085. if (!userDataCache[owner.id]) {
  1086. userDataCache[owner.id] = {
  1087. nickName: owner.nickName,
  1088. hashId: owner.id,
  1089. address: owner.address,
  1090. };
  1091. }
  1092. }
  1093. gRetBody = rspData;
  1094. }
  1095. function handleConfirmGoodsInfo() {
  1096. let rspData = getResponsePlainData();
  1097. if (!rspData) {
  1098. return;
  1099. }
  1100. if (rspData.code != 200) {
  1101. return;
  1102. }
  1103. let owner = data.owner;
  1104. if (owner) {
  1105. let userDataCache = magicJS.data.read(WuBianConstKey.AllUserSimpleInfo, {});
  1106. if (!userDataCache[owner.id]) {
  1107. userDataCache[owner.id] = {
  1108. nickName: owner.nickName,
  1109. hashId: owner.id,
  1110. address: owner.address,
  1111. };
  1112. } else {
  1113. userDataCache[owner.id].nickName = owner.nickName;
  1114. userDataCache[owner.id].hashId = owner.id;
  1115. userDataCache[owner.id].address = owner.address;
  1116. }
  1117. }
  1118. }
  1119. function handleFirstArtInfo() {
  1120. let rspData = getResponsePlainData();
  1121. if (!rspData) {
  1122. return;
  1123. }
  1124. if (rspData.code != 200) {
  1125. return;
  1126. }
  1127. /* 结果示例:
  1128. let result = {
  1129. "code": 200,
  1130. "data": {
  1131. "artInfo": {
  1132. "description": "",
  1133. "brandName": "无边星链",
  1134. "preBookTime": 600,
  1135. "contractAddress": "0x3a7ddb3129c52e8304ba1124a6d5c5f6622d8038",
  1136. "json": "{\"entrustEnable\":\"1\"}",
  1137. "count": 10000,
  1138. "isLike": 0,
  1139. "type": 1,
  1140. "title": "星链计划",
  1141. "image": "/uploads/20240909/0c5d20baa4c00a3390b131b84bf8d592.jpg",
  1142. "btnContent": "",
  1143. "attachfile": "",
  1144. "attachfileType": 1,
  1145. "images": [
  1146. ],
  1147. "groupName": "限价区",
  1148. "artId": "3oe8bev8",
  1149. "circulationNum": 8723,
  1150. "status": 2,
  1151. "content": "在浩渺的宇宙中,人类的探索从未停止。【星链计划】便是对人类太空探索梦想的独特诠释。这件藏品以精湛的工艺呈现出星链在太空中的壮丽景象。神秘深邃的背景中,无数颗卫星如同璀璨的明珠,以特定的轨道排列,仿佛一条闪耀着科技之光的银河。每一颗卫星都刻画得细致入微,金属质感的外壳散发着冷冽的光泽,彰显着现代科技的强大魅力。星链计划不仅代表着在宇宙领域的重大突破,更是人类向太空进军的勇敢尝试,也是对人类无限创造力的致敬。它不仅具有极高的艺术价值,更是历史与未来的交汇点,让你在欣赏它的同时,仿佛能感受到宇宙的浩瀚与人类的伟大。"
  1152. },
  1153. "authorInfo": {
  1154. "avatar": "/uploads/20240830/c18be0bb7951859b97a7a2afba9d33a2.jpg",
  1155. "address": "0x96d1e7792e13c655b311e68e71d56698346786ce",
  1156. "isAuthor": 1,
  1157. "id": "657v3oean1",
  1158. "bio": "",
  1159. "isFocus": "",
  1160. "nickName": "无边星链"
  1161. },
  1162. "covertInfo": "",
  1163. "payInfo": {
  1164. "category": 1,
  1165. "enableConsignment": 0,
  1166. "price": 138,
  1167. "limited": 0,
  1168. "refreshTime": 0,
  1169. "btnContent": "立即购买",
  1170. "serviceRate": "0.07",
  1171. "saleMaxPrice": 438,
  1172. "saleMinPrice": 1
  1173. }
  1174. },
  1175. "msg": "成功",
  1176. "fail": false
  1177. }
  1178. */
  1179. let artInfo = rspData.data.artInfo;
  1180. let authorInfo = rspData.data.authorInfo;
  1181. let payInfo = rspData.data.payInfo;
  1182. if (checkIsValidFirstArt(artInfo.artId)) {
  1183. if (isFirstOrderCaptureEnabled()) {
  1184. magicJS.data.write(WuBianConstKey.CreateFirstOrderData, {
  1185. title: artInfo.title,
  1186. artId: artInfo.artId,
  1187. price: artInfo.price,
  1188. num: payInfo.limited,
  1189. });
  1190. magicJS.data.write(WuBianConstKey.FirstGoodsId, artInfo.artId);
  1191. magicJS.data.write(WuBianConstKey.FirstGoodsName, artInfo.title);
  1192. magicJS.notification.appendNotifyInfo(`🎉产品[${artInfo.title}]首发数据采集成功!`);
  1193. }
  1194. } else {
  1195. if (isFastBuyCaptureEnabled()) {
  1196. let oldArtId = magicJS.data.read(WuBianConstKey.FastBuyArtId, null);
  1197. if (oldArtId != artInfo.artId) {
  1198. magicJS.data.write(WuBianConstKey.FastBuyArtPrice, payInfo.saleMinPrice || 1);
  1199. }
  1200. magicJS.data.write(WuBianConstKey.FastBuyArtId, artInfo.artId);
  1201. magicJS.data.write(WuBianConstKey.FastBuyArtName, artInfo.title);
  1202. magicJS.data.write(WuBianConstKey.FastBuyArtInfo, artInfo);
  1203. magicJS.notification.appendNotifyInfo(`🎉产品[${artInfo.title}]快捷购买数据采集成功!`);
  1204. }
  1205. }
  1206. artInfo.title = `${artInfo.title}(流通:${artInfo.circulationNum})`;
  1207. // 购买寄售强开
  1208. if (payInfo.enableConsignment == 0) {
  1209. payInfo.enableConsignment = 1;
  1210. authorInfo.nickName = `${authorInfo.nickName}#改`;
  1211. magicJS.logger.info(`强制开启批量购买功能`);
  1212. }
  1213. // payInfo只跟首发相关
  1214. // 强制开启首发购买
  1215. // artInfo.status = 1;
  1216. if (payInfo.limited < 1) {
  1217. // 购买数量强开
  1218. payInfo.limited = 999;
  1219. magicJS.logger.info(`强制开启购买数量限制`);
  1220. }
  1221. if (payInfo.refreshTime == 0) {
  1222. payInfo.refreshTime = Math.floor(Date.now() / 1000);
  1223. }
  1224. gRetBody = rspData;
  1225. }
  1226. function handleCreateFirstOrder() {
  1227. let rspData = getResponsePlainData();
  1228. if (!rspData) {
  1229. return;
  1230. }
  1231. if (isFirstOrderCaptureEnabled()) {
  1232. // `https://api.wubian.pro/vmf/app/order/createFirstOrder?token=80n19034n1`
  1233. let reqData = JSON.parse(magicJS.getRequestBody());
  1234. let queryParams = magicJS.parseQueryStr($request.url);
  1235. magicJS.data.write(WuBianConstKey.CreateFirstOrderData, {
  1236. token: queryParams.token,
  1237. artId: reqData.artHashId,
  1238. price: reqData.price,
  1239. num: reqData.num,
  1240. });
  1241. magicJS.data.write(WuBianConstKey.FirstGoodsId, reqData.artHashId);
  1242. magicJS.notification.appendNotifyInfo(`🎉产品[${getFirstArtName(reqData.artHashId)}]首发数据采集成功!`);
  1243. }
  1244. }
  1245. function handleOrderList() {
  1246. let rspData = getResponsePlainData();
  1247. if (!rspData) {
  1248. return;
  1249. }
  1250. /**
  1251. *
  1252. 结果示例:
  1253. {
  1254. "code": 200,
  1255. "data": {
  1256. "total": 283,
  1257. "next": 1,
  1258. "list": [
  1259. {
  1260. "state": 8,
  1261. "orderTypeName": "兑换订单",
  1262. "orderSn": "20240919200738282206032433",
  1263. "createdTime": "2024-09-19 20:07:38",
  1264. "count": 82781,
  1265. "authorName": "无边星链",
  1266. "title": "玉兔捣药分解活动!",
  1267. "price": 0,
  1268. "image": "/uploads/20240919/ca5a67f98ed6396fd548752caa3d8974.png",
  1269. "ver": "#22314/82781",
  1270. "orderType": 6,
  1271. "num": 6,
  1272. "lockTime": 0,
  1273. "stateDesc": "已兑换",
  1274. "buttonList": []
  1275. }
  1276. ]
  1277. },
  1278. "msg": "成功",
  1279. "fail": false
  1280. }
  1281. */
  1282. // let mapList = [
  1283. // ["CANCEL_CONSIGNMENT", "取消寄售"],
  1284. // ["VIEW_GOODS", "查看藏品"],
  1285. // ["CANCEL_ORDER", "取消订单"],
  1286. // ["PAY", "去支付"],
  1287. // ["GOODS_LIBRARY", "藏品库"],
  1288. // ["BUY_ANOTHER", "继续购买"],
  1289. // ["ORDER_CONFIRM", "订单确认中"],
  1290. // ["INVOICE", "申请发票"],
  1291. // ["UPDATE_PRICE", "修改价格"],
  1292. // ["CANCEL_COMPETE", "取消竞价"],
  1293. // ["CANCEL_ENTRUST", "取消委托"]
  1294. // ];
  1295. }
  1296. function handleOrderInfo() {
  1297. let rspData = getResponsePlainData();
  1298. if (!rspData) {
  1299. return;
  1300. }
  1301. }
  1302. Main().catch((e) => magicJS.logger.log(`-\n ${e}`)).finally(() => magicJS.done());
  1303. //---SyncByPyScript---MagicJS3-start
  1304. function MagicJS(e = "MagicJS", t = "INFO") { const n = () => { const e = "undefined" != typeof $loon, t = "undefined" != typeof $task, r = "undefined" != typeof module, n = "undefined" != typeof $httpClient && !e, o = "undefined" != typeof $storm, i = "undefined" != typeof $environment && void 0 !== $environment["stash-build"]; var s = n || e || o || i; const a = "undefined" != typeof importModule; return { isLoon: e, isQuanX: t, isNode: r, isSurge: n, isStorm: o, isStash: i, isSurgeLike: s, isScriptable: a, get name() { return e ? "Loon" : t ? "QuantumultX" : r ? "NodeJS" : n ? "Surge" : a ? "Scriptable" : "unknown" }, get build() { return n ? $environment["surge-build"] : i ? $environment["stash-build"] : o ? $storm.buildVersion : void 0 }, get language() { if (n || i) return $environment.language }, get version() { return n ? $environment["surge-version"] : i ? $environment["stash-version"] : o ? $storm.appVersion : r ? process.version : void 0 }, get system() { return n ? $environment.system : r ? process.platform : void 0 }, get systemVersion() { if (o) return $storm.systemVersion }, get deviceName() { if (o) return $storm.deviceName } } }, o = (r, e = "INFO") => { let n = e, t = "\n"; const o = { SNIFFER: 6, DEBUG: 5, INFO: 4, NOTIFY: 3, WARNING: 2, ERROR: 1, CRITICAL: 0, NONE: -1 }, i = { SNIFFER: "", DEBUG: "", INFO: "", NOTIFY: "", WARNING: "❗ ", ERROR: "❌ ", CRITICAL: "❌ ", NONE: "" }, s = (e, t = "INFO") => { o[n] < o[t.toUpperCase()] || console.log(`██[${r}][${t}]` + i[t.toUpperCase()] + e + "\n") }; return { getLevel: () => n, setLevel: e => { n = e }, sniffer: (...e) => { e = e.join(t); s(e, "SNIFFER") }, log: (...e) => { e = e.join(t); console.log(`██[${r}]` + e + "\n") }, debug: (...e) => { e = e.join(t); s(e, "DEBUG") }, info: (...e) => { e = e.join(t); s(e, "INFO") }, notify: (...e) => { e = e.join(t); s(e, "NOTIFY") }, warning: (...e) => { e = e.join(t); s(e, "WARNING") }, error: (...e) => { e = e.join(t); s(e, "ERROR") }, retry: (...e) => { e = e.join(t); s(e, "RETRY") } } }; return new class { constructor(e, t) { var r; this._startTime = Date.now(), this.version = "3.0.0", this.scriptName = e, this.env = n(), this.logger = o(e, t), this.http = "function" == typeof MagicHttp ? MagicHttp(this.env, this.logger) : void 0, this.data = "function" == typeof MagicData ? MagicData(this.env, this.logger) : void 0, this.notification = "function" == typeof MagicNotification ? MagicNotification(this.scriptName, this.env, this.logger, this.http) : void 0, this.utils = "function" == typeof MagicUtils ? MagicUtils(this.env, this.logger) : void 0, this.qinglong = "function" == typeof MagicQingLong ? MagicQingLong(this.env, this.data, this.logger) : void 0, void 0 !== this.data && (t = this.data.read("magic_loglevel"), r = this.data.read("magic_bark_url"), t && this.logger.setLevel(t.toUpperCase()), r) && this.notification.setBark(r), this.logger.info(e + ", 开始执行!") } get isRequest() { return "undefined" != typeof $request } get isStrictRequest() { return "undefined" != typeof $request && "undefined" == typeof $response } get isResponse() { return "undefined" != typeof $response } get isDebug() { return "DEBUG" === this.logger.level } get request() { return "undefined" != typeof $request ? $request : void 0 } get response() { if ("undefined" != typeof $response) return $response.hasOwnProperty("status") && ($response.statusCode = $response.status), $response.hasOwnProperty("statusCode") && ($response.status = $response.statusCode), $response } log(...e) { this.logger.log(e) } toStr(e, t = null) { try { return JSON.stringify(e) } catch { return t } } toObj(e, t = null) { try { return JSON.parse(e) } catch { return t } } checkRecordRequestBody() { if (this.isRequest) { var t = $request.body; if (t) { var r = this.env, n = $request.path; let e = this.scriptName + "#" + n.replace("/", "_"); e = e.replace("?", "#"), r.isQuanX && $prefs.setValueForKey(t, e), (r.isLoon || r.isSurge) && $persistentStore.write(t, e), r.isNode && require("fs").writeFileSync(e + ".json", t, { flag: "w" }, e => console.log(e)) } } } getRequestBody() { var e = this.env, t = $request.path; let r = this.scriptName + "#" + t.replace("/", "_"); if (r = r.replace("?", "#"), e.isSurge || e.isLoon) return $persistentStore.read(r); if (e.isQuanX) return $prefs.valueForKey(r); if (e.isNode) { t = r + ".json", e = require("fs"); if (!e.existsSync(t)) return JSON.parse(e.readFileSync(t)) } } getResponseBody() { if ($response) return $response.body } parseCookies(e, t = !1) { let r = {}; return t ? e && e.split(";").forEach(function (e) { e = e.split("="); r[e.shift().trim()] = decodeURIComponent(e.join("=")) }) : e && e.split(";").forEach(function (e) { e = e.split("="); r[e.shift().trim()] = e.join("=") }), r } serializeCookies(e, t = !1) { var r = []; if (t) for (var n in e) { var o = e[n], n = n + "=" + encodeURIComponent(o); r.push(n) } else for (var i in e) { var s = e[i], i = i + "=" + s; r.push(i) } return r.join("; ") } parseSetCookies(e) { var t = e.split(/,\s*/).map(e => { var e = e.trim().split(/;\s*(?=[^=]+=[^;]*)/), t = e[0].split("=").map(e => e.trim()); const r = { name: t[0], value: t[1] }; return e.slice(1).forEach(e => { var [e, t] = e.split("=").map(e => e.trim()); r[e] = t || !0 }), r }), r = []; let n = 0; for (; n < t.length;) { var o, i = t[n]; r.push(i), i.Expires ? ((o = t[n + 1]) && (i.Expires = i.Expires + "," + o.name), n += 2) : n += 1 } return r } objToQueryStr(t, r) { let n = ""; for (const o in t) { let e = t[o]; null != e && "" !== e && ("object" == typeof e ? e = JSON.stringify(e) : r && (e = encodeURIComponent(e)), n += `${o}=${e}&`) } return n = n.substring(0, n.length - 1) } parseQueryStr(e) { var t = {}, r = (e = -1 < e.indexOf("?") ? e.split("?")[1] : e).split("&"); for (let e = 0; e < r.length; e++) { var n = r[e].split("="); t[n[0]] = n[1] } return t } deepClone(e, t) { for (var r in t = t || {}, e) "object" == typeof e[r] ? (t[r] = e[r].constructor === Array ? [] : {}, this.deepClone(e[r], t[r])) : t[r] = e[r]; return t } formatDate(e, t) { var r, n = { "M+": e.getMonth() + 1, "d+": e.getDate(), "H+": e.getHours(), "m+": e.getMinutes(), "s+": e.getSeconds(), "q+": Math.floor((e.getMonth() + 3) / 3), S: e.getMilliseconds() }; for (r in /(y+)/.test(t) && (t = t.replace(RegExp.$1, (e.getFullYear() + "").substr(4 - RegExp.$1.length))), n) new RegExp("(" + r + ")").test(t) && (t = t.replace(RegExp.$1, 1 == RegExp.$1.length ? n[r] : ("00" + n[r]).substr(("" + n[r]).length))); return t } parseDate(a, e) { let l = { y: 0, M: 1, d: 0, H: 0, h: 0, m: 0, s: 0, S: 0 }; (e = e || "yyyy-MM-dd").replace(/([^yMdHmsS]*?)(([yMdHmsS])\3*)([^yMdHmsS]*?)/g, function (e, t, r, n, o, i, s) { return a = a.replace(new RegExp(t + "(\\d{" + r.length + "})" + o), function (e, t) { return l[n] = parseInt(t), "" }), "" }), l.M--; e = new Date(l.y, l.M, l.d, l.H, l.m, l.s); return 0 !== l.S && e.setMilliseconds(l.S), e } getBaseDoneHeaders(e = {}) { 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" }, e) } 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" }) } doWxpusherSend(e) { var t = this.getJsonDoneHeaders(), t = (t.Host = "wxpusher.zjiecode.com", t["Content-Type"] = "application/json;charset=UTF-8", { url: "https://wxpusher.zjiecode.com/api/send/message", headers: t, body: JSON.stringify(e) }); return this.http.post(t) } fastWxpusherSend(e, t = "", r = "") { return this.doWxpusherSend({ appToken: "AT_7wDWqSoT8xpJCQqJtHpshKhw7kXc0XCW", content: e, summary: t, contentType: 1, topicIds: [], uids: ["UID_6P4B00X6Zv8U2oKC0I2R09emxtqq"], url: r, verifyPay: !1, verifyPayType: 0 }) } isEmpty(e) { return void 0 === e || null == e || "" == e || "null" == e || "undefined" == e || 0 === e.length } base64Encode(e) { var t, r, n, o = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; let i, s, a; for (a = e.length, s = 0, i = ""; s < a;) { if (t = 255 & e.charCodeAt(s++), s == a) { i = (i += o.charAt(t >> 2)) + o.charAt((3 & t) << 4) + "=="; break } if (r = e.charCodeAt(s++), s == a) { i = (i = (i += o.charAt(t >> 2)) + o.charAt((3 & t) << 4 | (240 & r) >> 4)) + o.charAt((15 & r) << 2) + "="; break } n = e.charCodeAt(s++), i = (i = (i = (i += o.charAt(t >> 2)) + o.charAt((3 & t) << 4 | (240 & r) >> 4)) + o.charAt((15 & r) << 2 | (192 & n) >> 6)) + o.charAt(63 & n) } return i } base64Decode(e) { var t = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/="; if (/([^\s]+[^0-9a-zA-Z\+\/\=]|[^0-9a-zA-Z\+\/\=]\s+)/.test(e)) throw new Error("Invalid base64 input"); var r, n, o, i, s, a, l = e.replace(/\s/g, ""); let u = "", c = 0; for (; c < l.length;)o = t.indexOf(l.charAt(c++)), r = (15 & (i = t.indexOf(l.charAt(c++)))) << 4 | (s = t.indexOf(l.charAt(c++))) >> 2, n = (3 & s) << 6 | (a = t.indexOf(l.charAt(c++))), u += String.fromCharCode(o << 2 | i >> 4), 64 !== s && (u += String.fromCharCode(r)), 64 !== a && (u += String.fromCharCode(n)); return u = this.utf8Decode(u) } utf8Decode(t) { let e = [], r = 0, n = 0, o = 0; for (t = t.replace(/\r\n/g, "\n"); r < t.length;) { n = 255 & t.charCodeAt(r), o = 0, o = n <= 191 ? (n &= 127, 1) : n <= 223 ? (n &= 31, 2) : n <= 239 ? (n &= 15, 3) : (n &= 7, 4); for (let e = 1; e < o; ++e)n = n << 6 | 63 & t.charCodeAt(e + r); 4 === o ? (n -= 65536, e.push(String.fromCharCode(55296 | n >> 10 & 1023)), e.push(String.fromCharCode(56320 | 1023 & n))) : e.push(String.fromCharCode(n)), r += o } return e.join("") } parseJwt(e) { try { var t = e.split("."), r = t[0].replace(/-/g, "+").replace(/_/g, "/"), n = this.base64Decode(r).replace(/\0/g, ""), o = JSON.parse(n), i = t[1].replace(/-/g, "+").replace(/_/g, "/"), s = this.base64Decode(i).replace(/\0/g, ""); return { header: o, payload: JSON.parse(s), signature: t[2] } } catch (e) { return this.log(e), null } } costTime() { var e = this.scriptName + "执行完毕!", t = (this._endTime = (new Date).getTime(), this._endTime - this._startTime); this.logger.info(e + `耗时【${t / 1e3}】秒`) } done = (e = {}) => { this.costTime(), "undefined" != typeof $done && $done(e) } }(e, t) } function MagicHttp(d, p) { var e; let g; d.isNode && (e = require("axios"), g = e.create()); class t { constructor(e = !0) { this.handlers = [], this.isRequest = e } use(e, t, r) { return "function" == typeof e && p.debug("Register fulfilled " + e.name), "function" == typeof t && p.debug("Register rejected " + t.name), this.handlers.push({ fulfilled: e, rejected: t, synchronous: !(!r || "boolean" != typeof r.synchronous) && r.synchronous, runWhen: r ? r.runWhen : null }), this.handlers.length - 1 } eject(e) { this.handlers[e] && (this.handlers[e] = null) } forEach(t) { this.handlers.forEach(e => { null !== e && t(e) }) } } function f(e) { let r = { ...e }; return r.params && !d.isNode && (e = Object.keys(r.params).map(e => { var t = encodeURIComponent(e); return r.url = r.url.replace(new RegExp(e + "=[^&]*", "ig"), ""), r.url = r.url.replace(new RegExp(t + "=[^&]*", "ig"), ""), t + "=" + encodeURIComponent(r.params[e]) }).join("&"), r.url.indexOf("?") < 0 && (r.url += "?"), /(&|\?)$/g.test(r.url) || (r.url += "&"), r.url += e, delete r.params, p.debug("Params to QueryString: " + r.url)), r } const h = (e, t = null) => { if (e) { t = { ...e, config: e.config || t, status: e.statusCode || e.status, body: e.body || e.data, headers: e.headers || e.header }; if ("string" == typeof t.body) try { t.body = JSON.parse(t.body) } catch { } return delete t.data, t } return e }; const y = (e, t = null) => { if (e && 400 <= e.status) return p.debug("Raise exception when status code is " + e.status), { name: "RequestException", message: "Request failed with status code " + e.status, config: t || e.config, response: e } }, m = { request: new t, response: new t(!1) }; let v = [], b = [], S = !0; function N(e) { return e = f(e), p.debug(`HTTP ${e.method.toUpperCase()}:` + "\n" + JSON.stringify(e)), e } function E(t) { try { t = t && h(t), p.sniffer(`HTTP ${t.config.method.toUpperCase()}:` + "\n" + JSON.stringify(t.config) + "\nSTATUS CODE:\n" + t.status + "\nRESPONSE:\n" + ("object" == typeof t.body ? JSON.stringify(t.body) : t.body)); var e = y(t); return e ? Promise.reject(e) : t } catch (e) { return p.error(e), t } } const r = (e, r) => { let n; r = ((e, t) => { let r = "object" == typeof t ? { headers: {}, ...t } : { url: t, headers: {} }; return r.method || (r.method = e), !0 === (r = f(r)).rewrite && (d.isSurge ? (r.headers["X-Surge-Skip-Scripting"] = !1, delete r.rewrite) : d.isQuanX && (r.hints = !1, delete r.rewrite)), d.isSurgeLike ? (t = r.headers["content-type"] || r.headers["Content-Type"], "GET" !== r.method && t && 0 <= t.indexOf("application/json") && r.body instanceof Array && (r.body = JSON.stringify(r.body), p.debug("Convert Array object to String: " + r.body))) : d.isQuanX ? (r.hasOwnProperty("body") && "string" != typeof r.body && (r.body = JSON.stringify(r.body)), r.method = e) : d.isNode && ("POST" === e || "PUT" === e || "PATCH" === e || "DELETE" === e ? r.data = r.data || r.body : "GET" === e && (r.params = r.params || r.body), delete r.body), r })(e.toUpperCase(), r), n = d.isNode ? g : d.isSurgeLike ? i => new Promise((n, o) => { $httpClient[e.toLowerCase()](i, (e, t, r) => { e ? (e = { name: e.name || e, message: e.message || e, stack: e.stack || e, config: i, response: h(t) }, o(e)) : (t.config = i, t.body = r, n(t)) }) }) : n => new Promise((r, t) => { $task.fetch(n).then(e => { e = h(e, n); var t = y(e, n); if (t) return Promise.reject(t); r(e) }).catch(e => { e = { name: e.message || e.error, message: e.message || e.error, stack: e.error, config: n, response: e.response ? h(e.response) : null }; t(e) }) }); let o; var t = r; try { v = [], b = [], m.request.forEach(e => { "function" == typeof e.runWhen && !1 === e.runWhen(t) || (S = S && e.synchronous, v.unshift(e.fulfilled, e.rejected)) }), m.response.forEach(e => { b.push(e.fulfilled, e.rejected) }) } catch (e) { p.error(`Failed to register interceptors: ${e}.`) } var i = [N, void 0], s = [E, void 0]; if (S) { for (p.debug("Interceptors are executed in synchronous mode"), Array.prototype.unshift.apply(v, i), v = v.concat([N, void 0]); v.length;) { var a = v.shift(), l = v.shift(); try { "function" == typeof a && p.debug("Executing request fulfilled " + a.name), r = a(r) } catch (e) { "function" == typeof l && p.debug("Executing request rejected " + l.name), l(e); break } } try { o = (!d.isNode && r.timeout ? c : n)(r) } catch (e) { return Promise.reject(e) } for (Array.prototype.unshift.apply(b, s); b.length;)o = o.then(b.shift(), b.shift()); return o } { p.debug("Interceptors are executed in asynchronous mode"); let t = [n, void 0]; for (Array.prototype.unshift.apply(t, i), Array.prototype.unshift.apply(t, v), t = (t = t.concat(s)).concat(b), o = Promise.resolve(r); t.length;)try { let e = t.shift(); var u = t.shift(); "function" == typeof (e = !d.isNode && r.timeout && e === n ? c : e) && p.debug("Executing request fulfilled " + e.name), "function" == typeof u && p.debug("Executing request rejected " + u.name), o = o.then(e, u) } catch (e) { p.error("request exception: " + e) } return o } function c(r) { try { var e = new Promise((e, t) => { setTimeout(() => { var e = { message: `timeout of ${r.timeout}ms exceeded.`, config: r }; t(e) }, r.timeout) }); return Promise.race([n(r), e]) } catch (e) { p.error(`Request Timeout exception: ${e}.`) } } }; return { request: r, interceptors: m, convertHeadersToLowerCase: r => Object.keys(r).reduce((e, t) => (e[t.toLowerCase()] = r[t], e), {}), convertHeadersToCamelCase: r => Object.keys(r).reduce((e, t) => { return e[t.split("-").map(e => e[0].toUpperCase() + e.slice(1)).join("-")] = r[t], e }, {}), modifyResponse: h, get: e => r("GET", e), post: e => r("POST", e), put: e => r("PUT", e), patch: e => r("PATCH", e), delete: e => r("DELETE", e), head: e => r("HEAD", e), options: e => r("OPTIONS", e) } } function MagicData(d, p) { let g = { fs: void 0, data: {} }; if (d.isNode) { g.fs = require("fs"); try { g.fs.accessSync("./magic.json", g.fs.constants.R_OK | g.fs.constants.W_OK) } catch (e) { g.fs.writeFileSync("./magic.json", "{}", { encoding: "utf8" }) } g.data = require("./magic.json") } const s = (e, t) => "object" != typeof t && e === t, a = e => "true" === e || "false" !== e && (void 0 === e ? null : e), l = (e, t, r, n) => { if (r) try { e = !0 === (e = "string" == typeof e ? JSON.parse(e) : e).magic_session ? e[r] : null } catch { e = null } if ("string" == typeof e && "null" !== e) try { e = JSON.parse(e) } catch { } return null == (e = !1 === n && e && !0 === e.magic_session ? null : e) && null != t && (e = t), e = a(e) }, f = t => { if ("string" != typeof t) return t instanceof Array || null == t || t != t || "boolean" == typeof t ? {} : t; { let e = {}; try { var r = typeof (e = JSON.parse(t)); ("object" != r || e instanceof Array || "bool" == r || null === e) && (e = {}) } catch { } return e } }, u = (e, t = null, r = "", n = !1, o = null) => { let i = ""; return i = o || d.isNode ? ((e, t = null, r = "", n = !1, o = null) => { o = o || g.data; return val = o && void 0 !== o[e] && null !== o[e] ? o[e] : r ? {} : null, val = l(val, t, r, n) })(e, t, r, n, o) : (d.isSurgeLike ? i = $persistentStore.read(e) : d.isQuanX && (i = $prefs.valueForKey(e)), l(i, t, r, n)), p.debug(`READ DATA [${e}]${r ? `[${r}]` : ""} <${typeof i}>` + "\n" + JSON.stringify(i)), i }, c = (e, t, r = "", n = null) => { if (void 0 === t || t != t) return !1; d.isNode || "boolean" != typeof t && "number" != typeof t || (t = String(t)); let o = ""; var i, s, a, l, u, c; if (n || d.isNode ? o = ([i, s, a = "", l = null] = [e, t, r, n], c = l || g.data, c = f(c), a ? ((u = f(c[i])).magic_session = !0, u[a] = s, c[i] = u) : c[i] = s, null !== l && (l = c), c) : r ? (d.isSurgeLike ? o = $persistentStore.read(e) ? $persistentStore.read(e) : o : d.isQuanX && (o = $prefs.valueForKey(e) ? $prefs.valueForKey(e) : o), (o = f(o)).magic_session = !0, o[r] = t) : o = t, o && "object" == typeof o && (o = JSON.stringify(o, null, 4)), p.debug(`WRITE DATA [${e}]${r ? `[${r}]` : ""} <${typeof t}>` + "\n" + JSON.stringify(t)), !n) { if (d.isSurgeLike) return $persistentStore.write(o, e); if (d.isQuanX) return $prefs.setValueForKey(o, e); if (d.isNode) try { g.fs.writeFileSync("./magic.json", o) } catch (e) { return p.error(e), !1 } } return !0 }; return { read: u, write: c, del: (e, t = "", r = null) => { let n = {}; if (r || d.isNode) n = (o = e, i = t, s = r || g.data, s = f(s), i ? (delete (obj = f(s[o]))[i], s[o] = obj) : delete s[o], s), r ? r = n : g.fs.writeFileSync("./magic.json", JSON.stringify(n, null, 4)); else if (t) { d.isSurgeLike ? n = $persistentStore.read(e) : d.isQuanX && (n = $prefs.valueForKey(e)), delete (n = f(n))[t]; i = JSON.stringify(n, null, 4); c(e, i) } else { if (d.isStorm) return $persistentStore.remove(e); if (d.isSurgeLike) return $persistentStore.write(null, e); if (d.isQuanX) return $prefs.removeValueForKey(e) } var o, i, s; p.debug(`DELETE KEY [${e}]` + (t ? `[${t}]` : "")) }, update: (e, t, r, n = s, o = null) => { var i; return t = a(t), !0 !== n(u(e, null, r, !1, o), t) && (i = c(e, t, r, o), e = u(e, null, r, !1, o), n === s && "object" == typeof e ? i : n(t, e)) }, allSessions: (e, t = null) => { let r = {}; t = u(e, null, null, !0, t); return !0 === (t = f(t)).magic_session && delete (r = { ...t }).magic_session, p.debug(`READ ALL SESSIONS [${e}] <${typeof r}>` + "\n" + JSON.stringify(r, null, 4)), r }, allSessionNames: (e, t = null) => { let r = []; t = u(e, null, null, !0, t), t = f(t); return r = !0 !== t.magic_session ? [] : Object.keys(t).filter(e => "magic_session" !== e), p.debug(`READ ALL SESSIONS [${e}] <${typeof r}>` + "\n" + JSON.stringify(r, null, 4)), r }, defaultValueComparator: s, convertToObject: f } } function MagicNotification(i, o, s, a) { let l = null, u = null, c = []; function d(e = i, t = "", r = "", n = "") { n = (t => { try { let e = {}; var r; return "string" == typeof t ? 0 < t.length && (o.isLoon ? e = { openUrl: t } : o.isQuanX ? e = { "open-url": t } : o.isSurge && (e = { url: t })) : "object" == typeof t && (o.isLoon ? (e.openUrl = t["open-url"] || "", e.mediaUrl = t["media-url"] || "") : o.isQuanX ? e = t["open-url"] || t["media-url"] ? t : {} : o.isSurge && (r = t["open-url"] || t.openUrl, e = r ? { url: r } : {})), e } catch (e) { s.error("通知选项转换失败" + e) } return t })(n), 1 === arguments.length && (e = i, t = "", r = arguments[0]), s.notify("\ntitle:" + e + "\nsubTitle:" + t + "\nbody:" + r + "\noptions:" + ("object" == typeof n ? JSON.stringify(n) : n)), o.isSurge ? $notification.post(e, t, r, n) : o.isLoon ? n ? $notification.post(e, t, r, n) : $notification.post(e, t, r) : o.isQuanX && $notify(e, t, r, n), l && u && p(e, t, r) } function p(e = i, t = "", r = "", n) { if (void 0 === a || void 0 === a.post) throw "Bark notification needs to import MagicHttp module."; e = { url: l, headers: { "content-type": "application/json; charset=utf-8" }, body: { title: e, body: t ? t + "\n" + r : r, device_key: u } }; a.post(e).catch(e => { s.error("Bark notify error: " + e) }) } return { post: d, debug: function (e = i, t = "", r = "", n = "") { "DEBUG" === s.getLevel() && (1 === arguments.length && (e = i, t = "", r = arguments[0]), this.post(e, t, r, n)) }, bark: p, setBark: e => { try { var t = e.replace(/\/+$/g, ""); l = /^https?:\/\/([^/]*)/.exec(t)[0] + "/push", u = /\/([^\/]+)\/?$/.exec(t)[1] } catch (e) { s.error(`Bark url error: ${e}.`) } }, appendNotifyInfo: function (e, t) { 1 == t ? c = e : c.push(e) }, prependNotifyInfo: function (e) { c.splice(0, 0, e) }, msg: function (e, t, r, n) { var o = {}; r && (o["open-url"] = r), n && (o["media-url"] = n), (t = t && 0 != t.length ? t : Array.isArray(c) ? c.join("\n") : c) && 0 < t.length && d(i, "", t, o) } } } function MagicUtils(n, u) { const e = (e, t = "yyyy-MM-dd hh:mm:ss") => { var r, n = { "M+": e.getMonth() + 1, "d+": e.getDate(), "h+": e.getHours(), "m+": e.getMinutes(), "s+": e.getSeconds(), "q+": Math.floor((e.getMonth() + 3) / 3), S: e.getMilliseconds() }; for (r in /(y+)/.test(t) && (t = t.replace(RegExp.$1, (e.getFullYear() + "").substr(4 - RegExp.$1.length))), n) new RegExp("(" + r + ")").test(t) && (t = t.replace(RegExp.$1, 1 === RegExp.$1.length ? n[r] : ("00" + n[r]).substr(("" + n[r]).length))); return t }; return { retry: (i, s = 5, a = 0, l = null) => (...e) => new Promise((r, n) => { function o(...t) { Promise.resolve().then(() => i.apply(this, t)).then(e => { "function" == typeof l ? Promise.resolve().then(() => l(e)).then(() => { r(e) }).catch(e => { 1 <= s ? 0 < a ? setTimeout(() => o.apply(this, t), a) : o.apply(this, t) : n(e), s-- }) : r(e) }).catch(e => { u.error(e), 1 <= s && 0 < a ? setTimeout(() => o.apply(this, t), a) : 1 <= s ? o.apply(this, t) : n(e), s-- }) } o.apply(this, e) }), formatTime: e, now: () => e(new Date, "yyyy-MM-dd hh:mm:ss"), today: () => e(new Date, "yyyy-MM-dd"), sleep: t => new Promise(e => setTimeout(e, t)), assert: (e, t = null) => { var r; n.isNode ? (r = require("assert"), t ? r(e, t) : r(e)) : !0 !== e && u.error("AssertionError: " + (t || "The expression evaluated to a falsy value.")) } } } function MagicQingLong(e, a, o) { let i = "", s = "", l = "", u = "", c = "", t = ""; const d = "magic.json", p = MagicHttp(e, o); async function r() { return l = l || a.read("magic_qlclient"), u = u || a.read("magic_qlsecrt"), s = s || a.read("magic_qlname"), c = c || a.read("magic_qlpwd"), i && l && u ? (o.info("Get token from QingLong Panel"), await p.get({ url: "/open/auth/token", headers: { "content-type": "application/json" }, params: { client_id: l, client_secret: u } }).then(e => { if (!(0 < Object.keys(e.body).length && e.body.data && e.body.data.token)) throw new Error("Get QingLong Panel token failed."); o.info("Successfully logged in to QingLong Panel"), t = e.body.data.token, a.write("magic_qltoken", t) }).catch(e => { o.error("Error logging in to QingLong Panel.\n" + (e.message || e)) })) : i && s && c && await p.post({ url: "/api/user/login", headers: { "content-type": "application/json" }, body: { username: s, password: c } }).then(e => { o.info("Successfully logged in to QingLong Panel"), t = e.body.data.token, a.write("magic_qltoken", t) }).catch(e => { o.error("Error logging in to QingLong Panel.\n" + (e.message || e)) }), t } async function g(e) { let t = []; return await p.post({ url: "/api/envs", headers: { "content-type": "application/json" }, body: e }).then(e => { 200 === e.body.code ? e.body.data.forEach(e => { o.debug(`QINGLONG ADD ENV ${e.name} <${typeof e.value}> (${e.id})` + "\n" + JSON.stringify(e)), t.push(e.id) }) : o.error("Error adding environments variable from QingLong Panel.\n" + JSON.stringify(e)) }).catch(e => { o.error("Error adding environments variable from QingLong Panel.\n" + (e.message || e)) }), t } async function n(n = null, e = "", t) { let o = []; return await p.get({ url: "/api/envs", headers: { "content-type": "application/json" }, params: { searchValue: e } }).then(e => { if (200 !== e.body.code) throw new Error("Error reading environment variable from QingLong Panel.\n" + JSON.stringify(e)); e = e.body.data; if (n) { var t = []; for (const r of e) r.name === n && o.push(r); o = t } o = e }).catch(e => { throw new Error("Error reading environments variable from QingLong Panel.\n" + (e.message || e)) }), o } async function f(e, t = "") { let r = ""; return await p.get({ url: "/api/scripts/" + e, params: { path: t } }).then(e => { if (200 !== e.body.code) throw new Error("Error reading data from QingLong Panel.\n" + JSON.stringify(e)); r = e.body.data }).catch(e => { throw new Error("Error reading data from QingLong Panel.\n" + (e.message || e)) }), r } async function h(e, t = "", r = "") { let n = !1; return await p.put({ url: "/api/scripts", headers: { "content-type": "application/json" }, body: { filename: e, path: t, content: r } }).then(e => { 200 === e.body.code ? n = !0 : o.error("Error reading data from QingLong Panel.\n" + JSON.stringify(e)) }).catch(e => { o.error("Error reading data from QingLong Panel.\n" + (e.message || e)) }), n } return p.interceptors.request.use(function (e) { return i = i || a.read("magic_qlurl"), e.url.indexOf(i) < 0 && (e.url = "" + i + e.url), { ...e, timeout: 3e3 } }, void 0), p.interceptors.request.use(function (e) { return (l = l || a.read("magic_qlclient")) && (e.url = e.url.replace("/api/", "/open/")), e }, void 0, { runWhen: e => e.url.indexOf("api/user/login") < 0 && e.url.indexOf("open/auth/token") < 0 }), p.interceptors.request.use(async function (e) { return (t = t || a.read("magic_qltoken", "")) || await r(), e.headers.authorization = "Bearer " + t, e }, void 0, { runWhen: e => e.url.indexOf("api/user/login") < 0 && e.url.indexOf("open/auth/token") < 0 }), p.interceptors.request.use(function (e) { return e.params = { ...e.params, t: Date.now() }, e }, void 0, { runWhen: e => e.url.indexOf("open/auth/token") < 0 }), p.interceptors.request.use(function (e) { return i = i || a.read("magic_qlurl"), t = t || a.read("magic_qltoken"), o.debug("QingLong url: " + i + "\nQingLong token: " + t), e }, void 0), p.interceptors.response.use(void 0, async function (e) { try { var t = e.message || e.error || JSON.stringify(e); return (0 <= t.indexOf("NSURLErrorDomain") && 0 <= t.indexOf("-1012") || e.response && 401 === e.response.status) && e.config && !0 !== e.config.refreshToken ? (o.warning("QingLong Panel token has expired"), o.info("Refreshing the QingLong Panel token"), await r(), e.config.refreshToken = !0, o.info("Call the previous method again"), await p.request(e.config.method, e.config)) : Promise.reject(e) } catch (e) { return Promise.reject(e) } }), { url: i || a.read("magic_qlurl"), init: (e, t, r, n, o) => { i = e, l = t, u = r, s = n, c = o }, getToken: r, setEnv: async function (t, r, n = null) { if (i = i || a.read("magic_qlurl"), null === n) { var e = await g([{ name: t, value: r }]); if (e && 1 === e.length) return e[0] } else await p.put({ url: "/api/envs", headers: { "content-type": "application/json" }, body: { name: t, value: r, id: n } }).then(e => { if (200 === e.body.code) return o.debug(`QINGLONG UPDATE ENV ${t} <${typeof r}> (${n})` + "\n" + JSON.stringify(r)), !0; o.error("Error adding environment variable from QingLong Panel.\n" + JSON.stringify(e)) }).catch(e => (o.error("Error adding environment variable from QingLong Panel.\n" + (e.message || e)), !1)) }, setEnvs: g, getEnv: async function (e) { let t = null; for (const r of await n()) if (r.id === e) { t = r; break } return t }, getEnvs: n, delEnvs: async function (t) { return p.delete({ url: "/api/envs", headers: { accept: "application/json", "accept-language": "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6", connection: "keep-alive", "content-type": "application/json;charset=UTF-8", "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/102.0.5005.63 Safari/537.36 Edg/102.0.1245.30" }, body: t }).then(e => 200 === e.body.code ? (o.debug("QINGLONG DELETE ENV IDS: " + t), !0) : (o.error("Error deleting environments variable from QingLong Panel.\n" + JSON.stringify(e)), !1)).catch(e => { o.error("Error deleting environments variable from QingLong Panel.\n" + (e.message || e)) }) }, disableEnvs: async function (t) { let r = !1; return await p.put({ url: "/api/envs/disable", headers: { accept: "application/json", "accept-Language": "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6", connection: "keep-alive", "content-type": "application/json;charset=UTF-8", "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/102.0.5005.63 Safari/537.36 Edg/102.0.1245.30" }, body: t }).then(e => { 200 === e.body.code ? (o.debug("QINGLONG DISABLED ENV IDS: " + t), r = !0) : o.error("Error disabling environments variable from QingLong Panel.\n" + JSON.stringify(e)) }).catch(e => { o.error("Error disabling environments variable from QingLong Panel.\n" + (e.message || e)) }), r }, enableEnvs: async function (t) { let r = !1; return await p.put({ url: "/api/envs/enable", headers: { accept: "application/json", "accept-language": "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6", connection: "keep-alive", "content-type": "application/json;charset=UTF-8", "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/102.0.5005.63 Safari/537.36 Edg/102.0.1245.30" }, body: t }).then(e => { 200 === e.body.code ? (o.debug("QINGLONG ENABLED ENV IDS: " + t), r = !0) : o.error("Error enabling environments variable from Qilong panel.\n" + JSON.stringify(e)) }).catch(e => { o.error("Error enabling environments variable from Qilong panel.\n" + (e.message || e)) }), r }, addScript: async function (e, t = "", r = "") { let n = !1; return await p.post({ url: "/api/scripts", headers: { "content-type": "application/json" }, body: { filename: e, path: t, content: r } }).then(e => { 200 === e.body.code ? n = !0 : o.error("Error reading data from QingLong Panel.\n" + JSON.stringify(e)) }).catch(e => { o.error("Error reading data from QingLong Panel.\n" + (e.message || e)) }), n }, getScript: f, editScript: h, delScript: async function (e, t = "") { let r = !1; return await p.delete({ url: "/api/scripts", headers: { "content-type": "application/json" }, body: { filename: e, path: t } }).then(e => { 200 === e.body.code ? r = !0 : o.error("Error reading data from QingLong Panel.\n" + JSON.stringify(e)) }).catch(e => { o.error("Error reading data from QingLong Panel.\n" + (e.message || e)) }), r }, write: async function (e, t, r = "") { var n = await f(d, ""), o = a.convertToObject(n), e = a.write(e, t, r, o), n = JSON.stringify(o, null, 4); return await h(d, "", n) && e }, read: async function (e, t, r = "", n = !1) { var o = await f(d, ""), o = a.convertToObject(o); return a.read(e, t, r, n, o) }, del: async function (e, t = "") { var r = await f(d, ""), n = a.convertToObject(r), e = a.del(e, t, n), r = JSON.stringify(n, null, 4), t = await h(d, "", r); return e && t }, update: async function (e, t, r, n = a.defaultValueComparator) { var o = await f(d, ""), i = a.convertToObject(o), e = a.update(e, t, r, n, i); let s = !1; return !0 === e && (o = JSON.stringify(i, null, 4), s = await h(d, "", o)), e && s }, batchWrite: async function (...e) { var t, r = await f(d, ""), n = a.convertToObject(r); for (t of e) a.write(t[0], t[1], void 0 !== t[2] ? t[2] : "", n); return r = JSON.stringify(n, null, 4), h(d, "", r) }, batchRead: async function (...e) { var t, r = await f(d, ""), n = a.convertToObject(r), o = []; for (t of e) { var i = a.read(t[0], t[1], void 0 !== t[2] ? t[2] : "", "boolean" == typeof t[3] && t[3], n); o.push(i) } return o }, batchUpdate: async function (...e) { var t, r = await f(d, ""), n = a.convertToObject(r); for (t of e) a.update(t[0], t[1], void 0 !== t[2] ? t[2] : "", void 0 !== t[3] ? t.comparator : a.defaultValueComparator, n); return r = JSON.stringify(n, null, 4), h(d, "", r) }, batchDel: async function (...e) { var t, r = await f(d, ""), n = a.convertToObject(r); for (t of e) a.del(t[0], void 0 !== t[1] ? t[1] : "", n); return r = JSON.stringify(n, null, 4), h(d, "", r) }, allSessions: async function (e) { var t = await f(d, ""), t = a.convertToObject(t); return a.allSessions(e, t) }, allSessionNames: async function (e) { var t = await f(d, ""), t = a.convertToObject(t); return a.allSessionNames(e, t) } } }
  1305. //---SyncByPyScript---MagicJS3-end