koishi-plugin-node-async-bot-all 2.17.0 → 2.18.1

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 CHANGED
@@ -1,17 +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 getRW(ctx: Context, session: Session): Promise<Object>;
13
- export declare function getBA(ctx: Context, session: Session): Promise<Number>;
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>;
17
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;
@@ -25,6 +34,7 @@ export interface Config {
25
34
  steamAPI: string;
26
35
  memesAPI: Dict<string>;
27
36
  catAPI: string;
37
+ qqAPI: string;
28
38
  }
29
39
  export declare const Config: Schema<Config>;
30
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}" } }, steamid: { description: "将 Steam 好友码(SteamID 3)转为 SteamID 64。", messages: { msg: "{time}\n转换结果:{data}", failed: "{time}\n转换失败:{data}", null: "参数 Steam 好友码(SteamID 3) 不得为空。", command: "执行错误。" } }, 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}", error: "执行错误。", wait: "{quote}{time}\n请等待图片上传(可能较慢)。", failed: "{quote}{time}\n获取失败:{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,76 @@ 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(url);
296
+ log?.error(`Request Failed: ${errorMessage}`);
297
+ return {
298
+ success: false,
299
+ error: { name: error.name, message: errorMessage },
300
+ isJson: false
301
+ };
302
+ }
303
+ }
304
+ __name(request, "request");
305
+ async function readUserCardFile(userInfo) {
306
+ let card;
307
+ try {
308
+ const aPath = import_path.default.resolve(__dirname, "..") + import_path.default.sep + "res" + import_path.default.sep + "userCard.html";
309
+ card = await import_fs.default.promises.readFile(aPath, "utf8");
310
+ let sex_so;
311
+ let sex;
312
+ if (userInfo.sex == "male") {
313
+ sex = "♂";
314
+ sex_so = "sex-male";
315
+ } else if (userInfo.sex == "female") {
316
+ sex = "♀";
317
+ sex_so = "sex-female";
318
+ } else {
319
+ sex = "猫娘";
320
+ sex_so = "sex-unknown";
321
+ }
322
+ 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);
323
+ } catch (error) {
324
+ card = error.message;
325
+ }
326
+ return card;
327
+ }
328
+ __name(readUserCardFile, "readUserCardFile");
264
329
 
265
330
  // src/commands.ts
266
331
  async function getServer(ctx, session) {
267
332
  const log = ctx.logger("cx");
268
- log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
333
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
269
334
  let msg;
270
335
  let data;
271
336
  let error;
@@ -365,7 +430,7 @@ async function getServer(ctx, session) {
365
430
  __name(getServer, "getServer");
366
431
  async function getStatus(ctx, session) {
367
432
  const log = ctx.logger("status");
368
- log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
433
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
369
434
  const time = getHongKongTime();
370
435
  let msg;
371
436
  const vMsg = await getSystemUsage();
@@ -399,7 +464,7 @@ async function getStatus(ctx, session) {
399
464
  __name(getStatus, "getStatus");
400
465
  async function getRandom(ctx, session, min, max) {
401
466
  const log = ctx.logger("random");
402
- log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
467
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
403
468
  const time = getHongKongTime();
404
469
  let msg;
405
470
  let data;
@@ -422,7 +487,7 @@ async function getRandom(ctx, session, min, max) {
422
487
  __name(getRandom, "getRandom");
423
488
  async function getInfo(ctx, session) {
424
489
  const log = ctx.logger("info");
425
- log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
490
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
426
491
  const time = getHongKongTime();
427
492
  let msg;
428
493
  let data = await readInfoFile(ctx);
@@ -446,9 +511,9 @@ async function getInfo(ctx, session) {
446
511
  return msg;
447
512
  }
448
513
  __name(getInfo, "getInfo");
449
- async function getRW(ctx, session) {
514
+ async function getRandomWord(ctx, session) {
450
515
  const log = ctx.logger("rw");
451
- log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
516
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
452
517
  let msg;
453
518
  let data;
454
519
  const time = getHongKongTime();
@@ -495,10 +560,10 @@ async function getRW(ctx, session) {
495
560
  }
496
561
  return msg;
497
562
  }
498
- __name(getRW, "getRW");
499
- async function getBA(ctx, session) {
563
+ __name(getRandomWord, "getRandomWord");
564
+ async function getBlueArchive(ctx, session) {
500
565
  const log = ctx.logger("ba");
501
- log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
566
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
502
567
  const time = getHongKongTime();
503
568
  if (ctx.config.baAPI == void 0) {
504
569
  await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": "未指定 API" }));
@@ -509,14 +574,15 @@ async function getBA(ctx, session) {
509
574
  const link = random(2, ctx.config.baAPI) + `?cacheBuster=${random(1, 1, 2147483647)}`;
510
575
  log.info(`Link: ${link}`);
511
576
  await (0, import_koishi2.sleep)(ms);
512
- await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(link) }));
577
+ const status = await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(link) }));
578
+ if (!status) await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(link) }));
513
579
  await session.bot.deleteMessage(session.event.guild?.id, vid[0]);
514
580
  return 0;
515
581
  }
516
- __name(getBA, "getBA");
582
+ __name(getBlueArchive, "getBlueArchive");
517
583
  async function serverTest(ctx, session) {
518
584
  const log = ctx.logger("serverTest");
519
- log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
585
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
520
586
  const time = getHongKongTime();
521
587
  const host = ctx.config.serverPing[`${session.event.guild?.id}`];
522
588
  if (host == void 0) {
@@ -551,142 +617,128 @@ async function serverTest(ctx, session) {
551
617
  };
552
618
  }
553
619
  __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
620
  async function getMeme(ctx, session, count) {
595
621
  const log = ctx.logger("getMeme");
596
- log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
622
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
597
623
  let msg;
598
- let data;
599
624
  const time = getHongKongTime();
600
- const api = ctx.config.memesAPI[`${session.event.guild?.id}`];
601
- if (api == void 0) {
625
+ if (ctx.config.memesAPI[`${session.event.guild?.id}`] == void 0) {
602
626
  msg = {
603
627
  "time": time,
604
628
  "quote": import_koishi2.h.quote(session.messageId),
605
629
  "success": ".forbidden"
606
630
  };
607
- log.info("Sent:");
608
- log.info(msg);
631
+ log.warn("Sent:");
632
+ log.warn(msg);
609
633
  await session.send(session.text(msg["success"], msg));
610
634
  return 0;
611
635
  }
612
- const response = count ? await getHttp(log, api + `&type=1&count=${count}`, ctx.config.timeout) : await getHttp(log, api, ctx.config.timeout);
636
+ const api = count ? ctx.config.memesAPI[`${session.event.guild?.id}`] + `&type=1&count=${count}` : ctx.config.memesAPI[`${session.event.guild?.id}`];
637
+ const response = await request(api, {}, ctx.config.timeout, log);
613
638
  if (response.success) {
614
- data = response.data;
615
639
  msg = {
616
640
  "time": time,
617
- "title": data["data"]["title"],
618
- "image": import_koishi2.h.image(data["data"]["image"]),
641
+ "title": response.data.data.title,
642
+ "image": import_koishi2.h.image(response.data.data.image),
619
643
  "quote": import_koishi2.h.quote(session.messageId),
620
644
  "success": ".msg"
621
645
  };
622
- log.info("Sent:");
623
- log.info(msg);
646
+ log.debug("Sent:");
647
+ log.debug(msg);
624
648
  } else {
625
- if (response.error) {
626
- data = response.data;
627
- msg = {
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);
649
+ let err;
650
+ if (response.code) {
651
+ err = response.isJson ? response.error["data"] : response.error;
635
652
  } else {
636
- data = response.data;
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);
653
+ err = response.error.message;
645
654
  }
655
+ msg = {
656
+ "time": time,
657
+ "data": err,
658
+ "quote": import_koishi2.h.quote(session.messageId),
659
+ "success": ".failed"
660
+ };
661
+ log.warn("Sent:");
662
+ log.warn(msg);
646
663
  }
647
- await session.send(session.text(msg["success"], msg));
664
+ const status = await session.send(session.text(msg["success"], msg));
665
+ if (!status) await session.send(session.text(msg["success"], msg));
648
666
  return 0;
649
667
  }
650
668
  __name(getMeme, "getMeme");
651
669
  async function getCat(ctx, session) {
652
670
  const log = ctx.logger("cat");
653
- log.info(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
671
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
654
672
  const time = getHongKongTime();
655
673
  if (ctx.config.catAPI == void 0) {
656
674
  await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": "未指定 API", "time": time }));
657
675
  return 1;
658
676
  }
659
677
  const vid = await session.send(session.text(".wait", { "quote": import_koishi2.h.quote(session.messageId), "time": time }));
660
- const response = await getHttp(log, ctx.config.catAPI, ctx.config.timeout);
678
+ const response = await request(ctx.config.catAPI, {}, ctx.config.timeout, log);
661
679
  if (response.success) {
662
- const data = response.data;
663
- log.info(data[0]["url"]);
664
- await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(data[0]["url"]) }));
665
- log.info("Sent:");
666
- log.info(data[0]["url"]);
680
+ log.debug(response.data);
681
+ await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(response.data[0].url) }));
682
+ log.debug("Sent:");
683
+ log.debug(response.data[0].url);
667
684
  } else {
668
- if (response.error) {
669
- const data = response.data;
670
- await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": data["name"] === "AbortError" ? session.text(".error") : data["message"], "time": time }));
671
- log.info("Sent:");
672
- log.info(data["message"]);
685
+ if (response.code) {
686
+ await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": response.isJson ? response.error["error"] : response.error, "time": time }));
687
+ log.warn("Sent:");
688
+ log.warn(response.error);
673
689
  } else {
674
- const data = response.data;
675
- await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": data["data"], "time": time }));
676
- log.info("Sent:");
677
- log.info(data["data"]);
690
+ await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": response.error.message, "time": time }));
691
+ log.warn("Sent:");
692
+ log.warn(response.error);
678
693
  }
679
694
  }
680
695
  await session.bot.deleteMessage(session.event.guild?.id, vid[0]);
681
696
  return 0;
682
697
  }
683
698
  __name(getCat, "getCat");
699
+ async function getQQInfo(ctx, session, qq) {
700
+ const log = ctx.logger("getQQInfo");
701
+ log.debug(`Got: {"form":"${session.event.guild?.id}","user":"${session.event.user?.id}","timestamp":${session.event.timestamp},"messageId":"${session.event.message?.id}"}`);
702
+ const time = getHongKongTime();
703
+ if (ctx.config.qqAPI == void 0) {
704
+ await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": "未指定 API", "time": time }));
705
+ return 1;
706
+ }
707
+ const response = await request(ctx.config.qqAPI + `?qq=${qq}`, {}, ctx.config.timeout, log);
708
+ if (response.success) {
709
+ log.info(response.data);
710
+ const fullHtml = await readUserCardFile(response.data);
711
+ const page = await ctx.puppeteer.page();
712
+ await page.setViewport({
713
+ width: 450,
714
+ height: 650,
715
+ deviceScaleFactor: 2
716
+ });
717
+ await page.setContent(fullHtml, { waitUntil: "networkidle0" });
718
+ const image = await page.screenshot({ type: "png", omitBackground: true });
719
+ const status = await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(image, "image/png") }));
720
+ if (!status) await session.send(session.text(".msg", { "quote": import_koishi2.h.quote(session.messageId), "image": import_koishi2.h.image(image, "image/png") }));
721
+ log.debug("Sent: Image");
722
+ } else {
723
+ if (response.code) {
724
+ await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": response.isJson ? response.error["error"] : response.error, "time": time }));
725
+ log.warn("Sent:");
726
+ log.warn(response.error);
727
+ } else {
728
+ await session.send(session.text(".failed", { "quote": import_koishi2.h.quote(session.messageId), "data": response.error.message, "time": time }));
729
+ log.warn("Sent:");
730
+ log.warn(response.error);
731
+ }
732
+ }
733
+ return 0;
734
+ }
735
+ __name(getQQInfo, "getQQInfo");
684
736
 
685
737
  // package.json
686
- var version = "2.17.0";
738
+ var version = "2.18.1";
687
739
 
688
740
  // src/index.ts
689
- var inject = ["database", "installer"];
741
+ var inject = ["database", "installer", "puppeteer"];
690
742
  var name = "node-async-bot-all";
691
743
  var usage = "这是一个私有插件。";
692
744
  var Config = import_koishi3.Schema.intersect([
@@ -719,7 +771,10 @@ var Config = import_koishi3.Schema.intersect([
719
771
  }).description("群友 meme"),
720
772
  import_koishi3.Schema.object({
721
773
  catAPI: import_koishi3.Schema.string().default("https://api.thecatapi.com/v1/images/search").description("随机猫猫图 API")
722
- }).description("随机猫猫图")
774
+ }).description("随机猫猫图"),
775
+ import_koishi3.Schema.object({
776
+ qqAPI: import_koishi3.Schema.string().default("https://uapis.cn/api/v1/social/qq/userinfo").description("获取 QQ 信息 API")
777
+ }).description("获取 QQ 信息")
723
778
  ]).description("基础设置");
724
779
  function apply(ctx) {
725
780
  ctx.i18n.define("zh-CN", require_zh_CN());
@@ -766,7 +821,7 @@ function apply(ctx) {
766
821
  }
767
822
  });
768
823
  ctx.command("rw").action(async ({ session }) => {
769
- const rw = await getRW(ctx, session);
824
+ const rw = await getRandomWord(ctx, session);
770
825
  if (rw["success"] == 0) {
771
826
  return session?.text(".msg", rw);
772
827
  } else if (rw["success"] == 1) {
@@ -776,7 +831,7 @@ function apply(ctx) {
776
831
  }
777
832
  });
778
833
  ctx.command("randomBA").alias("随机ba图").action(async ({ session }) => {
779
- await getBA(ctx, session);
834
+ await getBlueArchive(ctx, session);
780
835
  });
781
836
  ctx.command("serverTest").alias("服之测测").action(async ({ session }) => {
782
837
  const server = await serverTest(ctx, session);
@@ -788,23 +843,16 @@ function apply(ctx) {
788
843
  return session?.text(".failed", server);
789
844
  }
790
845
  });
791
- ctx.command("steamId <Steam 好友码(SteamID 3):posint>").alias("steam").action(async ({ session }, id) => {
792
- if (id == null) {
793
- return session?.text(".null");
794
- }
795
- const steam = await getSteam(ctx, session, id);
796
- if (steam["success"] == 0) {
797
- return session?.text(".msg", steam);
798
- } else {
799
- return session?.text(".failed", steam);
800
- }
801
- });
802
846
  ctx.command("meme [序号:posint]").alias("memes").action(async ({ session }, count) => {
803
847
  await getMeme(ctx, session, count);
804
848
  });
805
849
  ctx.command("randomCat").alias("随机猫猫图").alias("随机猫猫").action(async ({ session }) => {
806
850
  await getCat(ctx, session);
807
851
  });
852
+ ctx.command("getQQInfo <QQ号:string>").alias("获取QQ信息").action(async ({ session }, qq) => {
853
+ if (qq == void 0 || isNaN(Number(qq))) return session?.text(".command");
854
+ await getQQInfo(ctx, session, qq);
855
+ });
808
856
  }
809
857
  __name(apply, "apply");
810
858
  // Annotate the CommonJS export names for ESM import in node:
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "koishi-plugin-node-async-bot-all",
3
3
  "description": "NodeAsync Bot插件(自用)",
4
- "version": "2.17.0",
4
+ "version": "2.18.1",
5
5
  "main": "lib/index.js",
6
6
  "typings": "lib/index.d.ts",
7
7
  "contributors": [
@@ -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>