@leoqlin/openclaw-qqbot 1.6.7-alpha1

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 (218) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +484 -0
  3. package/README.zh.md +479 -0
  4. package/bin/qqbot-cli.js +243 -0
  5. package/dist/index.d.ts +17 -0
  6. package/dist/index.js +26 -0
  7. package/dist/src/admin-resolver.d.ts +33 -0
  8. package/dist/src/admin-resolver.js +157 -0
  9. package/dist/src/api.d.ts +301 -0
  10. package/dist/src/api.js +890 -0
  11. package/dist/src/channel.d.ts +29 -0
  12. package/dist/src/channel.js +452 -0
  13. package/dist/src/config.d.ts +56 -0
  14. package/dist/src/config.js +278 -0
  15. package/dist/src/credential-backup.d.ts +31 -0
  16. package/dist/src/credential-backup.js +66 -0
  17. package/dist/src/deliver-debounce.d.ts +74 -0
  18. package/dist/src/deliver-debounce.js +174 -0
  19. package/dist/src/gateway.d.ts +18 -0
  20. package/dist/src/gateway.js +2005 -0
  21. package/dist/src/group-history.d.ts +136 -0
  22. package/dist/src/group-history.js +226 -0
  23. package/dist/src/image-server.d.ts +87 -0
  24. package/dist/src/image-server.js +570 -0
  25. package/dist/src/inbound-attachments.d.ts +60 -0
  26. package/dist/src/inbound-attachments.js +248 -0
  27. package/dist/src/known-users.d.ts +100 -0
  28. package/dist/src/known-users.js +263 -0
  29. package/dist/src/message-gating.d.ts +53 -0
  30. package/dist/src/message-gating.js +107 -0
  31. package/dist/src/message-queue.d.ts +89 -0
  32. package/dist/src/message-queue.js +257 -0
  33. package/dist/src/onboarding.d.ts +10 -0
  34. package/dist/src/onboarding.js +203 -0
  35. package/dist/src/outbound-deliver.d.ts +48 -0
  36. package/dist/src/outbound-deliver.js +392 -0
  37. package/dist/src/outbound.d.ts +205 -0
  38. package/dist/src/outbound.js +938 -0
  39. package/dist/src/proactive.d.ts +170 -0
  40. package/dist/src/proactive.js +399 -0
  41. package/dist/src/ref-index-store.d.ts +101 -0
  42. package/dist/src/ref-index-store.js +298 -0
  43. package/dist/src/reply-dispatcher.d.ts +35 -0
  44. package/dist/src/reply-dispatcher.js +311 -0
  45. package/dist/src/request-context.d.ts +25 -0
  46. package/dist/src/request-context.js +37 -0
  47. package/dist/src/runtime.d.ts +3 -0
  48. package/dist/src/runtime.js +10 -0
  49. package/dist/src/session-store.d.ts +52 -0
  50. package/dist/src/session-store.js +254 -0
  51. package/dist/src/slash-commands.d.ts +77 -0
  52. package/dist/src/slash-commands.js +1866 -0
  53. package/dist/src/startup-greeting.d.ts +30 -0
  54. package/dist/src/startup-greeting.js +97 -0
  55. package/dist/src/streaming.d.ts +247 -0
  56. package/dist/src/streaming.js +899 -0
  57. package/dist/src/stt.d.ts +21 -0
  58. package/dist/src/stt.js +70 -0
  59. package/dist/src/tools/channel.d.ts +16 -0
  60. package/dist/src/tools/channel.js +234 -0
  61. package/dist/src/tools/remind.d.ts +2 -0
  62. package/dist/src/tools/remind.js +256 -0
  63. package/dist/src/types.d.ts +367 -0
  64. package/dist/src/types.js +17 -0
  65. package/dist/src/typing-keepalive.d.ts +27 -0
  66. package/dist/src/typing-keepalive.js +64 -0
  67. package/dist/src/update-checker.d.ts +36 -0
  68. package/dist/src/update-checker.js +171 -0
  69. package/dist/src/utils/audio-convert.d.ts +98 -0
  70. package/dist/src/utils/audio-convert.js +755 -0
  71. package/dist/src/utils/chunked-upload.d.ts +68 -0
  72. package/dist/src/utils/chunked-upload.js +341 -0
  73. package/dist/src/utils/file-utils.d.ts +61 -0
  74. package/dist/src/utils/file-utils.js +172 -0
  75. package/dist/src/utils/image-size.d.ts +51 -0
  76. package/dist/src/utils/image-size.js +234 -0
  77. package/dist/src/utils/media-send.d.ts +158 -0
  78. package/dist/src/utils/media-send.js +499 -0
  79. package/dist/src/utils/media-tags.d.ts +14 -0
  80. package/dist/src/utils/media-tags.js +165 -0
  81. package/dist/src/utils/payload.d.ts +112 -0
  82. package/dist/src/utils/payload.js +186 -0
  83. package/dist/src/utils/pkg-version.d.ts +5 -0
  84. package/dist/src/utils/pkg-version.js +61 -0
  85. package/dist/src/utils/platform.d.ts +137 -0
  86. package/dist/src/utils/platform.js +390 -0
  87. package/dist/src/utils/ssrf-guard.d.ts +25 -0
  88. package/dist/src/utils/ssrf-guard.js +91 -0
  89. package/dist/src/utils/text-parsing.d.ts +36 -0
  90. package/dist/src/utils/text-parsing.js +75 -0
  91. package/dist/src/utils/upload-cache.d.ts +34 -0
  92. package/dist/src/utils/upload-cache.js +93 -0
  93. package/index.ts +31 -0
  94. package/node_modules/@eshaz/web-worker/LICENSE +201 -0
  95. package/node_modules/@eshaz/web-worker/README.md +134 -0
  96. package/node_modules/@eshaz/web-worker/browser.js +17 -0
  97. package/node_modules/@eshaz/web-worker/cjs/browser.js +16 -0
  98. package/node_modules/@eshaz/web-worker/cjs/node.js +219 -0
  99. package/node_modules/@eshaz/web-worker/index.d.ts +4 -0
  100. package/node_modules/@eshaz/web-worker/node.js +223 -0
  101. package/node_modules/@eshaz/web-worker/package.json +54 -0
  102. package/node_modules/@wasm-audio-decoders/common/index.js +5 -0
  103. package/node_modules/@wasm-audio-decoders/common/package.json +36 -0
  104. package/node_modules/@wasm-audio-decoders/common/src/WASMAudioDecoderCommon.js +231 -0
  105. package/node_modules/@wasm-audio-decoders/common/src/WASMAudioDecoderWorker.js +129 -0
  106. package/node_modules/@wasm-audio-decoders/common/src/puff/README +67 -0
  107. package/node_modules/@wasm-audio-decoders/common/src/puff/build_puff.js +31 -0
  108. package/node_modules/@wasm-audio-decoders/common/src/puff/puff.c +863 -0
  109. package/node_modules/@wasm-audio-decoders/common/src/puff/puff.h +35 -0
  110. package/node_modules/@wasm-audio-decoders/common/src/utilities.js +3 -0
  111. package/node_modules/@wasm-audio-decoders/common/types.d.ts +7 -0
  112. package/node_modules/mpg123-decoder/README.md +265 -0
  113. package/node_modules/mpg123-decoder/dist/mpg123-decoder.min.js +185 -0
  114. package/node_modules/mpg123-decoder/dist/mpg123-decoder.min.js.map +1 -0
  115. package/node_modules/mpg123-decoder/index.js +8 -0
  116. package/node_modules/mpg123-decoder/package.json +58 -0
  117. package/node_modules/mpg123-decoder/src/EmscriptenWasm.js +464 -0
  118. package/node_modules/mpg123-decoder/src/MPEGDecoder.js +200 -0
  119. package/node_modules/mpg123-decoder/src/MPEGDecoderWebWorker.js +21 -0
  120. package/node_modules/mpg123-decoder/types.d.ts +30 -0
  121. package/node_modules/silk-wasm/LICENSE +21 -0
  122. package/node_modules/silk-wasm/README.md +85 -0
  123. package/node_modules/silk-wasm/lib/index.cjs +16 -0
  124. package/node_modules/silk-wasm/lib/index.d.ts +70 -0
  125. package/node_modules/silk-wasm/lib/index.mjs +16 -0
  126. package/node_modules/silk-wasm/lib/silk.wasm +0 -0
  127. package/node_modules/silk-wasm/lib/utils.d.ts +4 -0
  128. package/node_modules/silk-wasm/package.json +39 -0
  129. package/node_modules/simple-yenc/.github/FUNDING.yml +1 -0
  130. package/node_modules/simple-yenc/.prettierignore +1 -0
  131. package/node_modules/simple-yenc/LICENSE +7 -0
  132. package/node_modules/simple-yenc/README.md +163 -0
  133. package/node_modules/simple-yenc/dist/esm.js +1 -0
  134. package/node_modules/simple-yenc/dist/index.js +1 -0
  135. package/node_modules/simple-yenc/package.json +50 -0
  136. package/node_modules/simple-yenc/rollup.config.js +27 -0
  137. package/node_modules/simple-yenc/src/simple-yenc.js +302 -0
  138. package/node_modules/ws/LICENSE +20 -0
  139. package/node_modules/ws/README.md +548 -0
  140. package/node_modules/ws/browser.js +8 -0
  141. package/node_modules/ws/index.js +22 -0
  142. package/node_modules/ws/lib/buffer-util.js +131 -0
  143. package/node_modules/ws/lib/constants.js +19 -0
  144. package/node_modules/ws/lib/event-target.js +292 -0
  145. package/node_modules/ws/lib/extension.js +203 -0
  146. package/node_modules/ws/lib/limiter.js +55 -0
  147. package/node_modules/ws/lib/permessage-deflate.js +528 -0
  148. package/node_modules/ws/lib/receiver.js +706 -0
  149. package/node_modules/ws/lib/sender.js +602 -0
  150. package/node_modules/ws/lib/stream.js +161 -0
  151. package/node_modules/ws/lib/subprotocol.js +62 -0
  152. package/node_modules/ws/lib/validation.js +152 -0
  153. package/node_modules/ws/lib/websocket-server.js +554 -0
  154. package/node_modules/ws/lib/websocket.js +1393 -0
  155. package/node_modules/ws/package.json +70 -0
  156. package/node_modules/ws/wrapper.mjs +21 -0
  157. package/openclaw.plugin.json +17 -0
  158. package/package.json +70 -0
  159. package/preload.cjs +33 -0
  160. package/scripts/cleanup-legacy-plugins.sh +124 -0
  161. package/scripts/link-sdk-core.cjs +185 -0
  162. package/scripts/postinstall-link-sdk.js +126 -0
  163. package/scripts/proactive-api-server.ts +369 -0
  164. package/scripts/send-proactive.ts +293 -0
  165. package/scripts/set-markdown.sh +156 -0
  166. package/scripts/test-sendmedia.ts +116 -0
  167. package/scripts/upgrade-via-npm.ps1 +460 -0
  168. package/scripts/upgrade-via-npm.sh +652 -0
  169. package/scripts/upgrade-via-source.sh +1026 -0
  170. package/skills/qqbot-channel/SKILL.md +263 -0
  171. package/skills/qqbot-channel/references/api_references.md +521 -0
  172. package/skills/qqbot-media/SKILL.md +60 -0
  173. package/skills/qqbot-remind/SKILL.md +159 -0
  174. package/src/admin-resolver.ts +181 -0
  175. package/src/api.ts +1284 -0
  176. package/src/channel.ts +477 -0
  177. package/src/config.ts +347 -0
  178. package/src/credential-backup.ts +72 -0
  179. package/src/deliver-debounce.ts +229 -0
  180. package/src/gateway.ts +2245 -0
  181. package/src/group-history.ts +328 -0
  182. package/src/image-server.ts +675 -0
  183. package/src/inbound-attachments.ts +321 -0
  184. package/src/known-users.ts +353 -0
  185. package/src/message-gating.ts +190 -0
  186. package/src/message-queue.ts +352 -0
  187. package/src/onboarding.ts +274 -0
  188. package/src/openclaw-plugin-sdk.d.ts +587 -0
  189. package/src/outbound-deliver.ts +473 -0
  190. package/src/outbound.ts +1131 -0
  191. package/src/proactive.ts +530 -0
  192. package/src/ref-index-store.ts +412 -0
  193. package/src/reply-dispatcher.ts +334 -0
  194. package/src/request-context.ts +49 -0
  195. package/src/runtime.ts +14 -0
  196. package/src/session-store.ts +303 -0
  197. package/src/slash-commands.ts +2030 -0
  198. package/src/startup-greeting.ts +120 -0
  199. package/src/streaming.ts +1077 -0
  200. package/src/stt.ts +86 -0
  201. package/src/tools/channel.ts +281 -0
  202. package/src/tools/remind.ts +308 -0
  203. package/src/types.ts +391 -0
  204. package/src/typing-keepalive.ts +59 -0
  205. package/src/update-checker.ts +186 -0
  206. package/src/utils/audio-convert.ts +859 -0
  207. package/src/utils/chunked-upload.ts +483 -0
  208. package/src/utils/file-utils.ts +193 -0
  209. package/src/utils/image-size.ts +266 -0
  210. package/src/utils/media-send.ts +631 -0
  211. package/src/utils/media-tags.ts +183 -0
  212. package/src/utils/payload.ts +265 -0
  213. package/src/utils/pkg-version.ts +64 -0
  214. package/src/utils/platform.ts +435 -0
  215. package/src/utils/ssrf-guard.ts +102 -0
  216. package/src/utils/text-parsing.ts +85 -0
  217. package/src/utils/upload-cache.ts +128 -0
  218. package/tsconfig.json +16 -0
@@ -0,0 +1,334 @@
1
+ import path from "node:path";
2
+ import type { ResolvedQQBotAccount } from "./types.js";
3
+ import { getAccessToken, sendC2CMessage, sendChannelMessage, sendGroupMessage, clearTokenCache, sendC2CImageMessage, sendGroupImageMessage, sendC2CVoiceMessage, sendGroupVoiceMessage, sendC2CVideoMessage, sendGroupVideoMessage, sendC2CFileMessage, sendGroupFileMessage } from "./api.js";
4
+ import { parseQQBotPayload, encodePayloadForCron, isCronReminderPayload, isMediaPayload, type MediaPayload } from "./utils/payload.js";
5
+ import { resolveTTSConfig, textToSilk, formatDuration } from "./utils/audio-convert.js";
6
+ import { checkFileSize, readFileAsync, fileExistsAsync, formatFileSize, getMaxUploadSize } from "./utils/file-utils.js";
7
+ import { getQQBotDataDir, normalizePath, sanitizeFileName } from "./utils/platform.js";
8
+
9
+ export interface MessageTarget {
10
+ type: "c2c" | "guild" | "dm" | "group";
11
+ senderId: string;
12
+ messageId: string;
13
+ channelId?: string;
14
+ groupOpenid?: string;
15
+ }
16
+
17
+ export interface ReplyContext {
18
+ target: MessageTarget;
19
+ account: ResolvedQQBotAccount;
20
+ cfg: unknown;
21
+ log?: {
22
+ info: (msg: string) => void;
23
+ error: (msg: string) => void;
24
+ debug?: (msg: string) => void;
25
+ };
26
+ }
27
+
28
+ /**
29
+ * 带 token 过期重试的消息发送
30
+ */
31
+ export async function sendWithTokenRetry<T>(
32
+ appId: string,
33
+ clientSecret: string,
34
+ sendFn: (token: string) => Promise<T>,
35
+ log?: ReplyContext["log"],
36
+ accountId?: string,
37
+ ): Promise<T> {
38
+ try {
39
+ const token = await getAccessToken(appId, clientSecret);
40
+ return await sendFn(token);
41
+ } catch (err) {
42
+ const errMsg = String(err);
43
+ if (errMsg.includes("401") || errMsg.includes("token") || errMsg.includes("access_token")) {
44
+ log?.info(`[qqbot:${accountId}] Token may be expired, refreshing...`);
45
+ clearTokenCache(appId);
46
+ const newToken = await getAccessToken(appId, clientSecret);
47
+ return await sendFn(newToken);
48
+ } else {
49
+ throw err;
50
+ }
51
+ }
52
+ }
53
+
54
+ /**
55
+ * 根据消息类型路由发送文本
56
+ */
57
+ export async function sendTextToTarget(
58
+ ctx: ReplyContext,
59
+ text: string,
60
+ refIdx?: string,
61
+ ): Promise<void> {
62
+ const { target, account } = ctx;
63
+ await sendWithTokenRetry(account.appId, account.clientSecret, async (token) => {
64
+ if (target.type === "c2c") {
65
+ await sendC2CMessage(token, target.senderId, text, target.messageId, refIdx);
66
+ } else if (target.type === "group" && target.groupOpenid) {
67
+ await sendGroupMessage(token, target.groupOpenid, text, target.messageId);
68
+ } else if (target.channelId) {
69
+ await sendChannelMessage(token, target.channelId, text, target.messageId);
70
+ } else if (target.type === "dm") {
71
+ await sendC2CMessage(token, target.senderId, text, target.messageId, refIdx);
72
+ }
73
+ }, ctx.log, account.accountId);
74
+ }
75
+
76
+ /**
77
+ * 发送错误提示给用户
78
+ */
79
+ export async function sendErrorToTarget(ctx: ReplyContext, errorText: string): Promise<void> {
80
+ try {
81
+ await sendTextToTarget(ctx, errorText);
82
+ } catch (sendErr) {
83
+ ctx.log?.error(`[qqbot:${ctx.account.accountId}] Failed to send error message: ${sendErr}`);
84
+ }
85
+ }
86
+
87
+ /**
88
+ * 处理结构化载荷(QQBOT_PAYLOAD: 前缀的 JSON)
89
+ * 返回 true 表示已处理,false 表示不是结构化载荷
90
+ */
91
+ export async function handleStructuredPayload(
92
+ ctx: ReplyContext,
93
+ replyText: string,
94
+ recordActivity: () => void,
95
+ ): Promise<boolean> {
96
+ const { target, account, cfg, log } = ctx;
97
+ const payloadResult = parseQQBotPayload(replyText);
98
+
99
+ if (!payloadResult.isPayload) return false;
100
+
101
+ if (payloadResult.error) {
102
+ log?.error(`[qqbot:${account.accountId}] Payload parse error: ${payloadResult.error}`);
103
+ return true;
104
+ }
105
+
106
+ if (!payloadResult.payload) return true;
107
+
108
+ const parsedPayload = payloadResult.payload;
109
+ log?.info(`[qqbot:${account.accountId}] Detected structured payload, type: ${parsedPayload.type}`);
110
+
111
+ if (isCronReminderPayload(parsedPayload)) {
112
+ log?.info(`[qqbot:${account.accountId}] Processing cron_reminder payload`);
113
+ const cronMessage = encodePayloadForCron(parsedPayload);
114
+ const confirmText = `⏰ 提醒已设置,将在指定时间发送: "${parsedPayload.content}"`;
115
+ try {
116
+ await sendTextToTarget(ctx, confirmText);
117
+ log?.info(`[qqbot:${account.accountId}] Cron reminder confirmation sent, cronMessage: ${cronMessage}`);
118
+ } catch (err) {
119
+ log?.error(`[qqbot:${account.accountId}] Failed to send cron confirmation: ${err}`);
120
+ }
121
+ recordActivity();
122
+ return true;
123
+ }
124
+
125
+ if (isMediaPayload(parsedPayload)) {
126
+ log?.info(`[qqbot:${account.accountId}] Processing media payload, mediaType: ${parsedPayload.mediaType}`);
127
+
128
+ if (parsedPayload.mediaType === "image") {
129
+ await handleImagePayload(ctx, parsedPayload);
130
+ } else if (parsedPayload.mediaType === "audio") {
131
+ await handleAudioPayload(ctx, parsedPayload);
132
+ } else if (parsedPayload.mediaType === "video") {
133
+ await handleVideoPayload(ctx, parsedPayload);
134
+ } else if (parsedPayload.mediaType === "file") {
135
+ await handleFilePayload(ctx, parsedPayload);
136
+ } else {
137
+ log?.error(`[qqbot:${account.accountId}] Unknown media type: ${(parsedPayload as MediaPayload).mediaType}`);
138
+ }
139
+ recordActivity();
140
+ return true;
141
+ }
142
+
143
+ log?.error(`[qqbot:${account.accountId}] Unknown payload type: ${(parsedPayload as any).type}`);
144
+ return true;
145
+ }
146
+
147
+ // ============ 媒体载荷处理 ============
148
+
149
+ async function handleImagePayload(ctx: ReplyContext, payload: MediaPayload): Promise<void> {
150
+ const { target, account, log } = ctx;
151
+ let imageUrl = normalizePath(payload.path);
152
+ const originalImagePath = payload.source === "file" ? imageUrl : undefined;
153
+
154
+ if (payload.source === "file") {
155
+ try {
156
+ if (!(await fileExistsAsync(imageUrl))) {
157
+ log?.error(`[qqbot:${account.accountId}] Image not found: ${imageUrl}`);
158
+ return;
159
+ }
160
+ const imgSzCheck = checkFileSize(imageUrl, getMaxUploadSize(1)); // IMAGE = 1
161
+ if (!imgSzCheck.ok) {
162
+ log?.error(`[qqbot:${account.accountId}] Image size check failed: ${imgSzCheck.error}`);
163
+ return;
164
+ }
165
+ const fileBuffer = await readFileAsync(imageUrl);
166
+ const base64Data = fileBuffer.toString("base64");
167
+ const ext = path.extname(imageUrl).toLowerCase();
168
+ const mimeTypes: Record<string, string> = {
169
+ ".jpg": "image/jpeg", ".jpeg": "image/jpeg", ".png": "image/png",
170
+ ".gif": "image/gif", ".webp": "image/webp", ".bmp": "image/bmp",
171
+ };
172
+ const mimeType = mimeTypes[ext];
173
+ if (!mimeType) {
174
+ log?.error(`[qqbot:${account.accountId}] Unsupported image format: ${ext}`);
175
+ return;
176
+ }
177
+ imageUrl = `data:${mimeType};base64,${base64Data}`;
178
+ log?.info(`[qqbot:${account.accountId}] Converted local image to Base64 (size: ${formatFileSize(fileBuffer.length)})`);
179
+ } catch (readErr) {
180
+ log?.error(`[qqbot:${account.accountId}] Failed to read local image: ${readErr}`);
181
+ return;
182
+ }
183
+ }
184
+
185
+ try {
186
+ await sendWithTokenRetry(account.appId, account.clientSecret, async (token) => {
187
+ if (target.type === "c2c") {
188
+ await sendC2CImageMessage(token, target.senderId, imageUrl, target.messageId, undefined, originalImagePath);
189
+ } else if (target.type === "group" && target.groupOpenid) {
190
+ await sendGroupImageMessage(token, target.groupOpenid, imageUrl, target.messageId);
191
+ } else if (target.channelId) {
192
+ await sendChannelMessage(token, target.channelId, `![](${payload.path})`, target.messageId);
193
+ }
194
+ }, log, account.accountId);
195
+ log?.info(`[qqbot:${account.accountId}] Sent image via media payload`);
196
+
197
+ if (payload.caption) {
198
+ await sendTextToTarget(ctx, payload.caption);
199
+ }
200
+ } catch (err) {
201
+ log?.error(`[qqbot:${account.accountId}] Failed to send image: ${err}`);
202
+ }
203
+ }
204
+
205
+ async function handleAudioPayload(ctx: ReplyContext, payload: MediaPayload): Promise<void> {
206
+ const { target, account, cfg, log } = ctx;
207
+ try {
208
+ const ttsText = payload.caption || payload.path;
209
+ if (!ttsText?.trim()) {
210
+ log?.error(`[qqbot:${account.accountId}] Voice missing text`);
211
+ } else {
212
+ const ttsCfg = resolveTTSConfig(cfg as Record<string, unknown>);
213
+ if (!ttsCfg) {
214
+ log?.error(`[qqbot:${account.accountId}] TTS not configured (channels.qqbot.tts in openclaw.json)`);
215
+ } else {
216
+ log?.info(`[qqbot:${account.accountId}] TTS: "${ttsText.slice(0, 50)}..." via ${ttsCfg.model}`);
217
+ const ttsDir = getQQBotDataDir("tts");
218
+ const { silkPath, silkBase64, duration } = await textToSilk(ttsText, ttsCfg, ttsDir);
219
+ log?.info(`[qqbot:${account.accountId}] TTS done: ${formatDuration(duration)}, file saved: ${silkPath}`);
220
+
221
+ await sendWithTokenRetry(account.appId, account.clientSecret, async (token) => {
222
+ if (target.type === "c2c") {
223
+ await sendC2CVoiceMessage(token, target.senderId, silkBase64, target.messageId, ttsText, silkPath);
224
+ } else if (target.type === "group" && target.groupOpenid) {
225
+ await sendGroupVoiceMessage(token, target.groupOpenid, silkBase64, target.messageId);
226
+ } else if (target.channelId) {
227
+ log?.error(`[qqbot:${account.accountId}] Voice not supported in channel, sending text fallback`);
228
+ await sendChannelMessage(token, target.channelId, ttsText, target.messageId);
229
+ }
230
+ }, log, account.accountId);
231
+ log?.info(`[qqbot:${account.accountId}] Voice message sent`);
232
+ }
233
+ }
234
+ } catch (err) {
235
+ log?.error(`[qqbot:${account.accountId}] TTS/voice send failed: ${err}`);
236
+ }
237
+ }
238
+
239
+ async function handleVideoPayload(ctx: ReplyContext, payload: MediaPayload): Promise<void> {
240
+ const { target, account, log } = ctx;
241
+ try {
242
+ const videoPath = normalizePath(payload.path ?? "");
243
+ if (!videoPath?.trim()) {
244
+ log?.error(`[qqbot:${account.accountId}] Video missing path`);
245
+ } else {
246
+ const isHttpUrl = videoPath.startsWith("http://") || videoPath.startsWith("https://");
247
+ log?.info(`[qqbot:${account.accountId}] Video send: "${videoPath.slice(0, 60)}..."`);
248
+
249
+ await sendWithTokenRetry(account.appId, account.clientSecret, async (token) => {
250
+ if (isHttpUrl) {
251
+ if (target.type === "c2c") {
252
+ await sendC2CVideoMessage(token, target.senderId, videoPath, undefined, target.messageId);
253
+ } else if (target.type === "group" && target.groupOpenid) {
254
+ await sendGroupVideoMessage(token, target.groupOpenid, videoPath, undefined, target.messageId);
255
+ } else if (target.channelId) {
256
+ log?.error(`[qqbot:${account.accountId}] Video not supported in channel`);
257
+ }
258
+ } else {
259
+ if (!(await fileExistsAsync(videoPath))) {
260
+ throw new Error(`视频文件不存在: ${videoPath}`);
261
+ }
262
+ const vPaySzCheck = checkFileSize(videoPath, getMaxUploadSize(2)); // VIDEO = 2
263
+ if (!vPaySzCheck.ok) {
264
+ throw new Error(vPaySzCheck.error!);
265
+ }
266
+ const fileBuffer = await readFileAsync(videoPath);
267
+ const videoBase64 = fileBuffer.toString("base64");
268
+ log?.info(`[qqbot:${account.accountId}] Read local video (${formatFileSize(fileBuffer.length)}): ${videoPath}`);
269
+
270
+ if (target.type === "c2c") {
271
+ await sendC2CVideoMessage(token, target.senderId, undefined, videoBase64, target.messageId, undefined, videoPath);
272
+ } else if (target.type === "group" && target.groupOpenid) {
273
+ await sendGroupVideoMessage(token, target.groupOpenid, undefined, videoBase64, target.messageId);
274
+ } else if (target.channelId) {
275
+ log?.error(`[qqbot:${account.accountId}] Video not supported in channel`);
276
+ }
277
+ }
278
+ }, log, account.accountId);
279
+ log?.info(`[qqbot:${account.accountId}] Video message sent`);
280
+
281
+ if (payload.caption) {
282
+ await sendTextToTarget(ctx, payload.caption);
283
+ }
284
+ }
285
+ } catch (err) {
286
+ log?.error(`[qqbot:${account.accountId}] Video send failed: ${err}`);
287
+ }
288
+ }
289
+
290
+ async function handleFilePayload(ctx: ReplyContext, payload: MediaPayload): Promise<void> {
291
+ const { target, account, log } = ctx;
292
+ try {
293
+ const filePath = normalizePath(payload.path ?? "");
294
+ if (!filePath?.trim()) {
295
+ log?.error(`[qqbot:${account.accountId}] File missing path`);
296
+ } else {
297
+ const isHttpUrl = filePath.startsWith("http://") || filePath.startsWith("https://");
298
+ const fileName = sanitizeFileName(path.basename(filePath));
299
+ log?.info(`[qqbot:${account.accountId}] File send: "${filePath.slice(0, 60)}..." (${isHttpUrl ? "URL" : "local"})`);
300
+
301
+ await sendWithTokenRetry(account.appId, account.clientSecret, async (token) => {
302
+ if (isHttpUrl) {
303
+ if (target.type === "c2c") {
304
+ await sendC2CFileMessage(token, target.senderId, undefined, filePath, target.messageId, fileName);
305
+ } else if (target.type === "group" && target.groupOpenid) {
306
+ await sendGroupFileMessage(token, target.groupOpenid, undefined, filePath, target.messageId, fileName);
307
+ } else if (target.channelId) {
308
+ log?.error(`[qqbot:${account.accountId}] File not supported in channel`);
309
+ }
310
+ } else {
311
+ if (!(await fileExistsAsync(filePath))) {
312
+ throw new Error(`文件不存在: ${filePath}`);
313
+ }
314
+ const fPaySzCheck = checkFileSize(filePath, getMaxUploadSize(4)); // FILE = 4
315
+ if (!fPaySzCheck.ok) {
316
+ throw new Error(fPaySzCheck.error!);
317
+ }
318
+ const fileBuffer = await readFileAsync(filePath);
319
+ const fileBase64 = fileBuffer.toString("base64");
320
+ if (target.type === "c2c") {
321
+ await sendC2CFileMessage(token, target.senderId, fileBase64, undefined, target.messageId, fileName, filePath);
322
+ } else if (target.type === "group" && target.groupOpenid) {
323
+ await sendGroupFileMessage(token, target.groupOpenid, fileBase64, undefined, target.messageId, fileName);
324
+ } else if (target.channelId) {
325
+ log?.error(`[qqbot:${account.accountId}] File not supported in channel`);
326
+ }
327
+ }
328
+ }, log, account.accountId);
329
+ log?.info(`[qqbot:${account.accountId}] File message sent`);
330
+ }
331
+ } catch (err) {
332
+ log?.error(`[qqbot:${account.accountId}] File send failed: ${err}`);
333
+ }
334
+ }
@@ -0,0 +1,49 @@
1
+ /**
2
+ * 请求级上下文(基于 AsyncLocalStorage)
3
+ *
4
+ * 解决并发消息下工具获取当前会话信息的竞态问题。
5
+ * gateway 在处理每条入站消息时通过 runWithRequestContext() 建立作用域,
6
+ * 作用域内的所有异步代码(包括 AI agent 调用、tool execute)
7
+ * 都能通过 getRequestContext() 安全地拿到当前请求的上下文。
8
+ */
9
+ import { AsyncLocalStorage } from "node:async_hooks";
10
+
11
+ export interface RequestContext {
12
+ /** 投递目标地址,如 qqbot:c2c:xxx 或 qqbot:group:xxx */
13
+ target: string;
14
+ /** 当前请求的 QQBot 账户 ID(多账户场景) */
15
+ accountId?: string;
16
+ }
17
+
18
+ const asyncLocalStorage = new AsyncLocalStorage<RequestContext>();
19
+
20
+ /**
21
+ * 在请求级作用域中执行回调。
22
+ * 作用域内所有同步/异步代码都能通过 getRequestContext() 获取上下文。
23
+ */
24
+ export function runWithRequestContext<T>(ctx: RequestContext, fn: () => T): T {
25
+ return asyncLocalStorage.run(ctx, fn);
26
+ }
27
+
28
+ /**
29
+ * 获取当前请求的上下文,不存在时返回 undefined。
30
+ */
31
+ export function getRequestContext(): RequestContext | undefined {
32
+ return asyncLocalStorage.getStore();
33
+ }
34
+
35
+ /**
36
+ * 获取当前请求的投递目标地址。
37
+ * 便捷方法,等价于 getRequestContext()?.target。
38
+ */
39
+ export function getRequestTarget(): string | undefined {
40
+ return asyncLocalStorage.getStore()?.target;
41
+ }
42
+
43
+ /**
44
+ * 获取当前请求的账户 ID。
45
+ * 便捷方法,等价于 getRequestContext()?.accountId。
46
+ */
47
+ export function getRequestAccountId(): string | undefined {
48
+ return asyncLocalStorage.getStore()?.accountId;
49
+ }
package/src/runtime.ts ADDED
@@ -0,0 +1,14 @@
1
+ import type { PluginRuntime } from "openclaw/plugin-sdk";
2
+
3
+ let runtime: PluginRuntime | null = null;
4
+
5
+ export function setQQBotRuntime(next: PluginRuntime) {
6
+ runtime = next;
7
+ }
8
+
9
+ export function getQQBotRuntime(): PluginRuntime {
10
+ if (!runtime) {
11
+ throw new Error("QQBot runtime not initialized");
12
+ }
13
+ return runtime;
14
+ }
@@ -0,0 +1,303 @@
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
+ /** 创建此 session 时使用的 appId(用于检测凭据变更) */
25
+ appId?: string;
26
+ }
27
+
28
+ import { getQQBotDataDir } from "./utils/platform.js";
29
+
30
+ // Session 文件目录
31
+ const SESSION_DIR = getQQBotDataDir("sessions");
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
+ * @param expectedAppId 当前使用的 appId,如果与保存时的 appId 不匹配则视为失效
68
+ * @returns Session 状态,如果不存在、已过期或 appId 不匹配返回 null
69
+ */
70
+ export function loadSession(accountId: string, expectedAppId?: string): SessionState | null {
71
+ const filePath = getSessionPath(accountId);
72
+
73
+ try {
74
+ if (!fs.existsSync(filePath)) {
75
+ return null;
76
+ }
77
+
78
+ const data = fs.readFileSync(filePath, "utf-8");
79
+ const state = JSON.parse(data) as SessionState;
80
+
81
+ // 检查是否过期
82
+ const now = Date.now();
83
+ if (now - state.savedAt > SESSION_EXPIRE_TIME) {
84
+ console.log(`[session-store] Session expired for ${accountId}, age: ${Math.round((now - state.savedAt) / 1000)}s`);
85
+ try {
86
+ fs.unlinkSync(filePath);
87
+ } catch {
88
+ // 忽略删除错误
89
+ }
90
+ return null;
91
+ }
92
+
93
+ // 检查 appId 是否匹配(凭据变更检测)
94
+ if (expectedAppId && state.appId && state.appId !== expectedAppId) {
95
+ console.log(`[session-store] appId mismatch for ${accountId}: saved=${state.appId}, current=${expectedAppId}. Discarding stale session.`);
96
+ try {
97
+ fs.unlinkSync(filePath);
98
+ } catch {
99
+ // 忽略删除错误
100
+ }
101
+ return null;
102
+ }
103
+
104
+ // 验证必要字段
105
+ if (!state.sessionId || state.lastSeq === null || state.lastSeq === undefined) {
106
+ console.log(`[session-store] Invalid session data for ${accountId}`);
107
+ return null;
108
+ }
109
+
110
+ console.log(`[session-store] Loaded session for ${accountId}: sessionId=${state.sessionId}, lastSeq=${state.lastSeq}, appId=${state.appId ?? "unknown"}, age=${Math.round((now - state.savedAt) / 1000)}s`);
111
+ return state;
112
+ } catch (err) {
113
+ console.error(`[session-store] Failed to load session for ${accountId}: ${err}`);
114
+ return null;
115
+ }
116
+ }
117
+
118
+ /**
119
+ * 保存 Session 状态(带节流,避免频繁写入)
120
+ * @param state Session 状态
121
+ */
122
+ export function saveSession(state: SessionState): void {
123
+ const { accountId } = state;
124
+
125
+ // 获取或初始化节流状态
126
+ let throttle = throttleState.get(accountId);
127
+ if (!throttle) {
128
+ throttle = {
129
+ pendingState: null,
130
+ lastSaveTime: 0,
131
+ throttleTimer: null,
132
+ };
133
+ throttleState.set(accountId, throttle);
134
+ }
135
+
136
+ const now = Date.now();
137
+ const timeSinceLastSave = now - throttle.lastSaveTime;
138
+
139
+ // 如果距离上次保存时间足够长,立即保存
140
+ if (timeSinceLastSave >= SAVE_THROTTLE_MS) {
141
+ doSaveSession(state);
142
+ throttle.lastSaveTime = now;
143
+ throttle.pendingState = null;
144
+
145
+ // 清除待定的节流定时器
146
+ if (throttle.throttleTimer) {
147
+ clearTimeout(throttle.throttleTimer);
148
+ throttle.throttleTimer = null;
149
+ }
150
+ } else {
151
+ // 记录待保存的状态
152
+ throttle.pendingState = state;
153
+
154
+ // 如果没有设置定时器,设置一个
155
+ if (!throttle.throttleTimer) {
156
+ const delay = SAVE_THROTTLE_MS - timeSinceLastSave;
157
+ throttle.throttleTimer = setTimeout(() => {
158
+ const t = throttleState.get(accountId);
159
+ if (t && t.pendingState) {
160
+ doSaveSession(t.pendingState);
161
+ t.lastSaveTime = Date.now();
162
+ t.pendingState = null;
163
+ }
164
+ if (t) {
165
+ t.throttleTimer = null;
166
+ }
167
+ }, delay);
168
+ }
169
+ }
170
+ }
171
+
172
+ /**
173
+ * 实际执行保存操作
174
+ */
175
+ function doSaveSession(state: SessionState): void {
176
+ const filePath = getSessionPath(state.accountId);
177
+
178
+ try {
179
+ ensureDir();
180
+
181
+ // 更新保存时间
182
+ const stateToSave: SessionState = {
183
+ ...state,
184
+ savedAt: Date.now(),
185
+ };
186
+
187
+ fs.writeFileSync(filePath, JSON.stringify(stateToSave, null, 2), "utf-8");
188
+ console.log(`[session-store] Saved session for ${state.accountId}: sessionId=${state.sessionId}, lastSeq=${state.lastSeq}`);
189
+ } catch (err) {
190
+ console.error(`[session-store] Failed to save session for ${state.accountId}: ${err}`);
191
+ }
192
+ }
193
+
194
+ /**
195
+ * 清除 Session 状态
196
+ * @param accountId 账户 ID
197
+ */
198
+ export function clearSession(accountId: string): void {
199
+ const filePath = getSessionPath(accountId);
200
+
201
+ // 清除节流状态
202
+ const throttle = throttleState.get(accountId);
203
+ if (throttle) {
204
+ if (throttle.throttleTimer) {
205
+ clearTimeout(throttle.throttleTimer);
206
+ }
207
+ throttleState.delete(accountId);
208
+ }
209
+
210
+ try {
211
+ if (fs.existsSync(filePath)) {
212
+ fs.unlinkSync(filePath);
213
+ console.log(`[session-store] Cleared session for ${accountId}`);
214
+ }
215
+ } catch (err) {
216
+ console.error(`[session-store] Failed to clear session for ${accountId}: ${err}`);
217
+ }
218
+ }
219
+
220
+ /**
221
+ * 更新 lastSeq(轻量级更新)
222
+ * @param accountId 账户 ID
223
+ * @param lastSeq 最新的消息序号
224
+ */
225
+ export function updateLastSeq(accountId: string, lastSeq: number): void {
226
+ const existing = loadSession(accountId);
227
+ if (existing && existing.sessionId) {
228
+ saveSession({
229
+ ...existing,
230
+ lastSeq,
231
+ });
232
+ }
233
+ }
234
+
235
+ /**
236
+ * 获取所有保存的 Session 状态
237
+ */
238
+ export function getAllSessions(): SessionState[] {
239
+ const sessions: SessionState[] = [];
240
+
241
+ try {
242
+ ensureDir();
243
+ const files = fs.readdirSync(SESSION_DIR);
244
+
245
+ for (const file of files) {
246
+ if (file.startsWith("session-") && file.endsWith(".json")) {
247
+ const filePath = path.join(SESSION_DIR, file);
248
+ try {
249
+ const data = fs.readFileSync(filePath, "utf-8");
250
+ const state = JSON.parse(data) as SessionState;
251
+ sessions.push(state);
252
+ } catch {
253
+ // 忽略解析错误
254
+ }
255
+ }
256
+ }
257
+ } catch {
258
+ // 目录不存在等错误
259
+ }
260
+
261
+ return sessions;
262
+ }
263
+
264
+ /**
265
+ * 清理过期的 Session 文件
266
+ */
267
+ export function cleanupExpiredSessions(): number {
268
+ let cleaned = 0;
269
+
270
+ try {
271
+ ensureDir();
272
+ const files = fs.readdirSync(SESSION_DIR);
273
+ const now = Date.now();
274
+
275
+ for (const file of files) {
276
+ if (file.startsWith("session-") && file.endsWith(".json")) {
277
+ const filePath = path.join(SESSION_DIR, file);
278
+ try {
279
+ const data = fs.readFileSync(filePath, "utf-8");
280
+ const state = JSON.parse(data) as SessionState;
281
+
282
+ if (now - state.savedAt > SESSION_EXPIRE_TIME) {
283
+ fs.unlinkSync(filePath);
284
+ cleaned++;
285
+ console.log(`[session-store] Cleaned expired session: ${file}`);
286
+ }
287
+ } catch {
288
+ // 忽略解析错误,但也删除损坏的文件
289
+ try {
290
+ fs.unlinkSync(filePath);
291
+ cleaned++;
292
+ } catch {
293
+ // 忽略
294
+ }
295
+ }
296
+ }
297
+ }
298
+ } catch {
299
+ // 目录不存在等错误
300
+ }
301
+
302
+ return cleaned;
303
+ }