-
Notifications
You must be signed in to change notification settings - Fork 74
/
Copy pathfqkk.js
355 lines (334 loc) · 20 KB
/
fqkk.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
/*
软件名称:番茄看看 微信扫描二维码打开
感谢 @肥皂 大佬
脚本说明:番茄看看自动阅读
脚本为自动完成番茄看看的阅读任务
每日收益2.7元左右,可多号撸。提现秒到
任务打开二维码地址 https://raw.githubusercontent.com/age174/-/main/3F545C70-389B-4155-ACB1-15B6FDA95501.jpeg
可以去boxjs修改自动提现金额
最低提现额度为0.3元,默认提现1元
本脚本以学习为主!
首次运行脚本,会提示获取数据
去番茄看看,点击阅读A任务,开始阅读,
完成一次阅读即可获取数据。
我的邀请码 : 感谢大佬们填写
注意:如果重定向跳转失败或者跑脚本没有key没有提交成功,请手动去做一个阅读A任务再执行脚本。
2.24更新 运行日志加入boxjs设置的循环次数和提现金额,key提交因为有很多302重定向,如跑脚本没有金币请查看日志的重定向是否错误
已修改循环方式,方式循环方式为一直阅读,直到当前无任务可做自动停止
2.27修复番茄看看因跟换域名无法获取数据的问题,自行更换重写和mitm
3.2增加剩余阅读次数查询
3.5番茄看看更新,优化为不指定账号抓包方便羊毛党等多账号;以前抓到的数据可以迁移一次,通过boxjs设置跑一次脚本就可以 ,现在番茄看看可以并发执行了,简而言之就是多账号不需要一个账号跑完再去跑下一个账号,而是多个账号同时跑任务了,需要并发几个账号就去boxjs修改数值就可以了
*/
const $ = new Env('番茄看看');
const fqkkurlArr = [], fqkkhdArr = []
let fqkk = $.getjson('fqkk', [])
let fqkkBanfirstTask = $.getval('fqkkBanfirstTask') || 'false' // 禁止脚本执行首个任务,避免每日脚本跑首次任务导致微信限制
let fqkkCkMoveFlag = $.getval('fqkkCkMove') || ''
let fqtx = ($.getval('fqtx') || '100'); // 此处修改提现金额,0.3元等于30,默认为提现一元,也就是100
let concurrency = ($.getval('fqkkConcurrency') || '1') - 0; // 并发执行任务的账号数,默单账号循环执行
concurrency = concurrency < 1 ? 1 : concurrency;
let fqkktz = ''
!(async () => {
if (typeof $request !== "undefined") {
await fqkkck();
} else if (fqkkCkMoveFlag == 'true') {
await fqkkCkMove();
} else {
let acList = fqkk.filter(o => o.hd).map((o, i) => ({no: i+1, uid: o.uid, gold: 0, score: 0, rest: 0, num: 0, url: o.url, headers: JSON.parse(o.hd)}));
let execAcList = [];
let slot = acList.length % concurrency == 0 ? acList.length / concurrency : parseInt(acList.length / concurrency) + 1;
acList.forEach((o, i) => {
let idx = i % slot;
if (execAcList[idx]) {
execAcList[idx].push(o);
} else {
execAcList[idx] = [o];
}
});
$.log(`番茄看看当前设置的提现金额为: ${fqtx / 100} 元`, `----------- 共${acList.length}个账号分${execAcList.length}组去执行 -----------`);
for (let arr of execAcList) {
let allAc = arr.map(ac=>ac.no).join(', ');
$.log(`\n=======================================\n开始【${$.name}账号:${allAc}】`);
let rtList = await Promise.all(arr.map((ac, i) => execTask(ac, i)));
for (let ac of rtList) {
let msg = '';
if (ac.uid && ac.gold >= fqtx) {
$.log(`检测到账号${ac.no}已满足设置的提现金额,前去执行提现任务\n`)
msg = await fqkktx(ac);
}
ac.msg = msg;
}
fqkktz += rtList.map(ac => `【账号${ac.no}】\n余额:${ac.gold}币\n今日奖励:${ac.score}\n已阅读数:${ac.num}\n待阅读数:${ac.rest}${ac.msg?'\n'+ac.msg:''}`).join('\n\n');
}
$.log('\n======== [脚本运行完毕,打印日志结果] ========\n'+fqkktz) }
})()
.catch((e) => $.logErr(e))
.finally(() => $.done());
function execTask(ac, i) {
return new Promise(resolve => {
setTimeout(async () => {
try {
let msg = await fqkk3(ac, '');
if (ac.rest) {
let skip = false;
if(fqkkBanfirstTask == 'ture' && ac.num <= 0){
saip = ture;
}
if (ac.rest <= 0 || skip) {
$.log(`账号${ac.no}今日已阅读${ac.num}次,本阶段待阅读${ac.rest}次,跳过阅读`);
} else {
$.log(`账号${ac.no}今日已阅读${ac.num}次,本阶段待阅读${ac.rest}次,开始阅读\n`);
let flag = 0;
let count = 1;
let allowErrorCount = 3;
do {
let [f, m] = await fqkk1(ac, count++);
flag = f;
msg = m;
if (flag > 0) {
$.log(`🌝账号${ac.no}在本阶段还有${ac.rest}次阅读待执行,立即阅读\n`);
} else if (flag < 0) {
allowErrorCount--;
} else {
$.log(`🌝账号${ac.no}在本阶段待阅读任务已完成:\n${msg}`);
}
} while (flag && allowErrorCount);
}
} else {
$.log(`账号${ac.no}今日已阅读${ac.num}次,本阶段还有${ac.rest}次待阅读:\n${msg}`);
}
let host = ac.url.match(/http?:\/\/(.+?)\//)[1];
let ck = ac.headers['Cookie'] || ac.headers['cookie'];
let [userId, gold] = await userInfo(host, ck);
ac.gold = gold;
} catch (e) {
$.logErr(e, `账号${ac.no} 循环执行任务出现异常`);
} finally{
resolve(ac);
}
}, i * 500);
})
}
//番茄看看数据获取
async function fqkkck() {
if ($request.url.indexOf("getTask") > -1) {
const fqkkurl = $request.url;
const fqkkhd = JSON.stringify($request.headers);
let host = fqkkurl.match(/http?:\/\/(.+?)\//)[1];
let ck = $request.headers['Cookie'] || $request.headers['cookie'];
let [userId, gold] = await userInfo(host, ck);
if (userId) {
// 获取到用户ID,记录
let status = 1;
let no = fqkk.length;
for (let i = 0, len = no; i < len; i++) {
let ac = fqkk[i] || {};
if (ac.uid) {
if (ac.uid == userId) {
no = i;
status = 0;
break;
}
} else if (no == len) {
no = i;
}
}
fqkk[no] = {uid: userId, url: fqkkurl, hd: fqkkhd};
$.setdata(JSON.stringify(fqkk, null, 2), 'fqkk');
$.log(fqkkurl);
$.log(fqkkhd);
$.msg($.name, "", `番茄看看[账号${no+1}] ${status?'新增':'更新'}数据成功!`);
} else {
// 未获取到用户ID,提示
$.msg($.name, "", '番茄看看用户ID获取失败⚠️');
}
}
}
async function fqkkCkMove() {
let fqkkcount = ($.getval('fqkkcount') || '0') - 0;
for (let i = 1; i <= fqkkcount; i++) {
fqkkurlArr.push($.getdata(`fqkkurl${i>1?i:''}`))
fqkkhdArr.push($.getdata(`fqkkhd${i>1?i:''}`))
}
if (fqkkhdArr.length > 0) {
let existsId = fqkk.map(o => o.uid - 0);
for (let i = 0, len = fqkkhdArr.length; i < len; i++) {
fqkkurl = fqkkurlArr[i];
fqkkhd = fqkkhdArr[i];
if(!fqkkurl || !fqkkhd){
continue;
}
let host = fqkkurl.match(/http?:\/\/(.+?)\//)[1];
let hd = JSON.parse(fqkkhd);
let ck = hd['Cookie'] || hd['cookie']
let [userId, gold] = await userInfo(host, ck);
if (userId && !existsId.includes(userId)) {
fqkk.push({uid: userId, url: fqkkurl,hd: fqkkhd});
existsId.push(userId);
}
}
$.setdata(JSON.stringify(fqkk, null, 2), 'fqkk');
$.msg($.name, "", `番茄看看数据迁移后共${fqkk.length}个账号!`);
}
$.setval('', 'fqkkCkMove');
}
function userInfo(host, ck) {
return new Promise((resolve) => {
let opts = {
url: `http://${host}/user/`,
headers: {
Cookie: ck
}
}
$.get(opts, async (err, resp, data) => {
let userId = 0;
let gold = -1;
try {
if (err) {
$.logErr(`❌ 用户信息API请求失败,清检查网络设置 \n ${JSON.stringify(err)}`)
} else {
userId = (data.match(/\[用户ID:(\d+?)\]/) || ['', '0'])[1] - 0;
if (userId) {
gold = (data.match(/余额.+?([\d\.]+?)币/) || ['', '0'])[1] - 0;
}
}
} catch (e) {
$.log(`=================\nurl: ${opts.url}\ndata: ${resp && resp.body}`);
$.logErr(e, resp);
} finally {
resolve([userId, gold])
}
})
})
}
//番茄看看领取
function fqkk3(ac, fqkey) {
return new Promise(resolve => {
let opts = {
url: ac.url.replace('getTask', 'finishTask'),
headers: ac.headers,
body: `readLastKey=${fqkey||''}`,
}
$.post(opts, (err, resp, data) => {
let msg = '未知问题';
try {
if (err) {
$.log(`${$.name} ${ac.no} 请求失败,请检查网路重试\n url: ${opts.url} \n data: ${JSON.stringify(err, null, 2)}`);
} else {
const result = JSON.parse(data);
if (result.data && result.data.infoView) {
ac.rest = (result.data.infoView.rest || 0) - 0;
ac.num = (result.data.infoView.num || 0) - 0;
ac.score = (result.data.infoView.score || 0) - 0;
msg = ac.rest > 0 ? '-' : (result.data.infoView.msg || msg);
}
if (fqkey) {
if (result.code == 0) {
$.log(`🌝账号${ac.no}:${result.msg}`, `今日阅读次数: ${result.data.infoView.num}, 今日阅读奖励: ${result.data.infoView.score}`);
} else {
ac.rest = -1;
$.log(`🚫账号${ac.no}:${result.msg}`, `今日阅读次数: ${result.data.infoView.num}, 今日阅读奖励: ${result.data.infoView.score}`, `resp: ${JSON.stringify(resp||'', null, 2)}`);
}
}
}
} catch (e) {
$.log(`======== 账号 ${ac.no} ========\nurl: ${opts.url}\ndata: ${resp && resp.body}`);
$.logErr(e, resp);
} finally {
resolve(msg)
}
})
})
}
function fqkk2(ac, fqkey) {
return new Promise((resolve) => {
let opts = {
url: ac.url.replace('getTask', `jump?key=${fqkey}`),
headers: ac.headers
};
$.get(opts, (err, resp, data) => {
let rtObj = '';
try {
if (err) {
$.log(`${$.name} ${ac.no} 请求失败,请检查网路重试\n url: ${opts.url} \n data: ${JSON.stringify(err, null, 2)}`);
} else {
rtObj = $.toObj(data, {});
}
} catch (e) {
$.log(`======== 账号 ${ac.no} ========\nurl: ${opts.url}\ndata: ${resp && resp.body}`);
$.logErr(e, resp);
} finally {
resolve(rtObj)
}
})
})
}
//番茄看看key
function fqkk1(ac, fqjs, timeout = 0) {
return new Promise((resolve) => {
setTimeout(() => {
let opts = {
url: ac.url,
headers: ac.headers,
body: '',
}
$.post(opts, async (err, resp, data) => {
let f = -1;
let m = '';
try {
const result = JSON.parse(data)
if (result.code == 0 && result.data && result.data.jkey) {
$.log(`🌝账号${ac.no}获取key回执成功,开始第${fqjs}次跳转阅读📖`);
let jumpObj = await fqkk2(ac, result.data.jkey);
if (jumpObj) {
$.log(`🌝账号${ac.no}等待10秒后提交本次阅读领取奖励`);
await $.wait(10000);
m = await fqkk3(ac, result.data.jkey);
f = ac.rest;
} else {
$.log(`🌝账号${ac.no}jump接口请求失败,重新执行阅读任务`);
}
} else {
f = 0;
$.log(`🚫账号${ac.no}获取key回执失败:${result.msg}`);
}
} catch (e) {
$.log(`======== 账号 ${ac.no} ========\nurl: ${opts.url}\ndata: ${resp && resp.body}`);
// $.logErr(e, resp);
} finally {
resolve([f, m]);
}
})
}, timeout)
})
}
//提现
function fqkktx(ac) {
return new Promise((resolve) => {
let opts = {
url: ac.url.match(/^(https?:\/\/.+?)\//)[1] + '/withdrawal/doWithdraw',
headers: ac.headers,
body: `amount=${fqtx}`
}
$.post(opts, async (err, resp, data) => {
let msg = '';
try {
const result = JSON.parse(data)
if (result.code == 0) {
msg = `提现成功🌝\n`;
$.msg(`番茄看看账号 ${ac.no}`,'','成功提现至微信: '+fqtx / 100 +' 元')
} else {
msg = `提现失败🚫: ${result.msg}\n\n`;
$.msg(`番茄看看账号 ${ac.no}`,'',`提现失败🚫: ${result.msg}`)
}
} catch (e) {
$.log(`======== 账号 ${ac.no} ========\nurl: ${opts.url}\ndata: ${resp && resp.body}`);
$.logErr(e, resp);
msg = `提现异常:${e}`;
} finally {
resolve(msg)
}
})
})
}
function Env(t,e){class s{constructor(t){this.env=t}send(t,e="GET"){t="string"==typeof t?{url:t}:t;let s=this.get;return"POST"===e&&(s=this.post),new Promise((e,i)=>{s.call(this,t,(t,s,r)=>{t?i(t):e(s)})})}get(t){return this.send.call(this.env,t)}post(t){return this.send.call(this.env,t,"POST")}}return new class{constructor(t,e){this.name=t,this.http=new s(this),this.data=null,this.dataFile="box.dat",this.logs=[],this.isMute=!1,this.isNeedRewrite=!1,this.logSeparator="\n",this.startTime=(new Date).getTime(),Object.assign(this,e),this.log("",`\ud83d\udd14${this.name}, \u5f00\u59cb!`)}isNode(){return"undefined"!=typeof module&&!!module.exports}isQuanX(){return"undefined"!=typeof $task}isSurge(){return"undefined"!=typeof $httpClient&&"undefined"==typeof $loon}isLoon(){return"undefined"!=typeof $loon}toObj(t,e=null){try{return JSON.parse(t)}catch{return e}}toStr(t,e=null){try{return JSON.stringify(t)}catch{return e}}getjson(t,e){let s=e;const i=this.getdata(t);if(i)try{s=JSON.parse(this.getdata(t))}catch{}return s}setjson(t,e){try{return this.setdata(JSON.stringify(t),e)}catch{return!1}}getScript(t){return new Promise(e=>{this.get({url:t},(t,s,i)=>e(i))})}runScript(t,e){return new Promise(s=>{let i=this.getdata("@chavy_boxjs_userCfgs.httpapi");i=i?i.replace(/\n/g,"").trim():i;let r=this.getdata("@chavy_boxjs_userCfgs.httpapi_timeout");r=r?1*r:20,r=e&&e.timeout?e.timeout:r;const[o,h]=i.split("@"),a={url:`http://${h}/v1/scripting/evaluate`,body:{script_text:t,mock_type:"cron",timeout:r},headers:{"X-Key":o,Accept:"*/*"}};this.post(a,(t,e,i)=>s(i))}).catch(t=>this.logErr(t))}loaddata(){if(!this.isNode())return{};{this.fs=this.fs?this.fs:require("fs"),this.path=this.path?this.path:require("path");const t=this.path.resolve(this.dataFile),e=this.path.resolve(process.cwd(),this.dataFile),s=this.fs.existsSync(t),i=!s&&this.fs.existsSync(e);if(!s&&!i)return{};{const i=s?t:e;try{return JSON.parse(this.fs.readFileSync(i))}catch(t){return{}}}}}writedata(){if(this.isNode()){this.fs=this.fs?this.fs:require("fs"),this.path=this.path?this.path:require("path");const t=this.path.resolve(this.dataFile),e=this.path.resolve(process.cwd(),this.dataFile),s=this.fs.existsSync(t),i=!s&&this.fs.existsSync(e),r=JSON.stringify(this.data);s?this.fs.writeFileSync(t,r):i?this.fs.writeFileSync(e,r):this.fs.writeFileSync(t,r)}}lodash_get(t,e,s){const i=e.replace(/\[(\d+)\]/g,".$1").split(".");let r=t;for(const t of i)if(r=Object(r)[t],void 0===r)return s;return r}lodash_set(t,e,s){return Object(t)!==t?t:(Array.isArray(e)||(e=e.toString().match(/[^.[\]]+/g)||[]),e.slice(0,-1).reduce((t,s,i)=>Object(t[s])===t[s]?t[s]:t[s]=Math.abs(e[i+1])>>0==+e[i+1]?[]:{},t)[e[e.length-1]]=s,t)}getdata(t){let e=this.getval(t);if(/^@/.test(t)){const[,s,i]=/^@(.*?)\.(.*?)$/.exec(t),r=s?this.getval(s):"";if(r)try{const t=JSON.parse(r);e=t?this.lodash_get(t,i,""):e}catch(t){e=""}}return e}setdata(t,e){let s=!1;if(/^@/.test(e)){const[,i,r]=/^@(.*?)\.(.*?)$/.exec(e),o=this.getval(i),h=i?"null"===o?null:o||"{}":"{}";try{const e=JSON.parse(h);this.lodash_set(e,r,t),s=this.setval(JSON.stringify(e),i)}catch(e){const o={};this.lodash_set(o,r,t),s=this.setval(JSON.stringify(o),i)}}else s=this.setval(t,e);return s}getval(t){return this.isSurge()||this.isLoon()?$persistentStore.read(t):this.isQuanX()?$prefs.valueForKey(t):this.isNode()?(this.data=this.loaddata(),this.data[t]):this.data&&this.data[t]||null}setval(t,e){return this.isSurge()||this.isLoon()?$persistentStore.write(t,e):this.isQuanX()?$prefs.setValueForKey(t,e):this.isNode()?(this.data=this.loaddata(),this.data[e]=t,this.writedata(),!0):this.data&&this.data[e]||null}initGotEnv(t){this.got=this.got?this.got:require("got"),this.cktough=this.cktough?this.cktough:require("tough-cookie"),this.ckjar=this.ckjar?this.ckjar:new this.cktough.CookieJar,t&&(t.headers=t.headers?t.headers:{},void 0===t.headers.Cookie&&void 0===t.cookieJar&&(t.cookieJar=this.ckjar))}get(t,e=(()=>{})){t.headers&&(delete t.headers["Content-Type"],delete t.headers["Content-Length"]),this.isSurge()||this.isLoon()?(this.isSurge()&&this.isNeedRewrite&&(t.headers=t.headers||{},Object.assign(t.headers,{"X-Surge-Skip-Scripting":!1})),$httpClient.get(t,(t,s,i)=>{!t&&s&&(s.body=i,s.statusCode=s.status),e(t,s,i)})):this.isQuanX()?(this.isNeedRewrite&&(t.opts=t.opts||{},Object.assign(t.opts,{hints:!1})),$task.fetch(t).then(t=>{const{statusCode:s,statusCode:i,headers:r,body:o}=t;e(null,{status:s,statusCode:i,headers:r,body:o},o)},t=>e(t))):this.isNode()&&(this.initGotEnv(t),this.got(t).on("redirect",(t,e)=>{try{if(t.headers["set-cookie"]){const s=t.headers["set-cookie"].map(this.cktough.Cookie.parse).toString();this.ckjar.setCookieSync(s,null),e.cookieJar=this.ckjar}}catch(t){this.logErr(t)}}).then(t=>{const{statusCode:s,statusCode:i,headers:r,body:o}=t;e(null,{status:s,statusCode:i,headers:r,body:o},o)},t=>{const{message:s,response:i}=t;e(s,i,i&&i.body)}))}post(t,e=(()=>{})){if(t.body&&t.headers&&!t.headers["Content-Type"]&&(t.headers["Content-Type"]="application/x-www-form-urlencoded"),t.headers&&delete t.headers["Content-Length"],this.isSurge()||this.isLoon())this.isSurge()&&this.isNeedRewrite&&(t.headers=t.headers||{},Object.assign(t.headers,{"X-Surge-Skip-Scripting":!1})),$httpClient.post(t,(t,s,i)=>{!t&&s&&(s.body=i,s.statusCode=s.status),e(t,s,i)});else if(this.isQuanX())t.method="POST",this.isNeedRewrite&&(t.opts=t.opts||{},Object.assign(t.opts,{hints:!1})),$task.fetch(t).then(t=>{const{statusCode:s,statusCode:i,headers:r,body:o}=t;e(null,{status:s,statusCode:i,headers:r,body:o},o)},t=>e(t));else if(this.isNode()){this.initGotEnv(t);const{url:s,...i}=t;this.got.post(s,i).then(t=>{const{statusCode:s,statusCode:i,headers:r,body:o}=t;e(null,{status:s,statusCode:i,headers:r,body:o},o)},t=>{const{message:s,response:i}=t;e(s,i,i&&i.body)})}}time(t){let e={"M+":(new Date).getMonth()+1,"d+":(new Date).getDate(),"H+":(new Date).getHours(),"m+":(new Date).getMinutes(),"s+":(new Date).getSeconds(),"q+":Math.floor(((new Date).getMonth()+3)/3),S:(new Date).getMilliseconds()};/(y+)/.test(t)&&(t=t.replace(RegExp.$1,((new Date).getFullYear()+"").substr(4-RegExp.$1.length)));for(let s in e)new RegExp("("+s+")").test(t)&&(t=t.replace(RegExp.$1,1==RegExp.$1.length?e[s]:("00"+e[s]).substr((""+e[s]).length)));return t}msg(e=t,s="",i="",r){const o=t=>{if(!t)return t;if("string"==typeof t)return this.isLoon()?t:this.isQuanX()?{"open-url":t}:this.isSurge()?{url:t}:void 0;if("object"==typeof t){if(this.isLoon()){let e=t.openUrl||t.url||t["open-url"],s=t.mediaUrl||t["media-url"];return{openUrl:e,mediaUrl:s}}if(this.isQuanX()){let e=t["open-url"]||t.url||t.openUrl,s=t["media-url"]||t.mediaUrl;return{"open-url":e,"media-url":s}}if(this.isSurge()){let e=t.url||t.openUrl||t["open-url"];return{url:e}}}};this.isMute||(this.isSurge()||this.isLoon()?$notification.post(e,s,i,o(r)):this.isQuanX()&&$notify(e,s,i,o(r)));let h=["","==============\ud83d\udce3\u7cfb\u7edf\u901a\u77e5\ud83d\udce3=============="];h.push(e),s&&h.push(s),i&&h.push(i),console.log(h.join("\n")),this.logs=this.logs.concat(h)}log(...t){t.length>0&&(this.logs=[...this.logs,...t]),console.log(t.join(this.logSeparator))}logErr(t,e){const s=!this.isSurge()&&!this.isQuanX()&&!this.isLoon();s?this.log("",`\u2757\ufe0f${this.name}, \u9519\u8bef!`,t.stack):this.log("",`\u2757\ufe0f${this.name}, \u9519\u8bef!`,t)}wait(t){return new Promise(e=>setTimeout(e,t))}done(t={}){const e=(new Date).getTime(),s=(e-this.startTime)/1e3;this.log("",`\ud83d\udd14${this.name}, \u7ed3\u675f! \ud83d\udd5b ${s} \u79d2`),this.log(),(this.isSurge()||this.isQuanX()||this.isLoon())&&$done(t)}}(t,e)}