@ryantest/openclaw-qqbot 0.0.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.
Files changed (197) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +483 -0
  3. package/README.zh.md +478 -0
  4. package/bin/qqbot-cli.js +243 -0
  5. package/clawdbot.plugin.json +16 -0
  6. package/dist/index.d.ts +17 -0
  7. package/dist/index.js +26 -0
  8. package/dist/src/admin-resolver.d.ts +27 -0
  9. package/dist/src/admin-resolver.js +122 -0
  10. package/dist/src/api.d.ts +156 -0
  11. package/dist/src/api.js +599 -0
  12. package/dist/src/channel.d.ts +11 -0
  13. package/dist/src/channel.js +354 -0
  14. package/dist/src/config.d.ts +25 -0
  15. package/dist/src/config.js +161 -0
  16. package/dist/src/credential-backup.d.ts +31 -0
  17. package/dist/src/credential-backup.js +66 -0
  18. package/dist/src/gateway.d.ts +18 -0
  19. package/dist/src/gateway.js +1265 -0
  20. package/dist/src/image-server.d.ts +68 -0
  21. package/dist/src/image-server.js +462 -0
  22. package/dist/src/inbound-attachments.d.ts +58 -0
  23. package/dist/src/inbound-attachments.js +234 -0
  24. package/dist/src/known-users.d.ts +100 -0
  25. package/dist/src/known-users.js +263 -0
  26. package/dist/src/message-queue.d.ts +50 -0
  27. package/dist/src/message-queue.js +115 -0
  28. package/dist/src/onboarding.d.ts +10 -0
  29. package/dist/src/onboarding.js +203 -0
  30. package/dist/src/outbound-deliver.d.ts +48 -0
  31. package/dist/src/outbound-deliver.js +462 -0
  32. package/dist/src/outbound.d.ts +203 -0
  33. package/dist/src/outbound.js +1102 -0
  34. package/dist/src/proactive.d.ts +170 -0
  35. package/dist/src/proactive.js +399 -0
  36. package/dist/src/ref-index-store.d.ts +70 -0
  37. package/dist/src/ref-index-store.js +273 -0
  38. package/dist/src/reply-dispatcher.d.ts +35 -0
  39. package/dist/src/reply-dispatcher.js +311 -0
  40. package/dist/src/runtime.d.ts +3 -0
  41. package/dist/src/runtime.js +10 -0
  42. package/dist/src/session-store.d.ts +52 -0
  43. package/dist/src/session-store.js +254 -0
  44. package/dist/src/slash-commands.d.ts +71 -0
  45. package/dist/src/slash-commands.js +1179 -0
  46. package/dist/src/startup-greeting.d.ts +30 -0
  47. package/dist/src/startup-greeting.js +78 -0
  48. package/dist/src/stt.d.ts +21 -0
  49. package/dist/src/stt.js +70 -0
  50. package/dist/src/tools/channel.d.ts +16 -0
  51. package/dist/src/tools/channel.js +234 -0
  52. package/dist/src/tools/remind.d.ts +2 -0
  53. package/dist/src/tools/remind.js +247 -0
  54. package/dist/src/types.d.ts +175 -0
  55. package/dist/src/types.js +1 -0
  56. package/dist/src/typing-keepalive.d.ts +27 -0
  57. package/dist/src/typing-keepalive.js +64 -0
  58. package/dist/src/update-checker.d.ts +34 -0
  59. package/dist/src/update-checker.js +166 -0
  60. package/dist/src/user-messages.d.ts +8 -0
  61. package/dist/src/user-messages.js +8 -0
  62. package/dist/src/utils/audio-convert.d.ts +89 -0
  63. package/dist/src/utils/audio-convert.js +704 -0
  64. package/dist/src/utils/file-utils.d.ts +55 -0
  65. package/dist/src/utils/file-utils.js +150 -0
  66. package/dist/src/utils/image-size.d.ts +51 -0
  67. package/dist/src/utils/image-size.js +234 -0
  68. package/dist/src/utils/media-tags.d.ts +14 -0
  69. package/dist/src/utils/media-tags.js +164 -0
  70. package/dist/src/utils/payload.d.ts +112 -0
  71. package/dist/src/utils/payload.js +186 -0
  72. package/dist/src/utils/platform.d.ts +137 -0
  73. package/dist/src/utils/platform.js +390 -0
  74. package/dist/src/utils/text-parsing.d.ts +32 -0
  75. package/dist/src/utils/text-parsing.js +80 -0
  76. package/dist/src/utils/upload-cache.d.ts +34 -0
  77. package/dist/src/utils/upload-cache.js +93 -0
  78. package/index.ts +31 -0
  79. package/moltbot.plugin.json +16 -0
  80. package/node_modules/@eshaz/web-worker/LICENSE +201 -0
  81. package/node_modules/@eshaz/web-worker/README.md +134 -0
  82. package/node_modules/@eshaz/web-worker/browser.js +17 -0
  83. package/node_modules/@eshaz/web-worker/cjs/browser.js +16 -0
  84. package/node_modules/@eshaz/web-worker/cjs/node.js +219 -0
  85. package/node_modules/@eshaz/web-worker/index.d.ts +4 -0
  86. package/node_modules/@eshaz/web-worker/node.js +223 -0
  87. package/node_modules/@eshaz/web-worker/package.json +54 -0
  88. package/node_modules/@wasm-audio-decoders/common/index.js +5 -0
  89. package/node_modules/@wasm-audio-decoders/common/package.json +36 -0
  90. package/node_modules/@wasm-audio-decoders/common/src/WASMAudioDecoderCommon.js +231 -0
  91. package/node_modules/@wasm-audio-decoders/common/src/WASMAudioDecoderWorker.js +129 -0
  92. package/node_modules/@wasm-audio-decoders/common/src/puff/README +67 -0
  93. package/node_modules/@wasm-audio-decoders/common/src/puff/build_puff.js +31 -0
  94. package/node_modules/@wasm-audio-decoders/common/src/puff/puff.c +863 -0
  95. package/node_modules/@wasm-audio-decoders/common/src/puff/puff.h +35 -0
  96. package/node_modules/@wasm-audio-decoders/common/src/utilities.js +3 -0
  97. package/node_modules/@wasm-audio-decoders/common/types.d.ts +7 -0
  98. package/node_modules/mpg123-decoder/README.md +265 -0
  99. package/node_modules/mpg123-decoder/dist/mpg123-decoder.min.js +185 -0
  100. package/node_modules/mpg123-decoder/dist/mpg123-decoder.min.js.map +1 -0
  101. package/node_modules/mpg123-decoder/index.js +8 -0
  102. package/node_modules/mpg123-decoder/package.json +58 -0
  103. package/node_modules/mpg123-decoder/src/EmscriptenWasm.js +464 -0
  104. package/node_modules/mpg123-decoder/src/MPEGDecoder.js +200 -0
  105. package/node_modules/mpg123-decoder/src/MPEGDecoderWebWorker.js +21 -0
  106. package/node_modules/mpg123-decoder/types.d.ts +30 -0
  107. package/node_modules/silk-wasm/LICENSE +21 -0
  108. package/node_modules/silk-wasm/README.md +85 -0
  109. package/node_modules/silk-wasm/lib/index.cjs +16 -0
  110. package/node_modules/silk-wasm/lib/index.d.ts +70 -0
  111. package/node_modules/silk-wasm/lib/index.mjs +16 -0
  112. package/node_modules/silk-wasm/lib/silk.wasm +0 -0
  113. package/node_modules/silk-wasm/lib/utils.d.ts +4 -0
  114. package/node_modules/silk-wasm/package.json +39 -0
  115. package/node_modules/simple-yenc/.github/FUNDING.yml +1 -0
  116. package/node_modules/simple-yenc/.prettierignore +1 -0
  117. package/node_modules/simple-yenc/LICENSE +7 -0
  118. package/node_modules/simple-yenc/README.md +163 -0
  119. package/node_modules/simple-yenc/dist/esm.js +1 -0
  120. package/node_modules/simple-yenc/dist/index.js +1 -0
  121. package/node_modules/simple-yenc/package.json +50 -0
  122. package/node_modules/simple-yenc/rollup.config.js +27 -0
  123. package/node_modules/simple-yenc/src/simple-yenc.js +302 -0
  124. package/node_modules/ws/LICENSE +20 -0
  125. package/node_modules/ws/README.md +548 -0
  126. package/node_modules/ws/browser.js +8 -0
  127. package/node_modules/ws/index.js +13 -0
  128. package/node_modules/ws/lib/buffer-util.js +131 -0
  129. package/node_modules/ws/lib/constants.js +19 -0
  130. package/node_modules/ws/lib/event-target.js +292 -0
  131. package/node_modules/ws/lib/extension.js +203 -0
  132. package/node_modules/ws/lib/limiter.js +55 -0
  133. package/node_modules/ws/lib/permessage-deflate.js +528 -0
  134. package/node_modules/ws/lib/receiver.js +706 -0
  135. package/node_modules/ws/lib/sender.js +602 -0
  136. package/node_modules/ws/lib/stream.js +161 -0
  137. package/node_modules/ws/lib/subprotocol.js +62 -0
  138. package/node_modules/ws/lib/validation.js +152 -0
  139. package/node_modules/ws/lib/websocket-server.js +554 -0
  140. package/node_modules/ws/lib/websocket.js +1393 -0
  141. package/node_modules/ws/package.json +69 -0
  142. package/node_modules/ws/wrapper.mjs +8 -0
  143. package/openclaw.plugin.json +16 -0
  144. package/package.json +76 -0
  145. package/scripts/cleanup-legacy-plugins.sh +124 -0
  146. package/scripts/proactive-api-server.ts +369 -0
  147. package/scripts/send-proactive.ts +293 -0
  148. package/scripts/set-markdown.sh +156 -0
  149. package/scripts/test-sendmedia.ts +116 -0
  150. package/scripts/upgrade-via-alt-pkg.sh +307 -0
  151. package/scripts/upgrade-via-npm.ps1 +296 -0
  152. package/scripts/upgrade-via-npm.sh +301 -0
  153. package/scripts/upgrade-via-source.sh +774 -0
  154. package/skills/qqbot-channel/SKILL.md +263 -0
  155. package/skills/qqbot-channel/references/api_references.md +521 -0
  156. package/skills/qqbot-media/SKILL.md +56 -0
  157. package/skills/qqbot-remind/SKILL.md +149 -0
  158. package/src/admin-resolver.ts +140 -0
  159. package/src/api.ts +819 -0
  160. package/src/bot-logs-2026-03-21T11-21-47(2).txt +46 -0
  161. package/src/channel.ts +381 -0
  162. package/src/config.ts +187 -0
  163. package/src/credential-backup.ts +72 -0
  164. package/src/gateway.log +43 -0
  165. package/src/gateway.ts +1404 -0
  166. package/src/image-server.ts +539 -0
  167. package/src/inbound-attachments.ts +304 -0
  168. package/src/known-users.ts +353 -0
  169. package/src/message-queue.ts +169 -0
  170. package/src/onboarding.ts +274 -0
  171. package/src/openclaw-2026-03-21.log +3729 -0
  172. package/src/openclaw-plugin-sdk.d.ts +522 -0
  173. package/src/outbound-deliver.ts +552 -0
  174. package/src/outbound.ts +1266 -0
  175. package/src/proactive.ts +530 -0
  176. package/src/ref-index-store.ts +357 -0
  177. package/src/reply-dispatcher.ts +334 -0
  178. package/src/runtime.ts +14 -0
  179. package/src/session-store.ts +303 -0
  180. package/src/slash-commands.ts +1305 -0
  181. package/src/startup-greeting.ts +98 -0
  182. package/src/stt.ts +86 -0
  183. package/src/tools/channel.ts +281 -0
  184. package/src/tools/remind.ts +296 -0
  185. package/src/types.ts +183 -0
  186. package/src/typing-keepalive.ts +59 -0
  187. package/src/update-checker.ts +179 -0
  188. package/src/user-messages.ts +7 -0
  189. package/src/utils/audio-convert.ts +803 -0
  190. package/src/utils/file-utils.ts +167 -0
  191. package/src/utils/image-size.ts +266 -0
  192. package/src/utils/media-tags.ts +182 -0
  193. package/src/utils/payload.ts +265 -0
  194. package/src/utils/platform.ts +435 -0
  195. package/src/utils/text-parsing.ts +82 -0
  196. package/src/utils/upload-cache.ts +128 -0
  197. package/tsconfig.json +16 -0
@@ -0,0 +1,265 @@
1
+ /**
2
+ * QQBot 结构化消息载荷工具
3
+ *
4
+ * 用于处理 AI 输出的结构化消息载荷,包括:
5
+ * - 定时提醒载荷 (cron_reminder)
6
+ * - 媒体消息载荷 (media)
7
+ */
8
+
9
+ // ============================================
10
+ // 类型定义
11
+ // ============================================
12
+
13
+ /**
14
+ * 定时提醒载荷
15
+ */
16
+ export interface CronReminderPayload {
17
+ type: 'cron_reminder';
18
+ /** 提醒内容 */
19
+ content: string;
20
+ /** 目标类型:c2c (私聊) 或 group (群聊) */
21
+ targetType: 'c2c' | 'group';
22
+ /** 目标地址:user_openid 或 group_openid */
23
+ targetAddress: string;
24
+ /** 原始消息 ID(可选) */
25
+ originalMessageId?: string;
26
+ }
27
+
28
+ /**
29
+ * 媒体消息载荷
30
+ */
31
+ export interface MediaPayload {
32
+ type: 'media';
33
+ /** 媒体类型:image, audio, video, file */
34
+ mediaType: 'image' | 'audio' | 'video' | 'file';
35
+ /** 来源类型:url 或 file */
36
+ source: 'url' | 'file';
37
+ /** 媒体路径或 URL */
38
+ path: string;
39
+ /** 媒体描述(可选) */
40
+ caption?: string;
41
+ }
42
+
43
+ /**
44
+ * QQBot 载荷联合类型
45
+ */
46
+ export type QQBotPayload = CronReminderPayload | MediaPayload;
47
+
48
+ /**
49
+ * 解析结果
50
+ */
51
+ export interface ParseResult {
52
+ /** 是否为结构化载荷 */
53
+ isPayload: boolean;
54
+ /** 解析后的载荷对象(如果是结构化载荷) */
55
+ payload?: QQBotPayload;
56
+ /** 原始文本(如果不是结构化载荷) */
57
+ text?: string;
58
+ /** 解析错误信息(如果解析失败) */
59
+ error?: string;
60
+ }
61
+
62
+ // ============================================
63
+ // 常量定义
64
+ // ============================================
65
+
66
+ /** AI 输出的结构化载荷前缀 */
67
+ const PAYLOAD_PREFIX = 'QQBOT_PAYLOAD:';
68
+
69
+ /** Cron 消息存储的前缀 */
70
+ const CRON_PREFIX = 'QQBOT_CRON:';
71
+
72
+ // ============================================
73
+ // 解析函数
74
+ // ============================================
75
+
76
+ /**
77
+ * 解析 AI 输出的结构化载荷
78
+ *
79
+ * 检测消息是否以 QQBOT_PAYLOAD: 前缀开头,如果是则提取并解析 JSON
80
+ *
81
+ * @param text AI 输出的原始文本
82
+ * @returns 解析结果
83
+ *
84
+ * @example
85
+ * const result = parseQQBotPayload('QQBOT_PAYLOAD:\n{"type": "media", "mediaType": "image", ...}');
86
+ * if (result.isPayload && result.payload) {
87
+ * // 处理结构化载荷
88
+ * }
89
+ */
90
+ export function parseQQBotPayload(text: string): ParseResult {
91
+ const trimmedText = text.trim();
92
+
93
+ // 检查是否以 QQBOT_PAYLOAD: 开头
94
+ if (!trimmedText.startsWith(PAYLOAD_PREFIX)) {
95
+ return {
96
+ isPayload: false,
97
+ text: text
98
+ };
99
+ }
100
+
101
+ // 提取 JSON 内容(去掉前缀)
102
+ const jsonContent = trimmedText.slice(PAYLOAD_PREFIX.length).trim();
103
+
104
+ if (!jsonContent) {
105
+ return {
106
+ isPayload: true,
107
+ error: '载荷内容为空'
108
+ };
109
+ }
110
+
111
+ try {
112
+ const payload = JSON.parse(jsonContent) as QQBotPayload;
113
+
114
+ // 验证必要字段
115
+ if (!payload.type) {
116
+ return {
117
+ isPayload: true,
118
+ error: '载荷缺少 type 字段'
119
+ };
120
+ }
121
+
122
+ // 根据 type 进行额外验证
123
+ if (payload.type === 'cron_reminder') {
124
+ if (!payload.content || !payload.targetType || !payload.targetAddress) {
125
+ return {
126
+ isPayload: true,
127
+ error: 'cron_reminder 载荷缺少必要字段 (content, targetType, targetAddress)'
128
+ };
129
+ }
130
+ } else if (payload.type === 'media') {
131
+ if (!payload.mediaType || !payload.source || !payload.path) {
132
+ return {
133
+ isPayload: true,
134
+ error: 'media 载荷缺少必要字段 (mediaType, source, path)'
135
+ };
136
+ }
137
+ }
138
+
139
+ return {
140
+ isPayload: true,
141
+ payload
142
+ };
143
+ } catch (e) {
144
+ return {
145
+ isPayload: true,
146
+ error: `JSON 解析失败: ${e instanceof Error ? e.message : String(e)}`
147
+ };
148
+ }
149
+ }
150
+
151
+ // ============================================
152
+ // Cron 编码/解码函数
153
+ // ============================================
154
+
155
+ /**
156
+ * 将定时提醒载荷编码为 Cron 消息格式
157
+ *
158
+ * 将 JSON 编码为 Base64,并添加 QQBOT_CRON: 前缀
159
+ *
160
+ * @param payload 定时提醒载荷
161
+ * @returns 编码后的消息字符串,格式为 QQBOT_CRON:{base64}
162
+ *
163
+ * @example
164
+ * const message = encodePayloadForCron({
165
+ * type: 'cron_reminder',
166
+ * content: '喝水时间到!',
167
+ * targetType: 'c2c',
168
+ * targetAddress: 'user_openid_xxx'
169
+ * });
170
+ * // 返回: QQBOT_CRON:eyJ0eXBlIjoiY3Jvbl9yZW1pbmRlciIs...
171
+ */
172
+ export function encodePayloadForCron(payload: CronReminderPayload): string {
173
+ const jsonString = JSON.stringify(payload);
174
+ const base64 = Buffer.from(jsonString, 'utf-8').toString('base64');
175
+ return `${CRON_PREFIX}${base64}`;
176
+ }
177
+
178
+ /**
179
+ * 解码 Cron 消息中的载荷
180
+ *
181
+ * 检测 QQBOT_CRON: 前缀,解码 Base64 并解析 JSON
182
+ *
183
+ * @param message Cron 触发时收到的消息
184
+ * @returns 解码结果,包含是否为 Cron 载荷、解析后的载荷对象或错误信息
185
+ *
186
+ * @example
187
+ * const result = decodeCronPayload('QQBOT_CRON:eyJ0eXBlIjoiY3Jvbl9yZW1pbmRlciIs...');
188
+ * if (result.isCronPayload && result.payload) {
189
+ * // 处理定时提醒
190
+ * }
191
+ */
192
+ export function decodeCronPayload(message: string): {
193
+ isCronPayload: boolean;
194
+ payload?: CronReminderPayload;
195
+ error?: string;
196
+ } {
197
+ const trimmedMessage = message.trim();
198
+
199
+ // 检查是否以 QQBOT_CRON: 开头
200
+ if (!trimmedMessage.startsWith(CRON_PREFIX)) {
201
+ return {
202
+ isCronPayload: false
203
+ };
204
+ }
205
+
206
+ // 提取 Base64 内容
207
+ const base64Content = trimmedMessage.slice(CRON_PREFIX.length);
208
+
209
+ if (!base64Content) {
210
+ return {
211
+ isCronPayload: true,
212
+ error: 'Cron 载荷内容为空'
213
+ };
214
+ }
215
+
216
+ try {
217
+ // Base64 解码
218
+ const jsonString = Buffer.from(base64Content, 'base64').toString('utf-8');
219
+ const payload = JSON.parse(jsonString) as CronReminderPayload;
220
+
221
+ // 验证类型
222
+ if (payload.type !== 'cron_reminder') {
223
+ return {
224
+ isCronPayload: true,
225
+ error: `期望 type 为 cron_reminder,实际为 ${payload.type}`
226
+ };
227
+ }
228
+
229
+ // 验证必要字段
230
+ if (!payload.content || !payload.targetType || !payload.targetAddress) {
231
+ return {
232
+ isCronPayload: true,
233
+ error: 'Cron 载荷缺少必要字段'
234
+ };
235
+ }
236
+
237
+ return {
238
+ isCronPayload: true,
239
+ payload
240
+ };
241
+ } catch (e) {
242
+ return {
243
+ isCronPayload: true,
244
+ error: `Cron 载荷解码失败: ${e instanceof Error ? e.message : String(e)}`
245
+ };
246
+ }
247
+ }
248
+
249
+ // ============================================
250
+ // 辅助函数
251
+ // ============================================
252
+
253
+ /**
254
+ * 判断载荷是否为定时提醒类型
255
+ */
256
+ export function isCronReminderPayload(payload: QQBotPayload): payload is CronReminderPayload {
257
+ return payload.type === 'cron_reminder';
258
+ }
259
+
260
+ /**
261
+ * 判断载荷是否为媒体消息类型
262
+ */
263
+ export function isMediaPayload(payload: QQBotPayload): payload is MediaPayload {
264
+ return payload.type === 'media';
265
+ }
@@ -0,0 +1,435 @@
1
+ /**
2
+ * 跨平台兼容工具
3
+ *
4
+ * 统一 Mac / Linux / Windows 三大系统的:
5
+ * - 用户主目录获取
6
+ * - 临时目录获取
7
+ * - 本地路径判断
8
+ * - ffmpeg / ffprobe 可执行文件路径
9
+ * - silk-wasm 原生模块兼容性检测
10
+ * - 启动诊断报告
11
+ */
12
+
13
+ import * as os from "node:os";
14
+ import * as path from "node:path";
15
+ import * as fs from "node:fs";
16
+ import { execFile } from "node:child_process";
17
+
18
+ // ============ 基础平台信息 ============
19
+
20
+ export type PlatformType = "darwin" | "linux" | "win32" | "other";
21
+
22
+ export function getPlatform(): PlatformType {
23
+ const p = process.platform;
24
+ if (p === "darwin" || p === "linux" || p === "win32") return p;
25
+ return "other";
26
+ }
27
+
28
+ export function isWindows(): boolean {
29
+ return process.platform === "win32";
30
+ }
31
+
32
+ // ============ 用户主目录 ============
33
+
34
+ /**
35
+ * 安全获取用户主目录
36
+ *
37
+ * 优先级:
38
+ * 1. os.homedir()(Node 原生,所有平台)
39
+ * 2. $HOME(Mac/Linux)或 %USERPROFILE%(Windows)
40
+ * 3. 降级到 /tmp(Linux/Mac)或 os.tmpdir()(Windows)
41
+ *
42
+ * 与之前 `process.env.HOME || "/home/ubuntu"` 的硬编码相比,
43
+ * 现在能正确处理 Windows 和非 ubuntu 用户。
44
+ */
45
+ export function getHomeDir(): string {
46
+ try {
47
+ const home = os.homedir();
48
+ if (home && fs.existsSync(home)) return home;
49
+ } catch {}
50
+
51
+ // fallback 环境变量
52
+ const envHome = process.env.HOME || process.env.USERPROFILE;
53
+ if (envHome && fs.existsSync(envHome)) return envHome;
54
+
55
+ // 最后降级
56
+ return os.tmpdir();
57
+ }
58
+
59
+ /**
60
+ * 获取 .openclaw/qqbot 下的子目录路径,并自动创建
61
+ * 替代各文件中分散的 path.join(HOME, ".openclaw", "qqbot", ...)
62
+ */
63
+ export function getQQBotDataDir(...subPaths: string[]): string {
64
+ const dir = path.join(getHomeDir(), ".openclaw", "qqbot", ...subPaths);
65
+ if (!fs.existsSync(dir)) {
66
+ fs.mkdirSync(dir, { recursive: true });
67
+ }
68
+ return dir;
69
+ }
70
+
71
+ /**
72
+ * 获取 .openclaw/media/qqbot 下的子目录路径,并自动创建
73
+ *
74
+ * 与 getQQBotDataDir 不同,此目录位于 OpenClaw 核心的媒体安全白名单
75
+ * (~/.openclaw/media) 之下,下载到这里的文件可以被框架的 image/media
76
+ * 工具直接访问,不会触发 "Local media path is not under an allowed directory" 错误。
77
+ *
78
+ * 用于存放从 QQ 下载的图片、语音等需要被框架处理的媒体文件。
79
+ */
80
+ export function getQQBotMediaDir(...subPaths: string[]): string {
81
+ const dir = path.join(getHomeDir(), ".openclaw", "media", "qqbot", ...subPaths);
82
+ if (!fs.existsSync(dir)) {
83
+ fs.mkdirSync(dir, { recursive: true });
84
+ }
85
+ return dir;
86
+ }
87
+
88
+ // ============ 临时目录 ============
89
+
90
+ /**
91
+ * 获取系统临时目录(跨平台安全)
92
+ * Mac: /var/folders/... 或 /tmp
93
+ * Linux: /tmp
94
+ * Windows: %TEMP% 或 C:\Users\xxx\AppData\Local\Temp
95
+ */
96
+ export function getTempDir(): string {
97
+ return os.tmpdir();
98
+ }
99
+
100
+ // ============ 波浪线路径展开 ============
101
+
102
+ /**
103
+ * 展开路径中的波浪线(~)为用户主目录
104
+ *
105
+ * Mac/Linux 用户经常使用 `~/Desktop/file.png` 这样的路径,
106
+ * 但 Node.js 的 fs 模块不会像 shell 一样自动展开 `~`。
107
+ *
108
+ * 支持:
109
+ * - `~/xxx` → `/Users/you/xxx`(Mac)或 `/home/you/xxx`(Linux)
110
+ * - `~` → `/Users/you`
111
+ * - 非 `~` 开头的路径原样返回
112
+ *
113
+ * 注意: 不支持 `~otheruser/xxx` 语法(极少使用,且需要系统调用获取其他用户信息)
114
+ */
115
+ export function expandTilde(p: string): string {
116
+ if (!p) return p;
117
+ if (p === "~") return getHomeDir();
118
+ if (p.startsWith("~/") || p.startsWith("~\\")) {
119
+ return path.join(getHomeDir(), p.slice(2));
120
+ }
121
+ return p;
122
+ }
123
+
124
+ /**
125
+ * 对路径进行完整的规范化处理:剥离 file:// 前缀 + 展开波浪线 + 去除首尾空白
126
+ * 所有文件操作前应通过此函数处理用户输入的路径
127
+ */
128
+ export function normalizePath(p: string): string {
129
+ let result = p.trim();
130
+ // 剥离 file:// 协议前缀: file:///Users/... → /Users/...
131
+ if (result.startsWith("file://")) {
132
+ result = result.slice("file://".length);
133
+ // 处理 URL 编码(file:// 路径中空格等字符可能被编码)
134
+ try {
135
+ result = decodeURIComponent(result);
136
+ } catch {
137
+ // decodeURIComponent 失败时保留原样
138
+ }
139
+ }
140
+ return expandTilde(result);
141
+ }
142
+
143
+ // ============ 文件名 UTF-8 规范化 ============
144
+
145
+ /**
146
+ * 规范化文件名为 QQ Bot API 要求的 UTF-8 编码格式
147
+ *
148
+ * 问题场景:
149
+ * - macOS HFS+/APFS 文件系统使用 NFD(Unicode 分解形式)存储文件名,
150
+ * 例如「中文.txt」被分解为多个码点,QQ Bot API 可能拒绝
151
+ * - 文件名可能包含 API 不接受的特殊控制字符
152
+ * - URL 路径中可能包含 percent-encoded 的文件名需要解码
153
+ *
154
+ * 处理:
155
+ * 1. Unicode NFC 规范化(将 NFD 分解形式合并为 NFC 组合形式)
156
+ * 2. 去除 ASCII 控制字符(0x00-0x1F, 0x7F)
157
+ * 3. 去除首尾空白
158
+ * 4. 对 percent-encoded 的文件名尝试 URI 解码
159
+ */
160
+ export function sanitizeFileName(name: string): string {
161
+ if (!name) return name;
162
+
163
+ let result = name.trim();
164
+
165
+ // 尝试 URI 解码(处理 URL 中 percent-encoded 的中文文件名)
166
+ // 例如 %E4%B8%AD%E6%96%87.txt → 中文.txt
167
+ if (result.includes("%")) {
168
+ try {
169
+ result = decodeURIComponent(result);
170
+ } catch {
171
+ // 解码失败(非合法 percent-encoding),保留原始值
172
+ }
173
+ }
174
+
175
+ // Unicode NFC 规范化:将 macOS NFD 分解形式合并为标准 NFC 组合形式
176
+ result = result.normalize("NFC");
177
+
178
+ // 去除 ASCII 控制字符(保留所有可打印字符和非 ASCII Unicode 字符)
179
+ result = result.replace(/[\x00-\x1F\x7F]/g, "");
180
+
181
+ return result;
182
+ }
183
+
184
+ // ============ 本地路径判断 ============
185
+
186
+ /**
187
+ * 判断字符串是否为本地文件路径(非 URL)
188
+ *
189
+ * 覆盖:
190
+ * - Unix 绝对路径: /Users/..., /home/..., /tmp/...
191
+ * - Windows 绝对路径: C:\..., D:/..., \\server\share
192
+ * - 相对路径: ./file, ../file
193
+ * - 波浪线路径: ~/Desktop/file.png
194
+ * - file:// 协议: file:///Users/..., file:///home/...
195
+ *
196
+ * 不匹配:
197
+ * - http:// / https:// URL
198
+ * - data: URL
199
+ */
200
+ export function isLocalPath(p: string): boolean {
201
+ if (!p) return false;
202
+ // file:// 协议(本地文件 URI)
203
+ if (p.startsWith("file://")) return true;
204
+ // 波浪线路径(Mac/Linux 用户常用)
205
+ if (p === "~" || p.startsWith("~/") || p.startsWith("~\\")) return true;
206
+ // Unix 绝对路径
207
+ if (p.startsWith("/")) return true;
208
+ // Windows 盘符: C:\ 或 C:/
209
+ if (/^[a-zA-Z]:[\\/]/.test(p)) return true;
210
+ // Windows UNC: \\server\share
211
+ if (p.startsWith("\\\\")) return true;
212
+ // 相对路径
213
+ if (p.startsWith("./") || p.startsWith("../")) return true;
214
+ // Windows 相对路径
215
+ if (p.startsWith(".\\") || p.startsWith("..\\")) return true;
216
+ return false;
217
+ }
218
+
219
+ /**
220
+ * 判断 markdown 中提取的路径是否像本地路径
221
+ * 比 isLocalPath 更宽松,用于从 markdown ![](path) 中检测误用
222
+ */
223
+ export function looksLikeLocalPath(p: string): boolean {
224
+ if (isLocalPath(p)) return true;
225
+ // 常见系统目录前缀(不以 / 开头时也匹配)
226
+ return /^(?:Users|home|tmp|var|private|[A-Z]:)/i.test(p);
227
+ }
228
+
229
+ // ============ ffmpeg 跨平台检测 ============
230
+
231
+ let _ffmpegPath: string | null | undefined; // undefined = 未检测, null = 不可用
232
+ let _ffmpegCheckPromise: Promise<string | null> | null = null;
233
+
234
+ /**
235
+ * 检测 ffmpeg 是否可用,返回可执行路径
236
+ *
237
+ * Windows 上检测 ffmpeg.exe,Mac/Linux 检测 ffmpeg
238
+ * 支持通过环境变量 FFMPEG_PATH 指定自定义路径
239
+ *
240
+ * @returns ffmpeg 可执行文件路径,不可用返回 null
241
+ */
242
+ export function detectFfmpeg(): Promise<string | null> {
243
+ if (_ffmpegPath !== undefined) return Promise.resolve(_ffmpegPath);
244
+ if (_ffmpegCheckPromise) return _ffmpegCheckPromise;
245
+
246
+ _ffmpegCheckPromise = (async () => {
247
+ // 1. 环境变量自定义路径
248
+ const envPath = process.env.FFMPEG_PATH;
249
+ if (envPath) {
250
+ const ok = await testExecutable(envPath, ["-version"]);
251
+ if (ok) {
252
+ _ffmpegPath = envPath;
253
+ console.log(`[platform] ffmpeg found via FFMPEG_PATH: ${envPath}`);
254
+ return _ffmpegPath;
255
+ }
256
+ console.warn(`[platform] FFMPEG_PATH set but not working: ${envPath}`);
257
+ }
258
+
259
+ // 2. 系统 PATH 中检测
260
+ const cmd = isWindows() ? "ffmpeg.exe" : "ffmpeg";
261
+ const ok = await testExecutable(cmd, ["-version"]);
262
+ if (ok) {
263
+ _ffmpegPath = cmd;
264
+ console.log(`[platform] ffmpeg detected in PATH`);
265
+ return _ffmpegPath;
266
+ }
267
+
268
+ // 3. 常见安装位置(Mac brew、Windows choco/scoop)
269
+ const commonPaths = isWindows()
270
+ ? [
271
+ "C:\\ffmpeg\\bin\\ffmpeg.exe",
272
+ path.join(process.env.LOCALAPPDATA || "", "Programs", "ffmpeg", "bin", "ffmpeg.exe"),
273
+ path.join(process.env.ProgramFiles || "", "ffmpeg", "bin", "ffmpeg.exe"),
274
+ ]
275
+ : [
276
+ "/usr/local/bin/ffmpeg", // Mac brew
277
+ "/opt/homebrew/bin/ffmpeg", // Mac ARM brew
278
+ "/usr/bin/ffmpeg", // Linux apt
279
+ "/snap/bin/ffmpeg", // Linux snap
280
+ ];
281
+
282
+ for (const p of commonPaths) {
283
+ if (p && fs.existsSync(p)) {
284
+ const works = await testExecutable(p, ["-version"]);
285
+ if (works) {
286
+ _ffmpegPath = p;
287
+ console.log(`[platform] ffmpeg found at: ${p}`);
288
+ return _ffmpegPath;
289
+ }
290
+ }
291
+ }
292
+
293
+ _ffmpegPath = null;
294
+ return null;
295
+ })().finally(() => {
296
+ _ffmpegCheckPromise = null;
297
+ });
298
+
299
+ return _ffmpegCheckPromise;
300
+ }
301
+
302
+ /** 测试可执行文件是否能正常运行 */
303
+ function testExecutable(cmd: string, args: string[]): Promise<boolean> {
304
+ return new Promise((resolve) => {
305
+ execFile(cmd, args, { timeout: 5000 }, (err) => {
306
+ resolve(!err);
307
+ });
308
+ });
309
+ }
310
+
311
+ /** 重置 ffmpeg 缓存(用于测试) */
312
+ export function resetFfmpegCache(): void {
313
+ _ffmpegPath = undefined;
314
+ _ffmpegCheckPromise = null;
315
+ }
316
+
317
+ // ============ silk-wasm 兼容性 ============
318
+
319
+ let _silkWasmAvailable: boolean | null = null;
320
+
321
+ /**
322
+ * 检测 silk-wasm 是否可用
323
+ *
324
+ * silk-wasm 依赖 WASM 运行时,在某些环境(如老版本 Node、某些容器)可能不可用。
325
+ * 提前检测避免运行时崩溃。
326
+ */
327
+ export async function checkSilkWasmAvailable(): Promise<boolean> {
328
+ if (_silkWasmAvailable !== null) return _silkWasmAvailable;
329
+
330
+ try {
331
+ const { isSilk } = await import("silk-wasm");
332
+ // 用一个空 buffer 快速测试 WASM 是否能加载
333
+ isSilk(new Uint8Array(0));
334
+ _silkWasmAvailable = true;
335
+ console.log("[platform] silk-wasm: available");
336
+ } catch (err) {
337
+ _silkWasmAvailable = false;
338
+ console.warn(`[platform] silk-wasm: NOT available (${err instanceof Error ? err.message : String(err)})`);
339
+ }
340
+ return _silkWasmAvailable;
341
+ }
342
+
343
+ // ============ 启动环境诊断 ============
344
+
345
+ export interface DiagnosticReport {
346
+ platform: string;
347
+ arch: string;
348
+ nodeVersion: string;
349
+ homeDir: string;
350
+ tempDir: string;
351
+ dataDir: string;
352
+ ffmpeg: string | null;
353
+ silkWasm: boolean;
354
+ warnings: string[];
355
+ }
356
+
357
+ /**
358
+ * 运行启动诊断,返回环境报告
359
+ * 在 gateway 启动时调用,打印环境信息并给出警告
360
+ */
361
+ export async function runDiagnostics(): Promise<DiagnosticReport> {
362
+ const warnings: string[] = [];
363
+
364
+ const platform = `${process.platform} (${os.release()})`;
365
+ const arch = process.arch;
366
+ const nodeVersion = process.version;
367
+ const homeDir = getHomeDir();
368
+ const tempDir = getTempDir();
369
+ const dataDir = getQQBotDataDir();
370
+
371
+ // 检测 ffmpeg
372
+ const ffmpegPath = await detectFfmpeg();
373
+ if (!ffmpegPath) {
374
+ warnings.push(
375
+ isWindows()
376
+ ? "⚠️ ffmpeg 未安装。语音/视频格式转换将受限。安装方式: choco install ffmpeg 或 scoop install ffmpeg 或从 https://ffmpeg.org 下载"
377
+ : getPlatform() === "darwin"
378
+ ? "⚠️ ffmpeg 未安装。语音/视频格式转换将受限。安装方式: brew install ffmpeg"
379
+ : "⚠️ ffmpeg 未安装。语音/视频格式转换将受限。安装方式: sudo apt install ffmpeg 或 sudo yum install ffmpeg"
380
+ );
381
+ }
382
+
383
+ // 检测 silk-wasm
384
+ const silkWasm = await checkSilkWasmAvailable();
385
+ if (!silkWasm) {
386
+ warnings.push("⚠️ silk-wasm 不可用。QQ 语音消息的收发将无法工作。请确认 Node.js 版本 >= 16 且 WASM 支持正常");
387
+ }
388
+
389
+ // 检查数据目录可写性
390
+ try {
391
+ const testFile = path.join(dataDir, ".write-test");
392
+ fs.writeFileSync(testFile, "test");
393
+ fs.unlinkSync(testFile);
394
+ } catch {
395
+ warnings.push(`⚠️ 数据目录不可写: ${dataDir}。请检查权限`);
396
+ }
397
+
398
+ // Windows 特殊提醒
399
+ if (isWindows()) {
400
+ // 检查路径中是否有中文或空格(可能导致某些工具异常)
401
+ if (/[\u4e00-\u9fa5]/.test(homeDir) || homeDir.includes(" ")) {
402
+ warnings.push(`⚠️ 用户目录包含中文或空格: ${homeDir}。某些工具可能无法正常工作,建议设置 QQBOT_DATA_DIR 环境变量指定纯英文路径`);
403
+ }
404
+ }
405
+
406
+ const report: DiagnosticReport = {
407
+ platform,
408
+ arch,
409
+ nodeVersion,
410
+ homeDir,
411
+ tempDir,
412
+ dataDir,
413
+ ffmpeg: ffmpegPath,
414
+ silkWasm,
415
+ warnings,
416
+ };
417
+
418
+ // 打印诊断报告
419
+ console.log("=== QQBot 环境诊断 ===");
420
+ console.log(` 平台: ${platform} (${arch})`);
421
+ console.log(` Node: ${nodeVersion}`);
422
+ console.log(` 主目录: ${homeDir}`);
423
+ console.log(` 数据目录: ${dataDir}`);
424
+ console.log(` ffmpeg: ${ffmpegPath ?? "未安装"}`);
425
+ console.log(` silk-wasm: ${silkWasm ? "可用" : "不可用"}`);
426
+ if (warnings.length > 0) {
427
+ console.log(" --- 警告 ---");
428
+ for (const w of warnings) {
429
+ console.log(` ${w}`);
430
+ }
431
+ }
432
+ console.log("======================");
433
+
434
+ return report;
435
+ }