@sliverp/qqbot 1.3.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.
Files changed (78) hide show
  1. package/README.md +231 -0
  2. package/clawdbot.plugin.json +16 -0
  3. package/dist/index.d.ts +17 -0
  4. package/dist/index.js +22 -0
  5. package/dist/src/api.d.ts +194 -0
  6. package/dist/src/api.js +555 -0
  7. package/dist/src/channel.d.ts +3 -0
  8. package/dist/src/channel.js +146 -0
  9. package/dist/src/config.d.ts +25 -0
  10. package/dist/src/config.js +148 -0
  11. package/dist/src/gateway.d.ts +17 -0
  12. package/dist/src/gateway.js +722 -0
  13. package/dist/src/image-server.d.ts +62 -0
  14. package/dist/src/image-server.js +401 -0
  15. package/dist/src/known-users.d.ts +100 -0
  16. package/dist/src/known-users.js +264 -0
  17. package/dist/src/onboarding.d.ts +10 -0
  18. package/dist/src/onboarding.js +190 -0
  19. package/dist/src/outbound.d.ts +149 -0
  20. package/dist/src/outbound.js +476 -0
  21. package/dist/src/proactive.d.ts +170 -0
  22. package/dist/src/proactive.js +398 -0
  23. package/dist/src/runtime.d.ts +3 -0
  24. package/dist/src/runtime.js +10 -0
  25. package/dist/src/session-store.d.ts +49 -0
  26. package/dist/src/session-store.js +242 -0
  27. package/dist/src/types.d.ts +116 -0
  28. package/dist/src/types.js +1 -0
  29. package/dist/src/utils/image-size.d.ts +51 -0
  30. package/dist/src/utils/image-size.js +234 -0
  31. package/dist/src/utils/payload.d.ts +112 -0
  32. package/dist/src/utils/payload.js +186 -0
  33. package/index.ts +27 -0
  34. package/moltbot.plugin.json +16 -0
  35. package/node_modules/ws/LICENSE +20 -0
  36. package/node_modules/ws/README.md +548 -0
  37. package/node_modules/ws/browser.js +8 -0
  38. package/node_modules/ws/index.js +13 -0
  39. package/node_modules/ws/lib/buffer-util.js +131 -0
  40. package/node_modules/ws/lib/constants.js +19 -0
  41. package/node_modules/ws/lib/event-target.js +292 -0
  42. package/node_modules/ws/lib/extension.js +203 -0
  43. package/node_modules/ws/lib/limiter.js +55 -0
  44. package/node_modules/ws/lib/permessage-deflate.js +528 -0
  45. package/node_modules/ws/lib/receiver.js +706 -0
  46. package/node_modules/ws/lib/sender.js +602 -0
  47. package/node_modules/ws/lib/stream.js +161 -0
  48. package/node_modules/ws/lib/subprotocol.js +62 -0
  49. package/node_modules/ws/lib/validation.js +152 -0
  50. package/node_modules/ws/lib/websocket-server.js +554 -0
  51. package/node_modules/ws/lib/websocket.js +1393 -0
  52. package/node_modules/ws/package.json +69 -0
  53. package/node_modules/ws/wrapper.mjs +8 -0
  54. package/openclaw.plugin.json +16 -0
  55. package/package.json +38 -0
  56. package/qqbot-1.3.0.tgz +0 -0
  57. package/scripts/proactive-api-server.ts +346 -0
  58. package/scripts/send-proactive.ts +273 -0
  59. package/scripts/upgrade.sh +106 -0
  60. package/skills/qqbot-cron/SKILL.md +490 -0
  61. package/skills/qqbot-media/SKILL.md +138 -0
  62. package/src/api.ts +752 -0
  63. package/src/channel.ts +303 -0
  64. package/src/config.ts +172 -0
  65. package/src/gateway.ts +1588 -0
  66. package/src/image-server.ts +474 -0
  67. package/src/known-users.ts +358 -0
  68. package/src/onboarding.ts +254 -0
  69. package/src/openclaw-plugin-sdk.d.ts +483 -0
  70. package/src/outbound.ts +571 -0
  71. package/src/proactive.ts +528 -0
  72. package/src/runtime.ts +14 -0
  73. package/src/session-store.ts +292 -0
  74. package/src/types.ts +123 -0
  75. package/src/utils/image-size.ts +266 -0
  76. package/src/utils/payload.ts +265 -0
  77. package/tsconfig.json +16 -0
  78. package/upgrade-and-run.sh +89 -0
@@ -0,0 +1,292 @@
1
+ /**
2
+ * Session 持久化存储
3
+ * 将 WebSocket 连接状态(sessionId、lastSeq)持久化到文件
4
+ * 支持进程重启后通过 Resume 机制快速恢复连接
5
+ */
6
+
7
+ import fs from "node:fs";
8
+ import path from "node:path";
9
+
10
+ // Session 状态接口
11
+ export interface SessionState {
12
+ /** WebSocket Session ID */
13
+ sessionId: string | null;
14
+ /** 最后收到的消息序号 */
15
+ lastSeq: number | null;
16
+ /** 上次连接成功的时间戳 */
17
+ lastConnectedAt: number;
18
+ /** 上次成功的权限级别索引 */
19
+ intentLevelIndex: number;
20
+ /** 关联的机器人账户 ID */
21
+ accountId: string;
22
+ /** 保存时间 */
23
+ savedAt: number;
24
+ }
25
+
26
+ // Session 文件目录
27
+ const SESSION_DIR = path.join(
28
+ process.env.HOME || "/tmp",
29
+ "clawd",
30
+ "qqbot-data"
31
+ );
32
+
33
+ // Session 过期时间(5分钟)- Resume 要求在断开后一定时间内恢复
34
+ const SESSION_EXPIRE_TIME = 5 * 60 * 1000;
35
+
36
+ // 写入节流时间(避免频繁写入)
37
+ const SAVE_THROTTLE_MS = 1000;
38
+
39
+ // 每个账户的节流状态
40
+ const throttleState = new Map<string, {
41
+ pendingState: SessionState | null;
42
+ lastSaveTime: number;
43
+ throttleTimer: ReturnType<typeof setTimeout> | null;
44
+ }>();
45
+
46
+ /**
47
+ * 确保目录存在
48
+ */
49
+ function ensureDir(): void {
50
+ if (!fs.existsSync(SESSION_DIR)) {
51
+ fs.mkdirSync(SESSION_DIR, { recursive: true });
52
+ }
53
+ }
54
+
55
+ /**
56
+ * 获取 Session 文件路径
57
+ */
58
+ function getSessionPath(accountId: string): string {
59
+ // 清理 accountId 中的特殊字符
60
+ const safeId = accountId.replace(/[^a-zA-Z0-9_-]/g, "_");
61
+ return path.join(SESSION_DIR, `session-${safeId}.json`);
62
+ }
63
+
64
+ /**
65
+ * 加载 Session 状态
66
+ * @param accountId 账户 ID
67
+ * @returns Session 状态,如果不存在或已过期返回 null
68
+ */
69
+ export function loadSession(accountId: string): SessionState | null {
70
+ const filePath = getSessionPath(accountId);
71
+
72
+ try {
73
+ if (!fs.existsSync(filePath)) {
74
+ return null;
75
+ }
76
+
77
+ const data = fs.readFileSync(filePath, "utf-8");
78
+ const state = JSON.parse(data) as SessionState;
79
+
80
+ // 检查是否过期
81
+ const now = Date.now();
82
+ if (now - state.savedAt > SESSION_EXPIRE_TIME) {
83
+ console.log(`[session-store] Session expired for ${accountId}, age: ${Math.round((now - state.savedAt) / 1000)}s`);
84
+ // 删除过期文件
85
+ try {
86
+ fs.unlinkSync(filePath);
87
+ } catch {
88
+ // 忽略删除错误
89
+ }
90
+ return null;
91
+ }
92
+
93
+ // 验证必要字段
94
+ if (!state.sessionId || state.lastSeq === null || state.lastSeq === undefined) {
95
+ console.log(`[session-store] Invalid session data for ${accountId}`);
96
+ return null;
97
+ }
98
+
99
+ console.log(`[session-store] Loaded session for ${accountId}: sessionId=${state.sessionId}, lastSeq=${state.lastSeq}, age=${Math.round((now - state.savedAt) / 1000)}s`);
100
+ return state;
101
+ } catch (err) {
102
+ console.error(`[session-store] Failed to load session for ${accountId}: ${err}`);
103
+ return null;
104
+ }
105
+ }
106
+
107
+ /**
108
+ * 保存 Session 状态(带节流,避免频繁写入)
109
+ * @param state Session 状态
110
+ */
111
+ export function saveSession(state: SessionState): void {
112
+ const { accountId } = state;
113
+
114
+ // 获取或初始化节流状态
115
+ let throttle = throttleState.get(accountId);
116
+ if (!throttle) {
117
+ throttle = {
118
+ pendingState: null,
119
+ lastSaveTime: 0,
120
+ throttleTimer: null,
121
+ };
122
+ throttleState.set(accountId, throttle);
123
+ }
124
+
125
+ const now = Date.now();
126
+ const timeSinceLastSave = now - throttle.lastSaveTime;
127
+
128
+ // 如果距离上次保存时间足够长,立即保存
129
+ if (timeSinceLastSave >= SAVE_THROTTLE_MS) {
130
+ doSaveSession(state);
131
+ throttle.lastSaveTime = now;
132
+ throttle.pendingState = null;
133
+
134
+ // 清除待定的节流定时器
135
+ if (throttle.throttleTimer) {
136
+ clearTimeout(throttle.throttleTimer);
137
+ throttle.throttleTimer = null;
138
+ }
139
+ } else {
140
+ // 记录待保存的状态
141
+ throttle.pendingState = state;
142
+
143
+ // 如果没有设置定时器,设置一个
144
+ if (!throttle.throttleTimer) {
145
+ const delay = SAVE_THROTTLE_MS - timeSinceLastSave;
146
+ throttle.throttleTimer = setTimeout(() => {
147
+ const t = throttleState.get(accountId);
148
+ if (t && t.pendingState) {
149
+ doSaveSession(t.pendingState);
150
+ t.lastSaveTime = Date.now();
151
+ t.pendingState = null;
152
+ }
153
+ if (t) {
154
+ t.throttleTimer = null;
155
+ }
156
+ }, delay);
157
+ }
158
+ }
159
+ }
160
+
161
+ /**
162
+ * 实际执行保存操作
163
+ */
164
+ function doSaveSession(state: SessionState): void {
165
+ const filePath = getSessionPath(state.accountId);
166
+
167
+ try {
168
+ ensureDir();
169
+
170
+ // 更新保存时间
171
+ const stateToSave: SessionState = {
172
+ ...state,
173
+ savedAt: Date.now(),
174
+ };
175
+
176
+ fs.writeFileSync(filePath, JSON.stringify(stateToSave, null, 2), "utf-8");
177
+ console.log(`[session-store] Saved session for ${state.accountId}: sessionId=${state.sessionId}, lastSeq=${state.lastSeq}`);
178
+ } catch (err) {
179
+ console.error(`[session-store] Failed to save session for ${state.accountId}: ${err}`);
180
+ }
181
+ }
182
+
183
+ /**
184
+ * 清除 Session 状态
185
+ * @param accountId 账户 ID
186
+ */
187
+ export function clearSession(accountId: string): void {
188
+ const filePath = getSessionPath(accountId);
189
+
190
+ // 清除节流状态
191
+ const throttle = throttleState.get(accountId);
192
+ if (throttle) {
193
+ if (throttle.throttleTimer) {
194
+ clearTimeout(throttle.throttleTimer);
195
+ }
196
+ throttleState.delete(accountId);
197
+ }
198
+
199
+ try {
200
+ if (fs.existsSync(filePath)) {
201
+ fs.unlinkSync(filePath);
202
+ console.log(`[session-store] Cleared session for ${accountId}`);
203
+ }
204
+ } catch (err) {
205
+ console.error(`[session-store] Failed to clear session for ${accountId}: ${err}`);
206
+ }
207
+ }
208
+
209
+ /**
210
+ * 更新 lastSeq(轻量级更新)
211
+ * @param accountId 账户 ID
212
+ * @param lastSeq 最新的消息序号
213
+ */
214
+ export function updateLastSeq(accountId: string, lastSeq: number): void {
215
+ const existing = loadSession(accountId);
216
+ if (existing && existing.sessionId) {
217
+ saveSession({
218
+ ...existing,
219
+ lastSeq,
220
+ });
221
+ }
222
+ }
223
+
224
+ /**
225
+ * 获取所有保存的 Session 状态
226
+ */
227
+ export function getAllSessions(): SessionState[] {
228
+ const sessions: SessionState[] = [];
229
+
230
+ try {
231
+ ensureDir();
232
+ const files = fs.readdirSync(SESSION_DIR);
233
+
234
+ for (const file of files) {
235
+ if (file.startsWith("session-") && file.endsWith(".json")) {
236
+ const filePath = path.join(SESSION_DIR, file);
237
+ try {
238
+ const data = fs.readFileSync(filePath, "utf-8");
239
+ const state = JSON.parse(data) as SessionState;
240
+ sessions.push(state);
241
+ } catch {
242
+ // 忽略解析错误
243
+ }
244
+ }
245
+ }
246
+ } catch {
247
+ // 目录不存在等错误
248
+ }
249
+
250
+ return sessions;
251
+ }
252
+
253
+ /**
254
+ * 清理过期的 Session 文件
255
+ */
256
+ export function cleanupExpiredSessions(): number {
257
+ let cleaned = 0;
258
+
259
+ try {
260
+ ensureDir();
261
+ const files = fs.readdirSync(SESSION_DIR);
262
+ const now = Date.now();
263
+
264
+ for (const file of files) {
265
+ if (file.startsWith("session-") && file.endsWith(".json")) {
266
+ const filePath = path.join(SESSION_DIR, file);
267
+ try {
268
+ const data = fs.readFileSync(filePath, "utf-8");
269
+ const state = JSON.parse(data) as SessionState;
270
+
271
+ if (now - state.savedAt > SESSION_EXPIRE_TIME) {
272
+ fs.unlinkSync(filePath);
273
+ cleaned++;
274
+ console.log(`[session-store] Cleaned expired session: ${file}`);
275
+ }
276
+ } catch {
277
+ // 忽略解析错误,但也删除损坏的文件
278
+ try {
279
+ fs.unlinkSync(filePath);
280
+ cleaned++;
281
+ } catch {
282
+ // 忽略
283
+ }
284
+ }
285
+ }
286
+ }
287
+ } catch {
288
+ // 目录不存在等错误
289
+ }
290
+
291
+ return cleaned;
292
+ }
package/src/types.ts ADDED
@@ -0,0 +1,123 @@
1
+ /**
2
+ * QQ Bot 配置类型
3
+ */
4
+ export interface QQBotConfig {
5
+ appId: string;
6
+ clientSecret?: string;
7
+ clientSecretFile?: string;
8
+ }
9
+
10
+ /**
11
+ * 解析后的 QQ Bot 账户
12
+ */
13
+ export interface ResolvedQQBotAccount {
14
+ accountId: string;
15
+ name?: string;
16
+ enabled: boolean;
17
+ appId: string;
18
+ clientSecret: string;
19
+ secretSource: "config" | "file" | "env" | "none";
20
+ /** 系统提示词 */
21
+ systemPrompt?: string;
22
+ /** 图床服务器公网地址 */
23
+ imageServerBaseUrl?: string;
24
+ /** 是否支持 markdown 消息(默认 false,需要机器人具备该权限才能启用) */
25
+ markdownSupport?: boolean;
26
+ config: QQBotAccountConfig;
27
+ }
28
+
29
+ /**
30
+ * QQ Bot 账户配置
31
+ */
32
+ export interface QQBotAccountConfig {
33
+ enabled?: boolean;
34
+ name?: string;
35
+ appId?: string;
36
+ clientSecret?: string;
37
+ clientSecretFile?: string;
38
+ dmPolicy?: "open" | "pairing" | "allowlist";
39
+ allowFrom?: string[];
40
+ /** 系统提示词,会添加在用户消息前面 */
41
+ systemPrompt?: string;
42
+ /** 图床服务器公网地址,用于发送图片,例如 http://your-ip:18765 */
43
+ imageServerBaseUrl?: string;
44
+ /** 是否支持 markdown 消息(默认 false,需要机器人具备该权限才能启用) */
45
+ markdownSupport?: boolean;
46
+ }
47
+
48
+ /**
49
+ * 富媒体附件
50
+ */
51
+ export interface MessageAttachment {
52
+ content_type: string; // 如 "image/png"
53
+ filename?: string;
54
+ height?: number;
55
+ width?: number;
56
+ size?: number;
57
+ url: string;
58
+ }
59
+
60
+ /**
61
+ * C2C 消息事件
62
+ */
63
+ export interface C2CMessageEvent {
64
+ author: {
65
+ id: string;
66
+ union_openid: string;
67
+ user_openid: string;
68
+ };
69
+ content: string;
70
+ id: string;
71
+ timestamp: string;
72
+ message_scene?: {
73
+ source: string;
74
+ };
75
+ attachments?: MessageAttachment[];
76
+ }
77
+
78
+ /**
79
+ * 频道 AT 消息事件
80
+ */
81
+ export interface GuildMessageEvent {
82
+ id: string;
83
+ channel_id: string;
84
+ guild_id: string;
85
+ content: string;
86
+ timestamp: string;
87
+ author: {
88
+ id: string;
89
+ username?: string;
90
+ bot?: boolean;
91
+ };
92
+ member?: {
93
+ nick?: string;
94
+ joined_at?: string;
95
+ };
96
+ attachments?: MessageAttachment[];
97
+ }
98
+
99
+ /**
100
+ * 群聊 AT 消息事件
101
+ */
102
+ export interface GroupMessageEvent {
103
+ author: {
104
+ id: string;
105
+ member_openid: string;
106
+ };
107
+ content: string;
108
+ id: string;
109
+ timestamp: string;
110
+ group_id: string;
111
+ group_openid: string;
112
+ attachments?: MessageAttachment[];
113
+ }
114
+
115
+ /**
116
+ * WebSocket 事件负载
117
+ */
118
+ export interface WSPayload {
119
+ op: number;
120
+ d?: unknown;
121
+ s?: number;
122
+ t?: string;
123
+ }
@@ -0,0 +1,266 @@
1
+ /**
2
+ * 图片尺寸工具
3
+ * 用于获取图片尺寸,生成 QQBot 的 markdown 图片格式
4
+ *
5
+ * QQBot markdown 图片格式: ![#宽px #高px](url)
6
+ */
7
+
8
+ import { Buffer } from "buffer";
9
+
10
+ export interface ImageSize {
11
+ width: number;
12
+ height: number;
13
+ }
14
+
15
+ /** 默认图片尺寸(当无法获取时使用) */
16
+ export const DEFAULT_IMAGE_SIZE: ImageSize = { width: 512, height: 512 };
17
+
18
+ /**
19
+ * 从 PNG 文件头解析图片尺寸
20
+ * PNG 文件头结构: 前 8 字节是签名,IHDR 块从第 8 字节开始
21
+ * IHDR 块: 长度(4) + 类型(4, "IHDR") + 宽度(4) + 高度(4) + ...
22
+ */
23
+ function parsePngSize(buffer: Buffer): ImageSize | null {
24
+ // PNG 签名: 89 50 4E 47 0D 0A 1A 0A
25
+ if (buffer.length < 24) return null;
26
+ if (buffer[0] !== 0x89 || buffer[1] !== 0x50 || buffer[2] !== 0x4E || buffer[3] !== 0x47) {
27
+ return null;
28
+ }
29
+ // IHDR 块从第 8 字节开始,宽度在第 16-19 字节,高度在第 20-23 字节
30
+ const width = buffer.readUInt32BE(16);
31
+ const height = buffer.readUInt32BE(20);
32
+ return { width, height };
33
+ }
34
+
35
+ /**
36
+ * 从 JPEG 文件解析图片尺寸
37
+ * JPEG 尺寸在 SOF0/SOF2 块中
38
+ */
39
+ function parseJpegSize(buffer: Buffer): ImageSize | null {
40
+ // JPEG 签名: FF D8 FF
41
+ if (buffer.length < 4) return null;
42
+ if (buffer[0] !== 0xFF || buffer[1] !== 0xD8) {
43
+ return null;
44
+ }
45
+
46
+ let offset = 2;
47
+ while (offset < buffer.length - 9) {
48
+ if (buffer[offset] !== 0xFF) {
49
+ offset++;
50
+ continue;
51
+ }
52
+
53
+ const marker = buffer[offset + 1];
54
+ // SOF0 (0xC0) 或 SOF2 (0xC2) 包含图片尺寸
55
+ if (marker === 0xC0 || marker === 0xC2) {
56
+ // 格式: FF C0 长度(2) 精度(1) 高度(2) 宽度(2)
57
+ if (offset + 9 <= buffer.length) {
58
+ const height = buffer.readUInt16BE(offset + 5);
59
+ const width = buffer.readUInt16BE(offset + 7);
60
+ return { width, height };
61
+ }
62
+ }
63
+
64
+ // 跳过当前块
65
+ if (offset + 3 < buffer.length) {
66
+ const blockLength = buffer.readUInt16BE(offset + 2);
67
+ offset += 2 + blockLength;
68
+ } else {
69
+ break;
70
+ }
71
+ }
72
+
73
+ return null;
74
+ }
75
+
76
+ /**
77
+ * 从 GIF 文件头解析图片尺寸
78
+ * GIF 文件头: GIF87a 或 GIF89a (6字节) + 宽度(2) + 高度(2)
79
+ */
80
+ function parseGifSize(buffer: Buffer): ImageSize | null {
81
+ if (buffer.length < 10) return null;
82
+ const signature = buffer.toString("ascii", 0, 6);
83
+ if (signature !== "GIF87a" && signature !== "GIF89a") {
84
+ return null;
85
+ }
86
+ const width = buffer.readUInt16LE(6);
87
+ const height = buffer.readUInt16LE(8);
88
+ return { width, height };
89
+ }
90
+
91
+ /**
92
+ * 从 WebP 文件解析图片尺寸
93
+ * WebP 文件头: RIFF(4) + 文件大小(4) + WEBP(4) + VP8/VP8L/VP8X(4) + ...
94
+ */
95
+ function parseWebpSize(buffer: Buffer): ImageSize | null {
96
+ if (buffer.length < 30) return null;
97
+
98
+ // 检查 RIFF 和 WEBP 签名
99
+ const riff = buffer.toString("ascii", 0, 4);
100
+ const webp = buffer.toString("ascii", 8, 12);
101
+ if (riff !== "RIFF" || webp !== "WEBP") {
102
+ return null;
103
+ }
104
+
105
+ const chunkType = buffer.toString("ascii", 12, 16);
106
+
107
+ // VP8 (有损压缩)
108
+ if (chunkType === "VP8 ") {
109
+ // VP8 帧头从第 23 字节开始,检查签名 9D 01 2A
110
+ if (buffer.length >= 30 && buffer[23] === 0x9D && buffer[24] === 0x01 && buffer[25] === 0x2A) {
111
+ const width = buffer.readUInt16LE(26) & 0x3FFF;
112
+ const height = buffer.readUInt16LE(28) & 0x3FFF;
113
+ return { width, height };
114
+ }
115
+ }
116
+
117
+ // VP8L (无损压缩)
118
+ if (chunkType === "VP8L") {
119
+ // VP8L 签名: 0x2F
120
+ if (buffer.length >= 25 && buffer[20] === 0x2F) {
121
+ const bits = buffer.readUInt32LE(21);
122
+ const width = (bits & 0x3FFF) + 1;
123
+ const height = ((bits >> 14) & 0x3FFF) + 1;
124
+ return { width, height };
125
+ }
126
+ }
127
+
128
+ // VP8X (扩展格式)
129
+ if (chunkType === "VP8X") {
130
+ if (buffer.length >= 30) {
131
+ // 宽度和高度在第 24-26 和 27-29 字节(24位小端)
132
+ const width = (buffer[24] | (buffer[25] << 8) | (buffer[26] << 16)) + 1;
133
+ const height = (buffer[27] | (buffer[28] << 8) | (buffer[29] << 16)) + 1;
134
+ return { width, height };
135
+ }
136
+ }
137
+
138
+ return null;
139
+ }
140
+
141
+ /**
142
+ * 从图片数据 Buffer 解析尺寸
143
+ */
144
+ export function parseImageSize(buffer: Buffer): ImageSize | null {
145
+ // 尝试各种格式
146
+ return parsePngSize(buffer)
147
+ ?? parseJpegSize(buffer)
148
+ ?? parseGifSize(buffer)
149
+ ?? parseWebpSize(buffer);
150
+ }
151
+
152
+ /**
153
+ * 从公网 URL 获取图片尺寸
154
+ * 只下载前 64KB 数据,足够解析大部分图片格式的头部
155
+ */
156
+ export async function getImageSizeFromUrl(url: string, timeoutMs = 5000): Promise<ImageSize | null> {
157
+ try {
158
+ const controller = new AbortController();
159
+ const timeoutId = setTimeout(() => controller.abort(), timeoutMs);
160
+
161
+ // 使用 Range 请求只获取前 64KB
162
+ const response = await fetch(url, {
163
+ signal: controller.signal,
164
+ headers: {
165
+ "Range": "bytes=0-65535",
166
+ "User-Agent": "QQBot-Image-Size-Detector/1.0",
167
+ },
168
+ });
169
+
170
+ clearTimeout(timeoutId);
171
+
172
+ if (!response.ok && response.status !== 206) {
173
+ console.log(`[image-size] Failed to fetch ${url}: ${response.status}`);
174
+ return null;
175
+ }
176
+
177
+ const arrayBuffer = await response.arrayBuffer();
178
+ const buffer = Buffer.from(arrayBuffer);
179
+
180
+ const size = parseImageSize(buffer);
181
+ if (size) {
182
+ console.log(`[image-size] Got size from URL: ${size.width}x${size.height} - ${url.slice(0, 60)}...`);
183
+ }
184
+
185
+ return size;
186
+ } catch (err) {
187
+ console.log(`[image-size] Error fetching ${url.slice(0, 60)}...: ${err}`);
188
+ return null;
189
+ }
190
+ }
191
+
192
+ /**
193
+ * 从 Base64 Data URL 获取图片尺寸
194
+ */
195
+ export function getImageSizeFromDataUrl(dataUrl: string): ImageSize | null {
196
+ try {
197
+ // 格式: 
198
+ const matches = dataUrl.match(/^data:image\/[^;]+;base64,(.+)$/);
199
+ if (!matches) {
200
+ return null;
201
+ }
202
+
203
+ const base64Data = matches[1];
204
+ const buffer = Buffer.from(base64Data, "base64");
205
+
206
+ const size = parseImageSize(buffer);
207
+ if (size) {
208
+ console.log(`[image-size] Got size from Base64: ${size.width}x${size.height}`);
209
+ }
210
+
211
+ return size;
212
+ } catch (err) {
213
+ console.log(`[image-size] Error parsing Base64: ${err}`);
214
+ return null;
215
+ }
216
+ }
217
+
218
+ /**
219
+ * 获取图片尺寸(自动判断来源)
220
+ * @param source - 图片 URL 或 Base64 Data URL
221
+ * @returns 图片尺寸,失败返回 null
222
+ */
223
+ export async function getImageSize(source: string): Promise<ImageSize | null> {
224
+ if (source.startsWith("data:")) {
225
+ return getImageSizeFromDataUrl(source);
226
+ }
227
+
228
+ if (source.startsWith("http://") || source.startsWith("https://")) {
229
+ return getImageSizeFromUrl(source);
230
+ }
231
+
232
+ return null;
233
+ }
234
+
235
+ /**
236
+ * 生成 QQBot markdown 图片格式
237
+ * 格式: ![#宽px #高px](url)
238
+ *
239
+ * @param url - 图片 URL
240
+ * @param size - 图片尺寸,如果为 null 则使用默认尺寸
241
+ * @returns QQBot markdown 图片字符串
242
+ */
243
+ export function formatQQBotMarkdownImage(url: string, size: ImageSize | null): string {
244
+ const { width, height } = size ?? DEFAULT_IMAGE_SIZE;
245
+ return `![#${width}px #${height}px](${url})`;
246
+ }
247
+
248
+ /**
249
+ * 检查 markdown 图片是否已经包含 QQBot 格式的尺寸信息
250
+ * 格式: ![#宽px #高px](url)
251
+ */
252
+ export function hasQQBotImageSize(markdownImage: string): boolean {
253
+ return /!\[#\d+px\s+#\d+px\]/.test(markdownImage);
254
+ }
255
+
256
+ /**
257
+ * 从已有的 QQBot 格式 markdown 图片中提取尺寸
258
+ * 格式: ![#宽px #高px](url)
259
+ */
260
+ export function extractQQBotImageSize(markdownImage: string): ImageSize | null {
261
+ const match = markdownImage.match(/!\[#(\d+)px\s+#(\d+)px\]/);
262
+ if (match) {
263
+ return { width: parseInt(match[1], 10), height: parseInt(match[2], 10) };
264
+ }
265
+ return null;
266
+ }