@cxyhhhhh/openclaw-qqbot 1.6.7-alpha.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 (218) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +470 -0
  3. package/README.zh.md +465 -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 +264 -0
  10. package/dist/src/api.js +777 -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 +2021 -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 +86 -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 +926 -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 +70 -0
  42. package/dist/src/ref-index-store.js +250 -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 +18 -0
  46. package/dist/src/request-context.js +30 -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 +1461 -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 +250 -0
  56. package/dist/src/streaming.js +914 -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 +248 -0
  63. package/dist/src/types.d.ts +364 -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 +34 -0
  68. package/dist/src/update-checker.js +160 -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 +59 -0
  72. package/dist/src/utils/chunked-upload.js +289 -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 +148 -0
  78. package/dist/src/utils/media-send.js +456 -0
  79. package/dist/src/utils/media-tags.d.ts +14 -0
  80. package/dist/src/utils/media-tags.js +164 -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 +51 -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 +32 -0
  90. package/dist/src/utils/text-parsing.js +69 -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 +13 -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 +69 -0
  156. package/node_modules/ws/wrapper.mjs +8 -0
  157. package/openclaw.plugin.json +17 -0
  158. package/package.json +67 -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 +113 -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 +451 -0
  168. package/scripts/upgrade-via-npm.sh +528 -0
  169. package/scripts/upgrade-via-source.sh +916 -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 +149 -0
  174. package/src/admin-resolver.ts +181 -0
  175. package/src/api.ts +1138 -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 +2257 -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 +349 -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 +1119 -0
  191. package/src/proactive.ts +530 -0
  192. package/src/ref-index-store.ts +335 -0
  193. package/src/reply-dispatcher.ts +334 -0
  194. package/src/request-context.ts +39 -0
  195. package/src/runtime.ts +14 -0
  196. package/src/session-store.ts +303 -0
  197. package/src/slash-commands.ts +1615 -0
  198. package/src/startup-greeting.ts +120 -0
  199. package/src/streaming.ts +1102 -0
  200. package/src/stt.ts +86 -0
  201. package/src/tools/channel.ts +281 -0
  202. package/src/tools/remind.ts +300 -0
  203. package/src/types.ts +386 -0
  204. package/src/typing-keepalive.ts +59 -0
  205. package/src/update-checker.ts +174 -0
  206. package/src/utils/audio-convert.ts +859 -0
  207. package/src/utils/chunked-upload.ts +419 -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 +585 -0
  211. package/src/utils/media-tags.ts +182 -0
  212. package/src/utils/payload.ts +265 -0
  213. package/src/utils/pkg-version.ts +54 -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 +75 -0
  217. package/src/utils/upload-cache.ts +128 -0
  218. package/tsconfig.json +16 -0
@@ -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
+ }
@@ -0,0 +1,102 @@
1
+ /**
2
+ * 远程 URL 安全校验
3
+ *
4
+ * 下载外部资源前,确保目标地址不会命中内部网络或云元数据端点,
5
+ * 避免模型输出的恶意链接触达内网服务。
6
+ */
7
+
8
+ import net from "node:net";
9
+ import dns from "node:dns/promises";
10
+
11
+ /* ---------- 内网 / 保留地址判定 ---------- */
12
+
13
+ /** IPv4 保留网段前缀(覆盖 RFC 1918、链路本地、回环等) */
14
+ const RESERVED_V4_PREFIXES = [
15
+ "127.", // loopback
16
+ "10.", // class-A private
17
+ "192.168.", // class-C private
18
+ "169.254.", // link-local / cloud metadata
19
+ ] as const;
20
+
21
+ /** 172.16.0.0 – 172.31.255.255 需要单独用正则匹配 */
22
+ const PRIVATE_172_RE = /^172\.(1[6-9]|2\d|3[01])\./;
23
+
24
+ /**
25
+ * 检查给定 IP 是否落在不可路由 / 私有网段内。
26
+ *
27
+ * 覆盖:
28
+ * - IPv4: 127/8, 10/8, 172.16/12, 192.168/16, 169.254/16, 0.0.0.0
29
+ * - IPv6: ::1, ::, fe80 (link-local), fc/fd (ULA)
30
+ */
31
+ export function isReservedAddr(ip: string): boolean {
32
+ // --- IPv4 ---
33
+ if (ip === "0.0.0.0") return true;
34
+ for (const pfx of RESERVED_V4_PREFIXES) {
35
+ if (ip.startsWith(pfx)) return true;
36
+ }
37
+ if (PRIVATE_172_RE.test(ip)) return true;
38
+
39
+ // --- IPv6 ---
40
+ const lower = ip.toLowerCase();
41
+ if (lower === "::1" || lower === "::") return true;
42
+ if (lower.startsWith("fe80:")) return true; // link-local
43
+ if (lower.startsWith("fc") || lower.startsWith("fd")) return true; // unique local
44
+ return false;
45
+ }
46
+
47
+ /* ---------- URL 合法性校验 ---------- */
48
+
49
+ const ALLOWED_SCHEMES = new Set(["http:", "https:"]);
50
+
51
+ /**
52
+ * 校验远程 URL 是否可安全请求。
53
+ *
54
+ * 规则:
55
+ * 1. 仅放行 http / https 协议
56
+ * 2. 若 URL 直接携带 IP 则即时判定
57
+ * 3. 若为域名则先做 DNS 解析,逐条检查解析结果
58
+ *
59
+ * @throws {Error} 当 URL 指向受限地址时
60
+ */
61
+ export async function validateRemoteUrl(raw: string): Promise<void> {
62
+ const url = new URL(raw);
63
+
64
+ if (!ALLOWED_SCHEMES.has(url.protocol)) {
65
+ throw new Error(
66
+ `不支持的协议 "${url.protocol}",仅允许 http/https(URL: ${raw})`,
67
+ );
68
+ }
69
+
70
+ // 去掉 IPv6 方括号
71
+ const host = url.hostname.replace(/^\[|\]$/g, "");
72
+
73
+ if (net.isIP(host)) {
74
+ assertPublicAddr(host, raw);
75
+ return;
76
+ }
77
+
78
+ // 域名 → 解析后逐条检查
79
+ try {
80
+ const ips = await dns.resolve(host);
81
+ for (const ip of ips) {
82
+ assertPublicAddr(ip, raw, host);
83
+ }
84
+ } catch (err) {
85
+ // 已经是我们自己抛的安全错误,继续向上传播
86
+ if (err instanceof Error && err.message.includes("内网")) throw err;
87
+ // DNS 查询失败不阻塞,后续 fetch 会产生网络错误
88
+ console.warn(`[url-check] DNS 解析 "${host}" 失败: ${err}`);
89
+ }
90
+ }
91
+
92
+ /* ---------- 内部辅助 ---------- */
93
+
94
+ /** 断言 IP 为公网地址,否则抛出错误 */
95
+ function assertPublicAddr(ip: string, originalUrl: string, domain?: string): void {
96
+ if (!isReservedAddr(ip)) return;
97
+
98
+ const target = domain ? `域名 "${domain}" 解析到内网地址 "${ip}"` : `内网地址 "${ip}"`;
99
+ throw new Error(
100
+ `禁止访问${target},已拦截潜在的 SSRF 请求(URL: ${originalUrl})`,
101
+ );
102
+ }
@@ -0,0 +1,75 @@
1
+ /**
2
+ * QQ Bot 文本解析工具函数
3
+ */
4
+
5
+ import type { RefAttachmentSummary } from "../ref-index-store.js";
6
+ import { inferAttachmentType } from "../group-history.js";
7
+
8
+ /**
9
+ * 解析 QQ 表情标签,将 <faceType=1,faceId="13",ext="base64..."> 格式
10
+ * 替换为 【表情: 中文名】 格式
11
+ * ext 字段为 Base64 编码的 JSON,格式如 {"text":"呲牙"}
12
+ */
13
+ export function parseFaceTags(text: string): string {
14
+ if (!text) return text;
15
+
16
+ return text.replace(/<faceType=\d+,faceId="[^"]*",ext="([^"]*)">/g, (_match, ext: string) => {
17
+ try {
18
+ const decoded = Buffer.from(ext, "base64").toString("utf-8");
19
+ const parsed = JSON.parse(decoded);
20
+ const faceName = parsed.text || "未知表情";
21
+ return `【表情: ${faceName}】`;
22
+ } catch {
23
+ return _match;
24
+ }
25
+ });
26
+ }
27
+
28
+ /**
29
+ * 过滤内部标记(如 [[reply_to: xxx]])
30
+ * 这些标记可能被 AI 错误地学习并输出,需要在发送前移除
31
+ */
32
+ export function filterInternalMarkers(text: string): string {
33
+ if (!text) return text;
34
+
35
+ let result = text.replace(/\[\[[a-z_]+:\s*[^\]]*\]\]/gi, "");
36
+ // 过滤框架内部图片引用标记:@image:image_xxx.png、@voice:voice_xxx.silk 等
37
+ result = result.replace(/@(?:image|voice|video|file):[a-zA-Z0-9_.-]+/g, "");
38
+ result = result.replace(/\n{3,}/g, "\n\n").trim();
39
+
40
+ return result;
41
+ }
42
+
43
+ /**
44
+ * 从 message_scene.ext 数组中解析引用索引
45
+ * ext 格式示例: ["", "ref_msg_idx=REFIDX_xxx", "msg_idx=REFIDX_yyy"]
46
+ */
47
+ export function parseRefIndices(ext?: string[]): { refMsgIdx?: string; msgIdx?: string } {
48
+ if (!ext || ext.length === 0) return {};
49
+ let refMsgIdx: string | undefined;
50
+ let msgIdx: string | undefined;
51
+ for (const item of ext) {
52
+ if (item.startsWith("ref_msg_idx=")) {
53
+ refMsgIdx = item.slice("ref_msg_idx=".length);
54
+ } else if (item.startsWith("msg_idx=")) {
55
+ msgIdx = item.slice("msg_idx=".length);
56
+ }
57
+ }
58
+ return { refMsgIdx, msgIdx };
59
+ }
60
+
61
+ /**
62
+ * 从附件列表中构建附件摘要(用于引用索引缓存)
63
+ */
64
+ export function buildAttachmentSummaries(
65
+ attachments?: Array<{ content_type: string; url: string; filename?: string; voice_wav_url?: string }>,
66
+ localPaths?: Array<string | null>,
67
+ ): RefAttachmentSummary[] | undefined {
68
+ if (!attachments || attachments.length === 0) return undefined;
69
+ return attachments.map((att, idx) => ({
70
+ type: inferAttachmentType(att.content_type),
71
+ filename: att.filename,
72
+ contentType: att.content_type,
73
+ localPath: localPaths?.[idx] ?? undefined,
74
+ }));
75
+ }