@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,390 @@
1
+ /**
2
+ * 跨平台兼容工具
3
+ *
4
+ * 统一 Mac / Linux / Windows 三大系统的:
5
+ * - 用户主目录获取
6
+ * - 临时目录获取
7
+ * - 本地路径判断
8
+ * - ffmpeg / ffprobe 可执行文件路径
9
+ * - silk-wasm 原生模块兼容性检测
10
+ * - 启动诊断报告
11
+ */
12
+ import * as os from "node:os";
13
+ import * as path from "node:path";
14
+ import * as fs from "node:fs";
15
+ import { execFile } from "node:child_process";
16
+ export function getPlatform() {
17
+ const p = process.platform;
18
+ if (p === "darwin" || p === "linux" || p === "win32")
19
+ return p;
20
+ return "other";
21
+ }
22
+ export function isWindows() {
23
+ return process.platform === "win32";
24
+ }
25
+ // ============ 用户主目录 ============
26
+ /**
27
+ * 安全获取用户主目录
28
+ *
29
+ * 优先级:
30
+ * 1. os.homedir()(Node 原生,所有平台)
31
+ * 2. $HOME(Mac/Linux)或 %USERPROFILE%(Windows)
32
+ * 3. 降级到 /tmp(Linux/Mac)或 os.tmpdir()(Windows)
33
+ *
34
+ * 与之前 `process.env.HOME || "/home/ubuntu"` 的硬编码相比,
35
+ * 现在能正确处理 Windows 和非 ubuntu 用户。
36
+ */
37
+ export function getHomeDir() {
38
+ try {
39
+ const home = os.homedir();
40
+ if (home && fs.existsSync(home))
41
+ return home;
42
+ }
43
+ catch { }
44
+ // fallback 环境变量
45
+ const envHome = process.env.HOME || process.env.USERPROFILE;
46
+ if (envHome && fs.existsSync(envHome))
47
+ return envHome;
48
+ // 最后降级
49
+ return os.tmpdir();
50
+ }
51
+ /**
52
+ * 获取 .openclaw/qqbot 下的子目录路径,并自动创建
53
+ * 替代各文件中分散的 path.join(HOME, ".openclaw", "qqbot", ...)
54
+ */
55
+ export function getQQBotDataDir(...subPaths) {
56
+ const dir = path.join(getHomeDir(), ".openclaw", "qqbot", ...subPaths);
57
+ if (!fs.existsSync(dir)) {
58
+ fs.mkdirSync(dir, { recursive: true });
59
+ }
60
+ return dir;
61
+ }
62
+ /**
63
+ * 获取 .openclaw/media/qqbot 下的子目录路径,并自动创建
64
+ *
65
+ * 与 getQQBotDataDir 不同,此目录位于 OpenClaw 核心的媒体安全白名单
66
+ * (~/.openclaw/media) 之下,下载到这里的文件可以被框架的 image/media
67
+ * 工具直接访问,不会触发 "Local media path is not under an allowed directory" 错误。
68
+ *
69
+ * 用于存放从 QQ 下载的图片、语音等需要被框架处理的媒体文件。
70
+ */
71
+ export function getQQBotMediaDir(...subPaths) {
72
+ const dir = path.join(getHomeDir(), ".openclaw", "media", "qqbot", ...subPaths);
73
+ if (!fs.existsSync(dir)) {
74
+ fs.mkdirSync(dir, { recursive: true });
75
+ }
76
+ return dir;
77
+ }
78
+ // ============ 临时目录 ============
79
+ /**
80
+ * 获取系统临时目录(跨平台安全)
81
+ * Mac: /var/folders/... 或 /tmp
82
+ * Linux: /tmp
83
+ * Windows: %TEMP% 或 C:\Users\xxx\AppData\Local\Temp
84
+ */
85
+ export function getTempDir() {
86
+ return os.tmpdir();
87
+ }
88
+ // ============ 波浪线路径展开 ============
89
+ /**
90
+ * 展开路径中的波浪线(~)为用户主目录
91
+ *
92
+ * Mac/Linux 用户经常使用 `~/Desktop/file.png` 这样的路径,
93
+ * 但 Node.js 的 fs 模块不会像 shell 一样自动展开 `~`。
94
+ *
95
+ * 支持:
96
+ * - `~/xxx` → `/Users/you/xxx`(Mac)或 `/home/you/xxx`(Linux)
97
+ * - `~` → `/Users/you`
98
+ * - 非 `~` 开头的路径原样返回
99
+ *
100
+ * 注意: 不支持 `~otheruser/xxx` 语法(极少使用,且需要系统调用获取其他用户信息)
101
+ */
102
+ export function expandTilde(p) {
103
+ if (!p)
104
+ return p;
105
+ if (p === "~")
106
+ return getHomeDir();
107
+ if (p.startsWith("~/") || p.startsWith("~\\")) {
108
+ return path.join(getHomeDir(), p.slice(2));
109
+ }
110
+ return p;
111
+ }
112
+ /**
113
+ * 对路径进行完整的规范化处理:剥离 file:// 前缀 + 展开波浪线 + 去除首尾空白
114
+ * 所有文件操作前应通过此函数处理用户输入的路径
115
+ */
116
+ export function normalizePath(p) {
117
+ let result = p.trim();
118
+ // 剥离 file:// 协议前缀: file:///Users/... → /Users/...
119
+ if (result.startsWith("file://")) {
120
+ result = result.slice("file://".length);
121
+ // 处理 URL 编码(file:// 路径中空格等字符可能被编码)
122
+ try {
123
+ result = decodeURIComponent(result);
124
+ }
125
+ catch {
126
+ // decodeURIComponent 失败时保留原样
127
+ }
128
+ }
129
+ return expandTilde(result);
130
+ }
131
+ // ============ 文件名 UTF-8 规范化 ============
132
+ /**
133
+ * 规范化文件名为 QQ Bot API 要求的 UTF-8 编码格式
134
+ *
135
+ * 问题场景:
136
+ * - macOS HFS+/APFS 文件系统使用 NFD(Unicode 分解形式)存储文件名,
137
+ * 例如「中文.txt」被分解为多个码点,QQ Bot API 可能拒绝
138
+ * - 文件名可能包含 API 不接受的特殊控制字符
139
+ * - URL 路径中可能包含 percent-encoded 的文件名需要解码
140
+ *
141
+ * 处理:
142
+ * 1. Unicode NFC 规范化(将 NFD 分解形式合并为 NFC 组合形式)
143
+ * 2. 去除 ASCII 控制字符(0x00-0x1F, 0x7F)
144
+ * 3. 去除首尾空白
145
+ * 4. 对 percent-encoded 的文件名尝试 URI 解码
146
+ */
147
+ export function sanitizeFileName(name) {
148
+ if (!name)
149
+ return name;
150
+ let result = name.trim();
151
+ // 尝试 URI 解码(处理 URL 中 percent-encoded 的中文文件名)
152
+ // 例如 %E4%B8%AD%E6%96%87.txt → 中文.txt
153
+ if (result.includes("%")) {
154
+ try {
155
+ result = decodeURIComponent(result);
156
+ }
157
+ catch {
158
+ // 解码失败(非合法 percent-encoding),保留原始值
159
+ }
160
+ }
161
+ // Unicode NFC 规范化:将 macOS NFD 分解形式合并为标准 NFC 组合形式
162
+ result = result.normalize("NFC");
163
+ // 去除 ASCII 控制字符(保留所有可打印字符和非 ASCII Unicode 字符)
164
+ result = result.replace(/[\x00-\x1F\x7F]/g, "");
165
+ return result;
166
+ }
167
+ // ============ 本地路径判断 ============
168
+ /**
169
+ * 判断字符串是否为本地文件路径(非 URL)
170
+ *
171
+ * 覆盖:
172
+ * - Unix 绝对路径: /Users/..., /home/..., /tmp/...
173
+ * - Windows 绝对路径: C:\..., D:/..., \\server\share
174
+ * - 相对路径: ./file, ../file
175
+ * - 波浪线路径: ~/Desktop/file.png
176
+ * - file:// 协议: file:///Users/..., file:///home/...
177
+ *
178
+ * 不匹配:
179
+ * - http:// / https:// URL
180
+ * - data: URL
181
+ */
182
+ export function isLocalPath(p) {
183
+ if (!p)
184
+ return false;
185
+ // file:// 协议(本地文件 URI)
186
+ if (p.startsWith("file://"))
187
+ return true;
188
+ // 波浪线路径(Mac/Linux 用户常用)
189
+ if (p === "~" || p.startsWith("~/") || p.startsWith("~\\"))
190
+ return true;
191
+ // Unix 绝对路径
192
+ if (p.startsWith("/"))
193
+ return true;
194
+ // Windows 盘符: C:\ 或 C:/
195
+ if (/^[a-zA-Z]:[\\/]/.test(p))
196
+ return true;
197
+ // Windows UNC: \\server\share
198
+ if (p.startsWith("\\\\"))
199
+ return true;
200
+ // 相对路径
201
+ if (p.startsWith("./") || p.startsWith("../"))
202
+ return true;
203
+ // Windows 相对路径
204
+ if (p.startsWith(".\\") || p.startsWith("..\\"))
205
+ return true;
206
+ return false;
207
+ }
208
+ /**
209
+ * 判断 markdown 中提取的路径是否像本地路径
210
+ * 比 isLocalPath 更宽松,用于从 markdown ![](path) 中检测误用
211
+ */
212
+ export function looksLikeLocalPath(p) {
213
+ if (isLocalPath(p))
214
+ return true;
215
+ // 常见系统目录前缀(不以 / 开头时也匹配)
216
+ return /^(?:Users|home|tmp|var|private|[A-Z]:)/i.test(p);
217
+ }
218
+ // ============ ffmpeg 跨平台检测 ============
219
+ let _ffmpegPath; // undefined = 未检测, null = 不可用
220
+ let _ffmpegCheckPromise = null;
221
+ /**
222
+ * 检测 ffmpeg 是否可用,返回可执行路径
223
+ *
224
+ * Windows 上检测 ffmpeg.exe,Mac/Linux 检测 ffmpeg
225
+ * 支持通过环境变量 FFMPEG_PATH 指定自定义路径
226
+ *
227
+ * @returns ffmpeg 可执行文件路径,不可用返回 null
228
+ */
229
+ export function detectFfmpeg() {
230
+ if (_ffmpegPath !== undefined)
231
+ return Promise.resolve(_ffmpegPath);
232
+ if (_ffmpegCheckPromise)
233
+ return _ffmpegCheckPromise;
234
+ _ffmpegCheckPromise = (async () => {
235
+ // 1. 环境变量自定义路径
236
+ const envPath = process.env.FFMPEG_PATH;
237
+ if (envPath) {
238
+ const ok = await testExecutable(envPath, ["-version"]);
239
+ if (ok) {
240
+ _ffmpegPath = envPath;
241
+ console.log(`[platform] ffmpeg found via FFMPEG_PATH: ${envPath}`);
242
+ return _ffmpegPath;
243
+ }
244
+ console.warn(`[platform] FFMPEG_PATH set but not working: ${envPath}`);
245
+ }
246
+ // 2. 系统 PATH 中检测
247
+ const cmd = isWindows() ? "ffmpeg.exe" : "ffmpeg";
248
+ const ok = await testExecutable(cmd, ["-version"]);
249
+ if (ok) {
250
+ _ffmpegPath = cmd;
251
+ console.log(`[platform] ffmpeg detected in PATH`);
252
+ return _ffmpegPath;
253
+ }
254
+ // 3. 常见安装位置(Mac brew、Windows choco/scoop)
255
+ const commonPaths = isWindows()
256
+ ? [
257
+ "C:\\ffmpeg\\bin\\ffmpeg.exe",
258
+ path.join(process.env.LOCALAPPDATA || "", "Programs", "ffmpeg", "bin", "ffmpeg.exe"),
259
+ path.join(process.env.ProgramFiles || "", "ffmpeg", "bin", "ffmpeg.exe"),
260
+ ]
261
+ : [
262
+ "/usr/local/bin/ffmpeg", // Mac brew
263
+ "/opt/homebrew/bin/ffmpeg", // Mac ARM brew
264
+ "/usr/bin/ffmpeg", // Linux apt
265
+ "/snap/bin/ffmpeg", // Linux snap
266
+ ];
267
+ for (const p of commonPaths) {
268
+ if (p && fs.existsSync(p)) {
269
+ const works = await testExecutable(p, ["-version"]);
270
+ if (works) {
271
+ _ffmpegPath = p;
272
+ console.log(`[platform] ffmpeg found at: ${p}`);
273
+ return _ffmpegPath;
274
+ }
275
+ }
276
+ }
277
+ _ffmpegPath = null;
278
+ return null;
279
+ })().finally(() => {
280
+ _ffmpegCheckPromise = null;
281
+ });
282
+ return _ffmpegCheckPromise;
283
+ }
284
+ /** 测试可执行文件是否能正常运行 */
285
+ function testExecutable(cmd, args) {
286
+ return new Promise((resolve) => {
287
+ execFile(cmd, args, { timeout: 5000 }, (err) => {
288
+ resolve(!err);
289
+ });
290
+ });
291
+ }
292
+ /** 重置 ffmpeg 缓存(用于测试) */
293
+ export function resetFfmpegCache() {
294
+ _ffmpegPath = undefined;
295
+ _ffmpegCheckPromise = null;
296
+ }
297
+ // ============ silk-wasm 兼容性 ============
298
+ let _silkWasmAvailable = null;
299
+ /**
300
+ * 检测 silk-wasm 是否可用
301
+ *
302
+ * silk-wasm 依赖 WASM 运行时,在某些环境(如老版本 Node、某些容器)可能不可用。
303
+ * 提前检测避免运行时崩溃。
304
+ */
305
+ export async function checkSilkWasmAvailable() {
306
+ if (_silkWasmAvailable !== null)
307
+ return _silkWasmAvailable;
308
+ try {
309
+ const { isSilk } = await import("silk-wasm");
310
+ // 用一个空 buffer 快速测试 WASM 是否能加载
311
+ isSilk(new Uint8Array(0));
312
+ _silkWasmAvailable = true;
313
+ console.log("[platform] silk-wasm: available");
314
+ }
315
+ catch (err) {
316
+ _silkWasmAvailable = false;
317
+ console.warn(`[platform] silk-wasm: NOT available (${err instanceof Error ? err.message : String(err)})`);
318
+ }
319
+ return _silkWasmAvailable;
320
+ }
321
+ /**
322
+ * 运行启动诊断,返回环境报告
323
+ * 在 gateway 启动时调用,打印环境信息并给出警告
324
+ */
325
+ export async function runDiagnostics() {
326
+ const warnings = [];
327
+ const platform = `${process.platform} (${os.release()})`;
328
+ const arch = process.arch;
329
+ const nodeVersion = process.version;
330
+ const homeDir = getHomeDir();
331
+ const tempDir = getTempDir();
332
+ const dataDir = getQQBotDataDir();
333
+ // 检测 ffmpeg
334
+ const ffmpegPath = await detectFfmpeg();
335
+ if (!ffmpegPath) {
336
+ warnings.push(isWindows()
337
+ ? "⚠️ ffmpeg 未安装。语音/视频格式转换将受限。安装方式: choco install ffmpeg 或 scoop install ffmpeg 或从 https://ffmpeg.org 下载"
338
+ : getPlatform() === "darwin"
339
+ ? "⚠️ ffmpeg 未安装。语音/视频格式转换将受限。安装方式: brew install ffmpeg"
340
+ : "⚠️ ffmpeg 未安装。语音/视频格式转换将受限。安装方式: sudo apt install ffmpeg 或 sudo yum install ffmpeg");
341
+ }
342
+ // 检测 silk-wasm
343
+ const silkWasm = await checkSilkWasmAvailable();
344
+ if (!silkWasm) {
345
+ warnings.push("⚠️ silk-wasm 不可用。QQ 语音消息的收发将无法工作。请确认 Node.js 版本 >= 16 且 WASM 支持正常");
346
+ }
347
+ // 检查数据目录可写性
348
+ try {
349
+ const testFile = path.join(dataDir, ".write-test");
350
+ fs.writeFileSync(testFile, "test");
351
+ fs.unlinkSync(testFile);
352
+ }
353
+ catch {
354
+ warnings.push(`⚠️ 数据目录不可写: ${dataDir}。请检查权限`);
355
+ }
356
+ // Windows 特殊提醒
357
+ if (isWindows()) {
358
+ // 检查路径中是否有中文或空格(可能导致某些工具异常)
359
+ if (/[\u4e00-\u9fa5]/.test(homeDir) || homeDir.includes(" ")) {
360
+ warnings.push(`⚠️ 用户目录包含中文或空格: ${homeDir}。某些工具可能无法正常工作,建议设置 QQBOT_DATA_DIR 环境变量指定纯英文路径`);
361
+ }
362
+ }
363
+ const report = {
364
+ platform,
365
+ arch,
366
+ nodeVersion,
367
+ homeDir,
368
+ tempDir,
369
+ dataDir,
370
+ ffmpeg: ffmpegPath,
371
+ silkWasm,
372
+ warnings,
373
+ };
374
+ // 打印诊断报告
375
+ console.log("=== QQBot 环境诊断 ===");
376
+ console.log(` 平台: ${platform} (${arch})`);
377
+ console.log(` Node: ${nodeVersion}`);
378
+ console.log(` 主目录: ${homeDir}`);
379
+ console.log(` 数据目录: ${dataDir}`);
380
+ console.log(` ffmpeg: ${ffmpegPath ?? "未安装"}`);
381
+ console.log(` silk-wasm: ${silkWasm ? "可用" : "不可用"}`);
382
+ if (warnings.length > 0) {
383
+ console.log(" --- 警告 ---");
384
+ for (const w of warnings) {
385
+ console.log(` ${w}`);
386
+ }
387
+ }
388
+ console.log("======================");
389
+ return report;
390
+ }
@@ -0,0 +1,25 @@
1
+ /**
2
+ * 远程 URL 安全校验
3
+ *
4
+ * 下载外部资源前,确保目标地址不会命中内部网络或云元数据端点,
5
+ * 避免模型输出的恶意链接触达内网服务。
6
+ */
7
+ /**
8
+ * 检查给定 IP 是否落在不可路由 / 私有网段内。
9
+ *
10
+ * 覆盖:
11
+ * - IPv4: 127/8, 10/8, 172.16/12, 192.168/16, 169.254/16, 0.0.0.0
12
+ * - IPv6: ::1, ::, fe80 (link-local), fc/fd (ULA)
13
+ */
14
+ export declare function isReservedAddr(ip: string): boolean;
15
+ /**
16
+ * 校验远程 URL 是否可安全请求。
17
+ *
18
+ * 规则:
19
+ * 1. 仅放行 http / https 协议
20
+ * 2. 若 URL 直接携带 IP 则即时判定
21
+ * 3. 若为域名则先做 DNS 解析,逐条检查解析结果
22
+ *
23
+ * @throws {Error} 当 URL 指向受限地址时
24
+ */
25
+ export declare function validateRemoteUrl(raw: string): Promise<void>;
@@ -0,0 +1,91 @@
1
+ /**
2
+ * 远程 URL 安全校验
3
+ *
4
+ * 下载外部资源前,确保目标地址不会命中内部网络或云元数据端点,
5
+ * 避免模型输出的恶意链接触达内网服务。
6
+ */
7
+ import net from "node:net";
8
+ import dns from "node:dns/promises";
9
+ /* ---------- 内网 / 保留地址判定 ---------- */
10
+ /** IPv4 保留网段前缀(覆盖 RFC 1918、链路本地、回环等) */
11
+ const RESERVED_V4_PREFIXES = [
12
+ "127.", // loopback
13
+ "10.", // class-A private
14
+ "192.168.", // class-C private
15
+ "169.254.", // link-local / cloud metadata
16
+ ];
17
+ /** 172.16.0.0 – 172.31.255.255 需要单独用正则匹配 */
18
+ const PRIVATE_172_RE = /^172\.(1[6-9]|2\d|3[01])\./;
19
+ /**
20
+ * 检查给定 IP 是否落在不可路由 / 私有网段内。
21
+ *
22
+ * 覆盖:
23
+ * - IPv4: 127/8, 10/8, 172.16/12, 192.168/16, 169.254/16, 0.0.0.0
24
+ * - IPv6: ::1, ::, fe80 (link-local), fc/fd (ULA)
25
+ */
26
+ export function isReservedAddr(ip) {
27
+ // --- IPv4 ---
28
+ if (ip === "0.0.0.0")
29
+ return true;
30
+ for (const pfx of RESERVED_V4_PREFIXES) {
31
+ if (ip.startsWith(pfx))
32
+ return true;
33
+ }
34
+ if (PRIVATE_172_RE.test(ip))
35
+ return true;
36
+ // --- IPv6 ---
37
+ const lower = ip.toLowerCase();
38
+ if (lower === "::1" || lower === "::")
39
+ return true;
40
+ if (lower.startsWith("fe80:"))
41
+ return true; // link-local
42
+ if (lower.startsWith("fc") || lower.startsWith("fd"))
43
+ return true; // unique local
44
+ return false;
45
+ }
46
+ /* ---------- URL 合法性校验 ---------- */
47
+ const ALLOWED_SCHEMES = new Set(["http:", "https:"]);
48
+ /**
49
+ * 校验远程 URL 是否可安全请求。
50
+ *
51
+ * 规则:
52
+ * 1. 仅放行 http / https 协议
53
+ * 2. 若 URL 直接携带 IP 则即时判定
54
+ * 3. 若为域名则先做 DNS 解析,逐条检查解析结果
55
+ *
56
+ * @throws {Error} 当 URL 指向受限地址时
57
+ */
58
+ export async function validateRemoteUrl(raw) {
59
+ const url = new URL(raw);
60
+ if (!ALLOWED_SCHEMES.has(url.protocol)) {
61
+ throw new Error(`不支持的协议 "${url.protocol}",仅允许 http/https(URL: ${raw})`);
62
+ }
63
+ // 去掉 IPv6 方括号
64
+ const host = url.hostname.replace(/^\[|\]$/g, "");
65
+ if (net.isIP(host)) {
66
+ assertPublicAddr(host, raw);
67
+ return;
68
+ }
69
+ // 域名 → 解析后逐条检查
70
+ try {
71
+ const ips = await dns.resolve(host);
72
+ for (const ip of ips) {
73
+ assertPublicAddr(ip, raw, host);
74
+ }
75
+ }
76
+ catch (err) {
77
+ // 已经是我们自己抛的安全错误,继续向上传播
78
+ if (err instanceof Error && err.message.includes("内网"))
79
+ throw err;
80
+ // DNS 查询失败不阻塞,后续 fetch 会产生网络错误
81
+ console.warn(`[url-check] DNS 解析 "${host}" 失败: ${err}`);
82
+ }
83
+ }
84
+ /* ---------- 内部辅助 ---------- */
85
+ /** 断言 IP 为公网地址,否则抛出错误 */
86
+ function assertPublicAddr(ip, originalUrl, domain) {
87
+ if (!isReservedAddr(ip))
88
+ return;
89
+ const target = domain ? `域名 "${domain}" 解析到内网地址 "${ip}"` : `内网地址 "${ip}"`;
90
+ throw new Error(`禁止访问${target},已拦截潜在的 SSRF 请求(URL: ${originalUrl})`);
91
+ }
@@ -0,0 +1,32 @@
1
+ /**
2
+ * QQ Bot 文本解析工具函数
3
+ */
4
+ import type { RefAttachmentSummary } from "../ref-index-store.js";
5
+ /**
6
+ * 解析 QQ 表情标签,将 <faceType=1,faceId="13",ext="base64..."> 格式
7
+ * 替换为 【表情: 中文名】 格式
8
+ * ext 字段为 Base64 编码的 JSON,格式如 {"text":"呲牙"}
9
+ */
10
+ export declare function parseFaceTags(text: string): string;
11
+ /**
12
+ * 过滤内部标记(如 [[reply_to: xxx]])
13
+ * 这些标记可能被 AI 错误地学习并输出,需要在发送前移除
14
+ */
15
+ export declare function filterInternalMarkers(text: string): string;
16
+ /**
17
+ * 从 message_scene.ext 数组中解析引用索引
18
+ * ext 格式示例: ["", "ref_msg_idx=REFIDX_xxx", "msg_idx=REFIDX_yyy"]
19
+ */
20
+ export declare function parseRefIndices(ext?: string[]): {
21
+ refMsgIdx?: string;
22
+ msgIdx?: string;
23
+ };
24
+ /**
25
+ * 从附件列表中构建附件摘要(用于引用索引缓存)
26
+ */
27
+ export declare function buildAttachmentSummaries(attachments?: Array<{
28
+ content_type: string;
29
+ url: string;
30
+ filename?: string;
31
+ voice_wav_url?: string;
32
+ }>, localPaths?: Array<string | null>): RefAttachmentSummary[] | undefined;
@@ -0,0 +1,69 @@
1
+ /**
2
+ * QQ Bot 文本解析工具函数
3
+ */
4
+ import { inferAttachmentType } from "../group-history.js";
5
+ /**
6
+ * 解析 QQ 表情标签,将 <faceType=1,faceId="13",ext="base64..."> 格式
7
+ * 替换为 【表情: 中文名】 格式
8
+ * ext 字段为 Base64 编码的 JSON,格式如 {"text":"呲牙"}
9
+ */
10
+ export function parseFaceTags(text) {
11
+ if (!text)
12
+ return text;
13
+ return text.replace(/<faceType=\d+,faceId="[^"]*",ext="([^"]*)">/g, (_match, ext) => {
14
+ try {
15
+ const decoded = Buffer.from(ext, "base64").toString("utf-8");
16
+ const parsed = JSON.parse(decoded);
17
+ const faceName = parsed.text || "未知表情";
18
+ return `【表情: ${faceName}】`;
19
+ }
20
+ catch {
21
+ return _match;
22
+ }
23
+ });
24
+ }
25
+ /**
26
+ * 过滤内部标记(如 [[reply_to: xxx]])
27
+ * 这些标记可能被 AI 错误地学习并输出,需要在发送前移除
28
+ */
29
+ export function filterInternalMarkers(text) {
30
+ if (!text)
31
+ return text;
32
+ let result = text.replace(/\[\[[a-z_]+:\s*[^\]]*\]\]/gi, "");
33
+ // 过滤框架内部图片引用标记:@image:image_xxx.png、@voice:voice_xxx.silk 等
34
+ result = result.replace(/@(?:image|voice|video|file):[a-zA-Z0-9_.-]+/g, "");
35
+ result = result.replace(/\n{3,}/g, "\n\n").trim();
36
+ return result;
37
+ }
38
+ /**
39
+ * 从 message_scene.ext 数组中解析引用索引
40
+ * ext 格式示例: ["", "ref_msg_idx=REFIDX_xxx", "msg_idx=REFIDX_yyy"]
41
+ */
42
+ export function parseRefIndices(ext) {
43
+ if (!ext || ext.length === 0)
44
+ return {};
45
+ let refMsgIdx;
46
+ let msgIdx;
47
+ for (const item of ext) {
48
+ if (item.startsWith("ref_msg_idx=")) {
49
+ refMsgIdx = item.slice("ref_msg_idx=".length);
50
+ }
51
+ else if (item.startsWith("msg_idx=")) {
52
+ msgIdx = item.slice("msg_idx=".length);
53
+ }
54
+ }
55
+ return { refMsgIdx, msgIdx };
56
+ }
57
+ /**
58
+ * 从附件列表中构建附件摘要(用于引用索引缓存)
59
+ */
60
+ export function buildAttachmentSummaries(attachments, localPaths) {
61
+ if (!attachments || attachments.length === 0)
62
+ return undefined;
63
+ return attachments.map((att, idx) => ({
64
+ type: inferAttachmentType(att.content_type),
65
+ filename: att.filename,
66
+ contentType: att.content_type,
67
+ localPath: localPaths?.[idx] ?? undefined,
68
+ }));
69
+ }
@@ -0,0 +1,34 @@
1
+ /**
2
+ * file_info 缓存 — 借鉴 Telegram file_id 机制
3
+ *
4
+ * QQ Bot API 上传文件后返回 file_info + ttl,在 TTL 内相同文件可直接复用 file_info
5
+ * 避免重复上传同一文件,节省带宽和时间。
6
+ *
7
+ * 缓存 key = md5(fileContent) + targetType(c2c/group) + targetId + fileType
8
+ */
9
+ /**
10
+ * 计算文件内容的 MD5 hash(用于缓存 key)
11
+ * 对于 Base64 数据直接 hash,对于文件路径读取后 hash
12
+ */
13
+ export declare function computeFileHash(data: string | Buffer): string;
14
+ /**
15
+ * 从缓存获取 file_info
16
+ * @returns file_info 字符串,未命中或已过期返回 null
17
+ */
18
+ export declare function getCachedFileInfo(contentHash: string, scope: "c2c" | "group", targetId: string, fileType: number): string | null;
19
+ /**
20
+ * 将上传结果写入缓存
21
+ * @param ttl - API 返回的 TTL(秒),缓存会提前 60 秒失效
22
+ */
23
+ export declare function setCachedFileInfo(contentHash: string, scope: "c2c" | "group", targetId: string, fileType: number, fileInfo: string, fileUuid: string, ttl: number): void;
24
+ /**
25
+ * 获取缓存统计
26
+ */
27
+ export declare function getUploadCacheStats(): {
28
+ size: number;
29
+ maxSize: number;
30
+ };
31
+ /**
32
+ * 清除所有缓存
33
+ */
34
+ export declare function clearUploadCache(): void;