koishi-plugin-node-async-bot-all 2.16.0 → 2.18.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/commands.d.ts +22 -3
- package/lib/fun.d.ts +4 -0
- package/lib/index.d.ts +11 -0
- package/lib/index.js +185 -106
- package/package.json +1 -1
- package/res/userCard.html +242 -0
package/lib/commands.d.ts
CHANGED
|
@@ -1,16 +1,35 @@
|
|
|
1
1
|
import { Context, Session } from 'koishi';
|
|
2
2
|
import { Installer } from "@koishijs/plugin-market";
|
|
3
|
+
import Puppeteer from 'koishi-plugin-puppeteer';
|
|
3
4
|
declare module 'koishi' {
|
|
4
5
|
interface Context {
|
|
5
6
|
installer: Installer;
|
|
7
|
+
puppeteer: Puppeteer;
|
|
6
8
|
}
|
|
7
9
|
}
|
|
10
|
+
export interface APIUserInfo {
|
|
11
|
+
qq: string;
|
|
12
|
+
nickname: string;
|
|
13
|
+
long_nick: string;
|
|
14
|
+
avatar_url: string;
|
|
15
|
+
age: number;
|
|
16
|
+
sex: string;
|
|
17
|
+
qid: string;
|
|
18
|
+
qq_level: number;
|
|
19
|
+
location: string;
|
|
20
|
+
email: string;
|
|
21
|
+
is_vip: boolean;
|
|
22
|
+
vip_level: number;
|
|
23
|
+
reg_time: string;
|
|
24
|
+
last_updated: string;
|
|
25
|
+
}
|
|
8
26
|
export declare function getServer(ctx: Context, session: Session): Promise<Object>;
|
|
9
27
|
export declare function getStatus(ctx: Context, session: Session): Promise<Object>;
|
|
10
28
|
export declare function getRandom(ctx: Context, session: Session, min: number, max: number): Promise<Object>;
|
|
11
29
|
export declare function getInfo(ctx: Context, session: Session): Promise<Object>;
|
|
12
|
-
export declare function
|
|
13
|
-
export declare function
|
|
30
|
+
export declare function getRandomWord(ctx: Context, session: Session): Promise<Object>;
|
|
31
|
+
export declare function getBlueArchive(ctx: Context, session: Session): Promise<Number>;
|
|
14
32
|
export declare function serverTest(ctx: Context, session: Session): Promise<Object>;
|
|
15
|
-
export declare function getSteam(ctx: Context, session: Session, id: number): Promise<Object>;
|
|
16
33
|
export declare function getMeme(ctx: Context, session: Session, count: number): Promise<Number>;
|
|
34
|
+
export declare function getCat(ctx: Context, session: Session): Promise<Number>;
|
|
35
|
+
export declare function getQQInfo(ctx: Context, session: Session, qq: string): Promise<number>;
|
package/lib/fun.d.ts
CHANGED
|
@@ -1,4 +1,6 @@
|
|
|
1
|
+
import { HttpResponse } from "./index";
|
|
1
2
|
import { Context } from "koishi";
|
|
3
|
+
import { APIUserInfo } from "./commands";
|
|
2
4
|
export declare function getSystemUsage(): Promise<Object>;
|
|
3
5
|
export declare function getHongKongTime(): string;
|
|
4
6
|
export declare function fetchWithTimeout(url: string, options: {} | undefined, timeout: number | undefined, log: any): Promise<Response>;
|
|
@@ -18,3 +20,5 @@ export declare function getHttp(log: any, url: string, timeout: number): Promise
|
|
|
18
20
|
data: object;
|
|
19
21
|
error?: boolean;
|
|
20
22
|
}>;
|
|
23
|
+
export declare function request<T = any>(url: string, options?: RequestInit, timeout?: number, log?: any): Promise<HttpResponse<T>>;
|
|
24
|
+
export declare function readUserCardFile(userInfo: APIUserInfo): Promise<string>;
|
package/lib/index.d.ts
CHANGED
|
@@ -1,5 +1,14 @@
|
|
|
1
1
|
import { Context, Dict, Schema } from 'koishi';
|
|
2
2
|
export declare const inject: string[];
|
|
3
|
+
export type HttpResponse<T> = {
|
|
4
|
+
success: true;
|
|
5
|
+
data: T;
|
|
6
|
+
} | {
|
|
7
|
+
success: false;
|
|
8
|
+
error: any;
|
|
9
|
+
code?: number;
|
|
10
|
+
isJson: boolean;
|
|
11
|
+
};
|
|
3
12
|
declare module 'koishi' {
|
|
4
13
|
interface Tables {
|
|
5
14
|
botData: botDataTables;
|
|
@@ -24,6 +33,8 @@ export interface Config {
|
|
|
24
33
|
serverPing: Dict<string>;
|
|
25
34
|
steamAPI: string;
|
|
26
35
|
memesAPI: Dict<string>;
|
|
36
|
+
catAPI: string;
|
|
37
|
+
qqAPI: string;
|
|
27
38
|
}
|
|
28
39
|
export declare const Config: Schema<Config>;
|
|
29
40
|
export declare function apply(ctx: Context): void;
|
package/lib/index.js
CHANGED
|
@@ -34,7 +34,7 @@ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: tru
|
|
|
34
34
|
// src/locales/zh-CN.yml
|
|
35
35
|
var require_zh_CN = __commonJS({
|
|
36
36
|
"src/locales/zh-CN.yml"(exports2, module2) {
|
|
37
|
-
module2.exports = { commands: { cxgame: { description: "查询服务器当前人数。", messages: { msg: "{time}{list}\n进服指南请在群公告中查看。", list: "【MC 服务器 {count}】\n➣ {version}:{players}\n➣ 玩家列表:{list}\n➣ 备注:{note}", listNoPlayer: "【MC 服务器 {count}】\n➣ {version}:{players}\n➣ 备注:{note}", listFailed: "【MC 服务器 {count}】\n➣ 查询失败:{data}\n➣ 请稍后重试。", forbidden: "{time}\n此指令不允许在本群使用。", failed: "{time}\n查询失败:{data}", timeout: "请求超时。", timeout2: "响应超时。", fewData: "服务端返回的数据过少。", close: "服务器已关闭。", error: "执行错误。", host: "没有到主机的路由。" } }, status: { description: "查询机器人状态。", messages: { msg: "{time}\n--- 系统状态 ---\n系统名称:{name}\nCPU使用率:{cpu}\n内存使用率:{memory}\n--- 机器人状态 ---\n昨日收/发消息数量:{msgCount}\n机器人版本:{version}\n运行时间:{online}", failed: "{time}\n状态获取失败。" } }, random: { description: "随机数生成器。", usage: "缺少参数时默认生成 0-10000 的随机数。\n使用示例:", examples: "random 1 128 生成1到128范围的随机数", messages: { msg: "{time}\n生成的随机数:{data}" } }, info: { description: "查询机器人信息。", messages: { msg: "{data}", failed: "{time}\n读取信息失败。" } }, rw: { description: "随机名言名句。", messages: { msg: "{time}\n{data}", error: "执行错误。", failed1: "{time}\n获取失败(1)。", failed2: "{time}\n获取失败({data})。" } }, randomba: { description: "随机BA图。", messages: { msg: "{quote}{image}", wait: "{quote}{time}\n请等待图片上传(可能较慢)。" } }, servertest: { description: "Ping服务器。", messages: { msg: "{time}\n== Ping {host} ==\n状态:{alive}\n丢包率:{packetLoss}\n返回IP:{ip}", forbidden: "{time}\n此指令不允许在本群使用。", failed: "{time}\nPing 失败:{data}" } },
|
|
37
|
+
module2.exports = { commands: { cxgame: { description: "查询服务器当前人数。", messages: { msg: "{time}{list}\n进服指南请在群公告中查看。", list: "【MC 服务器 {count}】\n➣ {version}:{players}\n➣ 玩家列表:{list}\n➣ 备注:{note}", listNoPlayer: "【MC 服务器 {count}】\n➣ {version}:{players}\n➣ 备注:{note}", listFailed: "【MC 服务器 {count}】\n➣ 查询失败:{data}\n➣ 请稍后重试。", forbidden: "{time}\n此指令不允许在本群使用。", failed: "{time}\n查询失败:{data}", timeout: "请求超时。", timeout2: "响应超时。", fewData: "服务端返回的数据过少。", close: "服务器已关闭。", error: "执行错误。", host: "没有到主机的路由。" } }, status: { description: "查询机器人状态。", messages: { msg: "{time}\n--- 系统状态 ---\n系统名称:{name}\nCPU使用率:{cpu}\n内存使用率:{memory}\n--- 机器人状态 ---\n昨日收/发消息数量:{msgCount}\n机器人版本:{version}\n运行时间:{online}", failed: "{time}\n状态获取失败。" } }, random: { description: "随机数生成器。", usage: "缺少参数时默认生成 0-10000 的随机数。\n使用示例:", examples: "random 1 128 生成1到128范围的随机数", messages: { msg: "{time}\n生成的随机数:{data}" } }, info: { description: "查询机器人信息。", messages: { msg: "{data}", failed: "{time}\n读取信息失败。" } }, rw: { description: "随机名言名句。", messages: { msg: "{time}\n{data}", error: "执行错误。", failed1: "{time}\n获取失败(1)。", failed2: "{time}\n获取失败({data})。" } }, randomba: { description: "随机BA图。", messages: { msg: "{quote}{image}", wait: "{quote}{time}\n请等待图片上传(可能较慢)。" } }, servertest: { description: "Ping服务器。", messages: { msg: "{time}\n== Ping {host} ==\n状态:{alive}\n丢包率:{packetLoss}\n返回IP:{ip}", forbidden: "{time}\n此指令不允许在本群使用。", failed: "{time}\nPing 失败:{data}" } }, meme: { description: "群友的怪话!", messages: { msg: "{quote}{time}\n{image}\n{title}", failed: "{quote}{time}\n获取失败:{data}", forbidden: "{quote}{time}\n此指令不允许在本群使用。", error: "执行错误。" } }, randomcat: { description: "随机猫猫图。", messages: { msg: "{quote}{image}", wait: "{quote}{time}\n请等待图片上传(可能较慢)。", failed: "{quote}{time}\n获取失败:{data}" } }, getqqinfo: { description: "获取 QQ 号的信息。", messages: { msg: "{quote}{image}", failed: "{quote}{time}\n获取失败:{data}", command: "QQ 号不得为空。" } } } };
|
|
38
38
|
}
|
|
39
39
|
});
|
|
40
40
|
|
|
@@ -261,11 +261,75 @@ async function getHttp(log, url, timeout) {
|
|
|
261
261
|
}
|
|
262
262
|
}
|
|
263
263
|
__name(getHttp, "getHttp");
|
|
264
|
+
async function request(url, options = {}, timeout = 8e3, log) {
|
|
265
|
+
const signal = AbortSignal.timeout(timeout);
|
|
266
|
+
try {
|
|
267
|
+
const response = await fetch(url, { ...options, signal });
|
|
268
|
+
let responseData;
|
|
269
|
+
let isJson;
|
|
270
|
+
const text = await response.text();
|
|
271
|
+
try {
|
|
272
|
+
responseData = JSON.parse(text);
|
|
273
|
+
isJson = true;
|
|
274
|
+
} catch {
|
|
275
|
+
responseData = text;
|
|
276
|
+
isJson = false;
|
|
277
|
+
}
|
|
278
|
+
if (!response.ok) {
|
|
279
|
+
log?.error(`HTTP Error ${response.status}: ${url}`, responseData);
|
|
280
|
+
return {
|
|
281
|
+
success: false,
|
|
282
|
+
code: response.status,
|
|
283
|
+
error: responseData || `HTTP ${response.status}`,
|
|
284
|
+
isJson
|
|
285
|
+
};
|
|
286
|
+
}
|
|
287
|
+
log?.info(`HTTP ${response.status}: ${url}`);
|
|
288
|
+
return {
|
|
289
|
+
success: true,
|
|
290
|
+
data: responseData
|
|
291
|
+
};
|
|
292
|
+
} catch (error) {
|
|
293
|
+
const isTimeout = error.name === "TimeoutError" || error.name === "AbortError";
|
|
294
|
+
const errorMessage = isTimeout ? `请求超时。(${timeout}ms)` : error.message;
|
|
295
|
+
log?.error(`Request Failed: ${errorMessage}`);
|
|
296
|
+
return {
|
|
297
|
+
success: false,
|
|
298
|
+
error: { name: error.name, message: errorMessage },
|
|
299
|
+
isJson: false
|
|
300
|
+
};
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
__name(request, "request");
|
|
304
|
+
async function readUserCardFile(userInfo) {
|
|
305
|
+
let card;
|
|
306
|
+
try {
|
|
307
|
+
const aPath = import_path.default.resolve(__dirname, "..") + import_path.default.sep + "res" + import_path.default.sep + "userCard.html";
|
|
308
|
+
card = await import_fs.default.promises.readFile(aPath, "utf8");
|
|
309
|
+
let sex_so;
|
|
310
|
+
let sex;
|
|
311
|
+
if (userInfo.sex == "male") {
|
|
312
|
+
sex = "♂";
|
|
313
|
+
sex_so = "sex-male";
|
|
314
|
+
} else if (userInfo.sex == "female") {
|
|
315
|
+
sex = "♀";
|
|
316
|
+
sex_so = "sex-female";
|
|
317
|
+
} else {
|
|
318
|
+
sex = "猫娘";
|
|
319
|
+
sex_so = "sex-unknown";
|
|
320
|
+
}
|
|
321
|
+
card = card.toString().replace("{avatarUrl}", userInfo.avatar_url).replace("{nickname}", userInfo.nickname).replace("{sex}", sex).replace("{sex-so}", sex_so).replace("{age}", String(userInfo.age)).replace("{longNick}", userInfo.long_nick == "" ? "<无>" : `“ ${userInfo.long_nick} ”`).replace("{qq}", userInfo.qq).replace("{qqLevel}", String(userInfo.qq_level)).replace("{qid}", userInfo.qid == "" ? "<无>" : userInfo.qid).replace("{location}", userInfo.location == "" ? "<无>" : userInfo.location).replace("{email}", userInfo.email == "" ? "<无>" : userInfo.email).replace("{regTime}", userInfo.reg_time).replace("{lastUpdated}", userInfo.last_updated);
|
|
322
|
+
} catch (error) {
|
|
323
|
+
card = error.message;
|
|
324
|
+
}
|
|
325
|
+
return card;
|
|
326
|
+
}
|
|
327
|
+
__name(readUserCardFile, "readUserCardFile");
|
|
264
328
|
|
|
265
329
|
// src/commands.ts
|
|
266
330
|
async function getServer(ctx, session) {
|
|
267
331
|
const log = ctx.logger("cx");
|
|
268
|
-
log.
|
|
332
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
269
333
|
let msg;
|
|
270
334
|
let data;
|
|
271
335
|
let error;
|
|
@@ -365,7 +429,7 @@ async function getServer(ctx, session) {
|
|
|
365
429
|
__name(getServer, "getServer");
|
|
366
430
|
async function getStatus(ctx, session) {
|
|
367
431
|
const log = ctx.logger("status");
|
|
368
|
-
log.
|
|
432
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
369
433
|
const time = getHongKongTime();
|
|
370
434
|
let msg;
|
|
371
435
|
const vMsg = await getSystemUsage();
|
|
@@ -399,7 +463,7 @@ async function getStatus(ctx, session) {
|
|
|
399
463
|
__name(getStatus, "getStatus");
|
|
400
464
|
async function getRandom(ctx, session, min, max) {
|
|
401
465
|
const log = ctx.logger("random");
|
|
402
|
-
log.
|
|
466
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
403
467
|
const time = getHongKongTime();
|
|
404
468
|
let msg;
|
|
405
469
|
let data;
|
|
@@ -422,7 +486,7 @@ async function getRandom(ctx, session, min, max) {
|
|
|
422
486
|
__name(getRandom, "getRandom");
|
|
423
487
|
async function getInfo(ctx, session) {
|
|
424
488
|
const log = ctx.logger("info");
|
|
425
|
-
log.
|
|
489
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
426
490
|
const time = getHongKongTime();
|
|
427
491
|
let msg;
|
|
428
492
|
let data = await readInfoFile(ctx);
|
|
@@ -446,9 +510,9 @@ async function getInfo(ctx, session) {
|
|
|
446
510
|
return msg;
|
|
447
511
|
}
|
|
448
512
|
__name(getInfo, "getInfo");
|
|
449
|
-
async function
|
|
513
|
+
async function getRandomWord(ctx, session) {
|
|
450
514
|
const log = ctx.logger("rw");
|
|
451
|
-
log.
|
|
515
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
452
516
|
let msg;
|
|
453
517
|
let data;
|
|
454
518
|
const time = getHongKongTime();
|
|
@@ -495,12 +559,12 @@ async function getRW(ctx, session) {
|
|
|
495
559
|
}
|
|
496
560
|
return msg;
|
|
497
561
|
}
|
|
498
|
-
__name(
|
|
499
|
-
async function
|
|
562
|
+
__name(getRandomWord, "getRandomWord");
|
|
563
|
+
async function getBlueArchive(ctx, session) {
|
|
500
564
|
const log = ctx.logger("ba");
|
|
501
|
-
log.
|
|
565
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
502
566
|
const time = getHongKongTime();
|
|
503
|
-
if (ctx.config.
|
|
567
|
+
if (ctx.config.baAPI == void 0) {
|
|
504
568
|
await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": "未指定 API" }));
|
|
505
569
|
return 1;
|
|
506
570
|
}
|
|
@@ -513,10 +577,10 @@ async function getBA(ctx, session) {
|
|
|
513
577
|
await session.bot.deleteMessage(session.event.guild?.id, vid[0]);
|
|
514
578
|
return 0;
|
|
515
579
|
}
|
|
516
|
-
__name(
|
|
580
|
+
__name(getBlueArchive, "getBlueArchive");
|
|
517
581
|
async function serverTest(ctx, session) {
|
|
518
582
|
const log = ctx.logger("serverTest");
|
|
519
|
-
log.
|
|
583
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
520
584
|
const time = getHongKongTime();
|
|
521
585
|
const host = ctx.config.serverPing[`${session.event.guild?.id}`];
|
|
522
586
|
if (host == void 0) {
|
|
@@ -551,109 +615,126 @@ async function serverTest(ctx, session) {
|
|
|
551
615
|
};
|
|
552
616
|
}
|
|
553
617
|
__name(serverTest, "serverTest");
|
|
554
|
-
async function getSteam(ctx, session, id) {
|
|
555
|
-
const log = ctx.logger("steamId");
|
|
556
|
-
log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
557
|
-
let msg;
|
|
558
|
-
let data;
|
|
559
|
-
const time = getHongKongTime();
|
|
560
|
-
const response = await getHttp(log, ctx.config.steamAPI + `?format=json&id=${id}`, ctx.config.timeout);
|
|
561
|
-
if (response.success) {
|
|
562
|
-
data = response.data;
|
|
563
|
-
msg = {
|
|
564
|
-
"time": time,
|
|
565
|
-
"data": data["data"],
|
|
566
|
-
"success": 0
|
|
567
|
-
};
|
|
568
|
-
log.info("Sent:");
|
|
569
|
-
log.info(msg);
|
|
570
|
-
} else {
|
|
571
|
-
if (response.error) {
|
|
572
|
-
data = response.data;
|
|
573
|
-
msg = {
|
|
574
|
-
"time": time,
|
|
575
|
-
"data": data["name"] === "AbortError" ? session.text(".command") : data["message"],
|
|
576
|
-
"success": 2
|
|
577
|
-
};
|
|
578
|
-
log.info("Sent:");
|
|
579
|
-
log.info(msg);
|
|
580
|
-
} else {
|
|
581
|
-
data = response.data;
|
|
582
|
-
msg = {
|
|
583
|
-
"time": time,
|
|
584
|
-
"data": data["data"],
|
|
585
|
-
"success": 1
|
|
586
|
-
};
|
|
587
|
-
log.info("Sent:");
|
|
588
|
-
log.info(msg);
|
|
589
|
-
}
|
|
590
|
-
}
|
|
591
|
-
return msg;
|
|
592
|
-
}
|
|
593
|
-
__name(getSteam, "getSteam");
|
|
594
618
|
async function getMeme(ctx, session, count) {
|
|
595
619
|
const log = ctx.logger("getMeme");
|
|
596
|
-
log.
|
|
620
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
597
621
|
let msg;
|
|
598
|
-
let data;
|
|
599
622
|
const time = getHongKongTime();
|
|
600
|
-
|
|
601
|
-
if (api == void 0) {
|
|
623
|
+
if (ctx.config.memesAPI[`${session.event.guild?.id}`] == void 0) {
|
|
602
624
|
msg = {
|
|
603
625
|
"time": time,
|
|
604
626
|
"quote": import_koishi2.h.quote(session.messageId),
|
|
605
627
|
"success": ".forbidden"
|
|
606
628
|
};
|
|
607
|
-
log.
|
|
608
|
-
log.
|
|
629
|
+
log.warn("Sent:");
|
|
630
|
+
log.warn(msg);
|
|
609
631
|
await session.send(session.text(msg["success"], msg));
|
|
610
632
|
return 0;
|
|
611
633
|
}
|
|
612
|
-
const
|
|
634
|
+
const api = count ? ctx.config.memesAPI[`${session.event.guild?.id}`] + `&type=1&count=${count}` : ctx.config.memesAPI[`${session.event.guild?.id}`];
|
|
635
|
+
const response = await request(api, {}, ctx.config.timeout, log);
|
|
613
636
|
if (response.success) {
|
|
614
|
-
data = response.data;
|
|
615
637
|
msg = {
|
|
616
638
|
"time": time,
|
|
617
|
-
"title": data
|
|
618
|
-
"image": import_koishi2.h.image(data
|
|
639
|
+
"title": response.data.data.title,
|
|
640
|
+
"image": import_koishi2.h.image(response.data.data.image),
|
|
619
641
|
"quote": import_koishi2.h.quote(session.messageId),
|
|
620
642
|
"success": ".msg"
|
|
621
643
|
};
|
|
622
|
-
log.
|
|
623
|
-
log.
|
|
644
|
+
log.debug("Sent:");
|
|
645
|
+
log.debug(msg);
|
|
624
646
|
} else {
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
|
|
628
|
-
"time": time,
|
|
629
|
-
"data": data["name"] === "AbortError" ? session.text(".error") : data["message"],
|
|
630
|
-
"quote": import_koishi2.h.quote(session.messageId),
|
|
631
|
-
"success": ".failed"
|
|
632
|
-
};
|
|
633
|
-
log.info("Sent:");
|
|
634
|
-
log.info(msg);
|
|
647
|
+
let err;
|
|
648
|
+
if (response.code) {
|
|
649
|
+
err = response.isJson ? response.error["data"] : response.error;
|
|
635
650
|
} else {
|
|
636
|
-
|
|
637
|
-
msg = {
|
|
638
|
-
"time": time,
|
|
639
|
-
"data": data["data"],
|
|
640
|
-
"quote": import_koishi2.h.quote(session.messageId),
|
|
641
|
-
"success": ".failed"
|
|
642
|
-
};
|
|
643
|
-
log.info("Sent:");
|
|
644
|
-
log.info(msg);
|
|
651
|
+
err = response.error.message;
|
|
645
652
|
}
|
|
653
|
+
msg = {
|
|
654
|
+
"time": time,
|
|
655
|
+
"data": err,
|
|
656
|
+
"quote": import_koishi2.h.quote(session.messageId),
|
|
657
|
+
"success": ".failed"
|
|
658
|
+
};
|
|
659
|
+
log.warn("Sent:");
|
|
660
|
+
log.warn(msg);
|
|
646
661
|
}
|
|
647
662
|
await session.send(session.text(msg["success"], msg));
|
|
648
663
|
return 0;
|
|
649
664
|
}
|
|
650
665
|
__name(getMeme, "getMeme");
|
|
666
|
+
async function getCat(ctx, session) {
|
|
667
|
+
const log = ctx.logger("cat");
|
|
668
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
669
|
+
const time = getHongKongTime();
|
|
670
|
+
if (ctx.config.catAPI == void 0) {
|
|
671
|
+
await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": "未指定 API", "time": time }));
|
|
672
|
+
return 1;
|
|
673
|
+
}
|
|
674
|
+
const vid = await session.send(session.text(".wait", { "quote": import_koishi2.h.quote(session.messageId), "time": time }));
|
|
675
|
+
const response = await request(ctx.config.catAPI, {}, ctx.config.timeout, log);
|
|
676
|
+
if (response.success) {
|
|
677
|
+
log.debug(response.data);
|
|
678
|
+
await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(response.data[0].url) }));
|
|
679
|
+
log.debug("Sent:");
|
|
680
|
+
log.debug(response.data[0].url);
|
|
681
|
+
} else {
|
|
682
|
+
if (response.code) {
|
|
683
|
+
await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": response.isJson ? response.error["error"] : response.error, "time": time }));
|
|
684
|
+
log.warn("Sent:");
|
|
685
|
+
log.warn(response.error);
|
|
686
|
+
} else {
|
|
687
|
+
await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": response.error.message, "time": time }));
|
|
688
|
+
log.warn("Sent:");
|
|
689
|
+
log.warn(response.error);
|
|
690
|
+
}
|
|
691
|
+
}
|
|
692
|
+
await session.bot.deleteMessage(session.event.guild?.id, vid[0]);
|
|
693
|
+
return 0;
|
|
694
|
+
}
|
|
695
|
+
__name(getCat, "getCat");
|
|
696
|
+
async function getQQInfo(ctx, session, qq) {
|
|
697
|
+
const log = ctx.logger("getQQInfo");
|
|
698
|
+
log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
|
|
699
|
+
const time = getHongKongTime();
|
|
700
|
+
if (ctx.config.qqAPI == void 0) {
|
|
701
|
+
await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": "未指定 API", "time": time }));
|
|
702
|
+
return 1;
|
|
703
|
+
}
|
|
704
|
+
const response = await request(ctx.config.qqAPI + `?qq=${qq}`, {}, ctx.config.timeout, log);
|
|
705
|
+
if (response.success) {
|
|
706
|
+
log.info(response.data);
|
|
707
|
+
const fullHtml = await readUserCardFile(response.data);
|
|
708
|
+
const page = await ctx.puppeteer.page();
|
|
709
|
+
await page.setViewport({
|
|
710
|
+
width: 450,
|
|
711
|
+
height: 650,
|
|
712
|
+
deviceScaleFactor: 2
|
|
713
|
+
});
|
|
714
|
+
await page.setContent(fullHtml, { waitUntil: "networkidle0" });
|
|
715
|
+
const image = await page.screenshot({ type: "png", omitBackground: true });
|
|
716
|
+
await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(image, "image/png") }));
|
|
717
|
+
log.debug("Sent: Image");
|
|
718
|
+
} else {
|
|
719
|
+
if (response.code) {
|
|
720
|
+
await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": response.isJson ? response.error["error"] : response.error, "time": time }));
|
|
721
|
+
log.warn("Sent:");
|
|
722
|
+
log.warn(response.error);
|
|
723
|
+
} else {
|
|
724
|
+
await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": response.error.message, "time": time }));
|
|
725
|
+
log.warn("Sent:");
|
|
726
|
+
log.warn(response.error);
|
|
727
|
+
}
|
|
728
|
+
}
|
|
729
|
+
return 0;
|
|
730
|
+
}
|
|
731
|
+
__name(getQQInfo, "getQQInfo");
|
|
651
732
|
|
|
652
733
|
// package.json
|
|
653
|
-
var version = "2.
|
|
734
|
+
var version = "2.18.0";
|
|
654
735
|
|
|
655
736
|
// src/index.ts
|
|
656
|
-
var inject = ["database", "installer"];
|
|
737
|
+
var inject = ["database", "installer", "puppeteer"];
|
|
657
738
|
var name = "node-async-bot-all";
|
|
658
739
|
var usage = "这是一个私有插件。";
|
|
659
740
|
var Config = import_koishi3.Schema.intersect([
|
|
@@ -683,7 +764,13 @@ var Config = import_koishi3.Schema.intersect([
|
|
|
683
764
|
}).description("转换 Steam ID"),
|
|
684
765
|
import_koishi3.Schema.object({
|
|
685
766
|
memesAPI: import_koishi3.Schema.dict(String).role("table").description("群友 meme API")
|
|
686
|
-
}).description("群友 meme")
|
|
767
|
+
}).description("群友 meme"),
|
|
768
|
+
import_koishi3.Schema.object({
|
|
769
|
+
catAPI: import_koishi3.Schema.string().default("https://api.thecatapi.com/v1/images/search").description("随机猫猫图 API")
|
|
770
|
+
}).description("随机猫猫图"),
|
|
771
|
+
import_koishi3.Schema.object({
|
|
772
|
+
qqAPI: import_koishi3.Schema.string().default("https://uapis.cn/api/v1/social/qq/userinfo").description("获取 QQ 信息 API")
|
|
773
|
+
}).description("获取 QQ 信息")
|
|
687
774
|
]).description("基础设置");
|
|
688
775
|
function apply(ctx) {
|
|
689
776
|
ctx.i18n.define("zh-CN", require_zh_CN());
|
|
@@ -709,7 +796,7 @@ function apply(ctx) {
|
|
|
709
796
|
return session?.text(".failed", cx);
|
|
710
797
|
}
|
|
711
798
|
});
|
|
712
|
-
ctx.command("status").action(async ({ session }) => {
|
|
799
|
+
ctx.command("status").alias("stats").alias("状态").action(async ({ session }) => {
|
|
713
800
|
const status = await getStatus(ctx, session);
|
|
714
801
|
if (status["success"] == 0) {
|
|
715
802
|
return session?.text(".msg", status);
|
|
@@ -730,7 +817,7 @@ function apply(ctx) {
|
|
|
730
817
|
}
|
|
731
818
|
});
|
|
732
819
|
ctx.command("rw").action(async ({ session }) => {
|
|
733
|
-
const rw = await
|
|
820
|
+
const rw = await getRandomWord(ctx, session);
|
|
734
821
|
if (rw["success"] == 0) {
|
|
735
822
|
return session?.text(".msg", rw);
|
|
736
823
|
} else if (rw["success"] == 1) {
|
|
@@ -740,7 +827,7 @@ function apply(ctx) {
|
|
|
740
827
|
}
|
|
741
828
|
});
|
|
742
829
|
ctx.command("randomBA").alias("随机ba图").action(async ({ session }) => {
|
|
743
|
-
await
|
|
830
|
+
await getBlueArchive(ctx, session);
|
|
744
831
|
});
|
|
745
832
|
ctx.command("serverTest").alias("服之测测").action(async ({ session }) => {
|
|
746
833
|
const server = await serverTest(ctx, session);
|
|
@@ -752,23 +839,15 @@ function apply(ctx) {
|
|
|
752
839
|
return session?.text(".failed", server);
|
|
753
840
|
}
|
|
754
841
|
});
|
|
755
|
-
ctx.command("steamId <Steam 好友码(SteamID 3):posint>").alias("steam").action(async ({ session }, id) => {
|
|
756
|
-
if (id == null) {
|
|
757
|
-
return session?.text(".null");
|
|
758
|
-
}
|
|
759
|
-
const steam = await getSteam(ctx, session, id);
|
|
760
|
-
if (steam["success"] == 0) {
|
|
761
|
-
return session?.text(".msg", steam);
|
|
762
|
-
} else {
|
|
763
|
-
return session?.text(".failed", steam);
|
|
764
|
-
}
|
|
765
|
-
});
|
|
766
842
|
ctx.command("meme [序号:posint]").alias("memes").action(async ({ session }, count) => {
|
|
767
|
-
|
|
768
|
-
|
|
769
|
-
|
|
770
|
-
|
|
771
|
-
|
|
843
|
+
await getMeme(ctx, session, count);
|
|
844
|
+
});
|
|
845
|
+
ctx.command("randomCat").alias("随机猫猫图").alias("随机猫猫").action(async ({ session }) => {
|
|
846
|
+
await getCat(ctx, session);
|
|
847
|
+
});
|
|
848
|
+
ctx.command("getQQInfo <QQ号:string>").alias("获取QQ信息").action(async ({ session }, qq) => {
|
|
849
|
+
if (qq == void 0) return session?.text(".command");
|
|
850
|
+
await getQQInfo(ctx, session, qq);
|
|
772
851
|
});
|
|
773
852
|
}
|
|
774
853
|
__name(apply, "apply");
|
package/package.json
CHANGED
|
@@ -0,0 +1,242 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="zh-CN">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8">
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
6
|
+
<title>User Profile Card</title>
|
|
7
|
+
<style>
|
|
8
|
+
/* 全局重置 */
|
|
9
|
+
* {
|
|
10
|
+
box-sizing: border-box;
|
|
11
|
+
margin: 0;
|
|
12
|
+
padding: 0;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
body {
|
|
16
|
+
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
|
|
17
|
+
background-color: #f0f2f5;
|
|
18
|
+
display: flex;
|
|
19
|
+
justify-content: center;
|
|
20
|
+
align-items: flex-start; /* 顶部对齐 */
|
|
21
|
+
min-height: 100vh;
|
|
22
|
+
padding: 20px;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
/* 卡片容器 */
|
|
26
|
+
.card {
|
|
27
|
+
width: 400px; /* 固定宽度,配合 puppeteer 视口 */
|
|
28
|
+
background: #ffffff;
|
|
29
|
+
border-radius: 16px;
|
|
30
|
+
box-shadow: 0 10px 30px rgba(0, 0, 0, 0.1);
|
|
31
|
+
overflow: hidden;
|
|
32
|
+
position: relative;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/* 顶部背景图/颜色 */
|
|
36
|
+
.card-header-bg {
|
|
37
|
+
height: 120px;
|
|
38
|
+
background: linear-gradient(135deg, #00C6FF 0%, #0072FF 100%);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/* 个人信息主要区域 */
|
|
42
|
+
.profile-main {
|
|
43
|
+
padding: 0 24px;
|
|
44
|
+
margin-top: -50px; /* 让头像上浮 */
|
|
45
|
+
text-align: center;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
/* 头像容器 */
|
|
49
|
+
.avatar-container {
|
|
50
|
+
position: relative;
|
|
51
|
+
display: inline-block;
|
|
52
|
+
padding: 4px;
|
|
53
|
+
background: #fff;
|
|
54
|
+
border-radius: 50%;
|
|
55
|
+
box-shadow: 0 4px 10px rgba(0, 0, 0, 0.1);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
.avatar {
|
|
59
|
+
width: 90px;
|
|
60
|
+
height: 90px;
|
|
61
|
+
border-radius: 50%;
|
|
62
|
+
object-fit: cover;
|
|
63
|
+
display: block;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
/* VIP 徽章 */
|
|
67
|
+
.vip-badge {
|
|
68
|
+
position: absolute;
|
|
69
|
+
bottom: 5px;
|
|
70
|
+
right: 5px;
|
|
71
|
+
background: #FFD700;
|
|
72
|
+
color: #d63031;
|
|
73
|
+
font-size: 10px;
|
|
74
|
+
font-weight: bold;
|
|
75
|
+
padding: 2px 6px;
|
|
76
|
+
border-radius: 8px;
|
|
77
|
+
border: 2px solid #fff;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
.nickname-row {
|
|
81
|
+
margin-top: 12px;
|
|
82
|
+
display: flex;
|
|
83
|
+
align-items: center;
|
|
84
|
+
justify-content: center;
|
|
85
|
+
gap: 8px;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
.nickname {
|
|
89
|
+
font-size: 22px;
|
|
90
|
+
font-weight: 700;
|
|
91
|
+
color: #333;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
/* 性别图标 */
|
|
95
|
+
.sex-icon {
|
|
96
|
+
font-size: 14px;
|
|
97
|
+
padding: 2px 6px;
|
|
98
|
+
border-radius: 4px;
|
|
99
|
+
color: white;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
.sex-male {
|
|
103
|
+
background-color: #007bff;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
.sex-female {
|
|
107
|
+
background-color: #ff6b81;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
.sex-unknown {
|
|
111
|
+
background-color: #ff6be1;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
.long-nick {
|
|
115
|
+
margin-top: 8px;
|
|
116
|
+
color: #666;
|
|
117
|
+
font-size: 14px;
|
|
118
|
+
font-style: italic;
|
|
119
|
+
margin-bottom: 24px;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
/* 详细信息列表 */
|
|
123
|
+
.info-list {
|
|
124
|
+
padding: 0 24px 24px 24px;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
.info-item {
|
|
128
|
+
display: flex;
|
|
129
|
+
justify-content: space-between;
|
|
130
|
+
padding: 12px 0;
|
|
131
|
+
border-bottom: 1px solid #f0f0f0;
|
|
132
|
+
font-size: 14px;
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
.info-item:last-child {
|
|
136
|
+
border-bottom: none;
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
.label {
|
|
140
|
+
color: #888;
|
|
141
|
+
font-weight: 500;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
.value {
|
|
145
|
+
color: #333;
|
|
146
|
+
font-weight: 600;
|
|
147
|
+
text-align: right;
|
|
148
|
+
max-width: 220px;
|
|
149
|
+
white-space: nowrap;
|
|
150
|
+
overflow: hidden;
|
|
151
|
+
text-overflow: ellipsis;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
/* 标签样式 (如 VIP 等级) */
|
|
155
|
+
.tag {
|
|
156
|
+
display: inline-block;
|
|
157
|
+
padding: 2px 8px;
|
|
158
|
+
border-radius: 12px;
|
|
159
|
+
font-size: 12px;
|
|
160
|
+
margin-left: 5px;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
.tag-vip {
|
|
164
|
+
background-color: #fff3cd;
|
|
165
|
+
color: #856404;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
.tag-level {
|
|
169
|
+
background-color: #e2e3e5;
|
|
170
|
+
color: #383d41;
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
/* 底部时间戳 */
|
|
174
|
+
.footer {
|
|
175
|
+
background-color: #f8f9fa;
|
|
176
|
+
padding: 12px 24px;
|
|
177
|
+
font-size: 12px;
|
|
178
|
+
color: #adb5bd;
|
|
179
|
+
text-align: center;
|
|
180
|
+
border-top: 1px solid #eee;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
</style>
|
|
184
|
+
</head>
|
|
185
|
+
<body>
|
|
186
|
+
|
|
187
|
+
<div class="card" id="card-element">
|
|
188
|
+
<!-- 顶部背景 -->
|
|
189
|
+
<div class="card-header-bg"></div>
|
|
190
|
+
|
|
191
|
+
<!-- 主要信息 -->
|
|
192
|
+
<div class="profile-main">
|
|
193
|
+
<div class="avatar-container">
|
|
194
|
+
<img src="{avatarUrl}" alt="Avatar" class="avatar">
|
|
195
|
+
</div>
|
|
196
|
+
|
|
197
|
+
<div class="nickname-row">
|
|
198
|
+
<span class="nickname">{nickname}</span>
|
|
199
|
+
<!-- 性别: 男 -->
|
|
200
|
+
<span class="sex-icon {sex-so}">{sex} | {age} 岁</span>
|
|
201
|
+
</div>
|
|
202
|
+
|
|
203
|
+
<div class="long-nick">{longNick}</div>
|
|
204
|
+
</div>
|
|
205
|
+
|
|
206
|
+
<!-- 详细数据列表 -->
|
|
207
|
+
<div class="info-list">
|
|
208
|
+
<div class="info-item">
|
|
209
|
+
<span class="label">QQ账号</span>
|
|
210
|
+
<span class="value">{qq}</span>
|
|
211
|
+
</div>
|
|
212
|
+
<div class="info-item">
|
|
213
|
+
<span class="label">QQ等级</span>
|
|
214
|
+
<span class="value">
|
|
215
|
+
<span class="tag tag-level">LV {qqLevel}</span>
|
|
216
|
+
</span>
|
|
217
|
+
</div>
|
|
218
|
+
<div class="info-item">
|
|
219
|
+
<span class="label">个性域名 (QID)</span>
|
|
220
|
+
<span class="value">{qid}</span>
|
|
221
|
+
</div>
|
|
222
|
+
<div class="info-item">
|
|
223
|
+
<span class="label">所在地</span>
|
|
224
|
+
<span class="value">{location}</span>
|
|
225
|
+
</div>
|
|
226
|
+
<div class="info-item">
|
|
227
|
+
<span class="label">电子邮箱</span>
|
|
228
|
+
<span class="value">{email}</span>
|
|
229
|
+
</div>
|
|
230
|
+
<div class="info-item">
|
|
231
|
+
<span class="label">注册时间</span>
|
|
232
|
+
<span class="value">{regTime}</span>
|
|
233
|
+
</div>
|
|
234
|
+
</div>
|
|
235
|
+
|
|
236
|
+
<div class="footer">
|
|
237
|
+
数据更新于: {lastUpdated}
|
|
238
|
+
</div>
|
|
239
|
+
</div>
|
|
240
|
+
|
|
241
|
+
</body>
|
|
242
|
+
</html>
|