@yanhaidao/wecom 2.4.120 → 2.5.110

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 (323) hide show
  1. package/README.md +4 -5
  2. package/dist/index.js +68 -0
  3. package/dist/src/accounts.js +20 -0
  4. package/dist/src/agent/handler.js +895 -0
  5. package/dist/src/agent/index.js +5 -0
  6. package/dist/src/app/account-runtime.js +216 -0
  7. package/dist/src/app/bootstrap.js +19 -0
  8. package/dist/src/app/index.js +118 -0
  9. package/dist/src/capability/agent/delivery-service.js +63 -0
  10. package/dist/src/capability/agent/fallback-policy.js +6 -0
  11. package/dist/src/capability/agent/ingress-service.js +33 -0
  12. package/dist/src/capability/agent/upstream-delivery-service.js +71 -0
  13. package/dist/src/capability/bot/dispatch-config.js +45 -0
  14. package/dist/src/capability/bot/fallback-delivery.js +147 -0
  15. package/dist/src/capability/bot/local-path-delivery.js +178 -0
  16. package/dist/src/capability/bot/sandbox-media.js +138 -0
  17. package/dist/src/capability/bot/service.js +49 -0
  18. package/dist/src/capability/bot/stream-delivery.js +321 -0
  19. package/dist/src/capability/bot/stream-finalizer.js +81 -0
  20. package/dist/src/capability/bot/stream-orchestrator.js +318 -0
  21. package/dist/src/capability/bot/types.js +1 -0
  22. package/{src/capability/calendar/client.ts → dist/src/capability/calendar/client.js} +118 -241
  23. package/{src/capability/calendar/schema.ts → dist/src/capability/calendar/schema.js} +0 -38
  24. package/dist/src/capability/calendar/tool.js +365 -0
  25. package/dist/src/capability/calendar/types.js +12 -0
  26. package/{src/capability/doc/client.ts → dist/src/capability/doc/client.js} +370 -605
  27. package/{src/capability/doc/schema.ts → dist/src/capability/doc/schema.js} +345 -394
  28. package/dist/src/capability/doc/tool.js +1556 -0
  29. package/dist/src/capability/doc/types.js +113 -0
  30. package/dist/src/capability/mcp/index.js +3 -0
  31. package/dist/src/capability/mcp/schema.js +102 -0
  32. package/dist/src/capability/mcp/tool.js +146 -0
  33. package/dist/src/capability/mcp/transport.js +293 -0
  34. package/dist/src/channel.js +224 -0
  35. package/dist/src/config/accounts.js +236 -0
  36. package/dist/src/config/derived-paths.js +31 -0
  37. package/dist/src/config/index.js +7 -0
  38. package/dist/src/config/media.js +110 -0
  39. package/dist/src/config/network.js +32 -0
  40. package/dist/src/config/routing.js +20 -0
  41. package/dist/src/config/runtime-config.js +25 -0
  42. package/dist/src/config/schema.js +4 -0
  43. package/{src/config-schema.ts → dist/src/config-schema.js} +1 -1
  44. package/dist/src/context-store.js +219 -0
  45. package/{src/crypto/aes.ts → dist/src/crypto/aes.js} +11 -28
  46. package/dist/src/crypto/index.js +9 -0
  47. package/{src/crypto/signature.ts → dist/src/crypto/signature.js} +3 -18
  48. package/{src/crypto/xml.ts → dist/src/crypto/xml.js} +3 -11
  49. package/dist/src/crypto.js +145 -0
  50. package/dist/src/domain/models.js +1 -0
  51. package/dist/src/domain/policies.js +32 -0
  52. package/{src/dynamic-agent.ts → dist/src/dynamic-agent.js} +36 -73
  53. package/dist/src/gateway-monitor.js +139 -0
  54. package/dist/src/http.js +114 -0
  55. package/{src/media.ts → dist/src/media.js} +21 -40
  56. package/dist/src/monitor/limits.js +7 -0
  57. package/dist/src/monitor/state.js +28 -0
  58. package/dist/src/monitor.js +84 -0
  59. package/dist/src/observability/audit-log.js +30 -0
  60. package/dist/src/observability/legacy-operational-event-store.js +22 -0
  61. package/dist/src/observability/raw-envelope-log.js +24 -0
  62. package/dist/src/observability/status-registry.js +9 -0
  63. package/dist/src/observability/transport-session-view.js +14 -0
  64. package/dist/src/onboarding.js +546 -0
  65. package/dist/src/outbound.js +557 -0
  66. package/dist/src/runtime/dispatcher.js +57 -0
  67. package/{src/runtime/index.ts → dist/src/runtime/index.js} +0 -1
  68. package/dist/src/runtime/outbound-intent.js +1 -0
  69. package/dist/src/runtime/reply-orchestrator.js +38 -0
  70. package/dist/src/runtime/routing-bridge.js +26 -0
  71. package/dist/src/runtime/session-manager.js +112 -0
  72. package/dist/src/runtime/source-registry.js +174 -0
  73. package/dist/src/runtime.js +1 -0
  74. package/dist/src/shared/command-auth.js +57 -0
  75. package/{src/shared/index.ts → dist/src/shared/index.js} +0 -1
  76. package/dist/src/shared/media-asset.js +65 -0
  77. package/dist/src/shared/media-service.js +59 -0
  78. package/dist/src/shared/media-types.js +1 -0
  79. package/{src/shared/xml-parser.ts → dist/src/shared/xml-parser.js} +72 -63
  80. package/dist/src/store/active-reply-store.js +41 -0
  81. package/dist/src/store/interfaces.js +1 -0
  82. package/dist/src/store/memory-store.js +33 -0
  83. package/dist/src/store/stream-batch-store.js +319 -0
  84. package/{src/target.ts → dist/src/target.js} +15 -48
  85. package/dist/src/transport/agent-api/client.js +168 -0
  86. package/dist/src/transport/agent-api/core.js +337 -0
  87. package/dist/src/transport/agent-api/delivery.js +28 -0
  88. package/dist/src/transport/agent-api/media-upload.js +4 -0
  89. package/dist/src/transport/agent-api/reply.js +24 -0
  90. package/dist/src/transport/agent-api/upstream-delivery.js +30 -0
  91. package/dist/src/transport/agent-api/upstream-media-upload.js +46 -0
  92. package/dist/src/transport/agent-api/upstream-reply.js +26 -0
  93. package/dist/src/transport/agent-callback/http-handler.js +30 -0
  94. package/dist/src/transport/agent-callback/inbound.js +4 -0
  95. package/dist/src/transport/agent-callback/reply.js +8 -0
  96. package/dist/src/transport/agent-callback/request-handler.js +189 -0
  97. package/dist/src/transport/agent-callback/session.js +15 -0
  98. package/dist/src/transport/bot-webhook/active-reply.js +27 -0
  99. package/dist/src/transport/bot-webhook/http-handler.js +31 -0
  100. package/dist/src/transport/bot-webhook/inbound-normalizer.js +496 -0
  101. package/dist/src/transport/bot-webhook/inbound.js +4 -0
  102. package/dist/src/transport/bot-webhook/message-shape.js +98 -0
  103. package/dist/src/transport/bot-webhook/protocol.js +124 -0
  104. package/dist/src/transport/bot-webhook/reply.js +9 -0
  105. package/dist/src/transport/bot-webhook/request-handler.js +285 -0
  106. package/dist/src/transport/bot-webhook/session.js +15 -0
  107. package/dist/src/transport/bot-ws/inbound.js +147 -0
  108. package/dist/src/transport/bot-ws/media.js +236 -0
  109. package/dist/src/transport/bot-ws/reply.js +310 -0
  110. package/dist/src/transport/bot-ws/sdk-adapter.js +257 -0
  111. package/dist/src/transport/bot-ws/session.js +15 -0
  112. package/dist/src/transport/http/common.js +78 -0
  113. package/dist/src/transport/http/registry.js +71 -0
  114. package/dist/src/transport/http/request-handler.js +51 -0
  115. package/{src/transport/index.ts → dist/src/transport/index.js} +2 -10
  116. package/dist/src/types/account.js +1 -0
  117. package/dist/src/types/config.js +1 -0
  118. package/dist/src/types/constants.js +28 -0
  119. package/dist/src/types/events.js +1 -0
  120. package/dist/src/types/index.js +1 -0
  121. package/dist/src/types/legacy-stream.js +1 -0
  122. package/dist/src/types/message.js +5 -0
  123. package/dist/src/types/runtime-context.js +1 -0
  124. package/dist/src/types/runtime.js +1 -0
  125. package/dist/src/types.js +1 -0
  126. package/dist/src/upstream/index.js +111 -0
  127. package/dist/src/wecom_msg_adapter/markdown_adapter.js +280 -0
  128. package/openclaw.plugin.json +15 -0
  129. package/package.json +18 -1
  130. package/.github/workflows/release.yml +0 -143
  131. package/GOVERNANCE.md +0 -26
  132. package/MENU_EVENT_CONF.md +0 -500
  133. package/MENU_EVENT_PLAN.md +0 -440
  134. package/SKILLS_CAL.md +0 -895
  135. package/SKILLS_DOC.md +0 -2288
  136. package/UPSTREAM_CONFIG.md +0 -170
  137. package/UPSTREAM_PLAN.md +0 -175
  138. package/assets/01.bot-add.png +0 -0
  139. package/assets/01.bot-setp2.png +0 -0
  140. package/assets/01.image.jpg +0 -0
  141. package/assets/02.agent.add.png +0 -0
  142. package/assets/02.agent.api-set.png +0 -0
  143. package/assets/02.image.jpg +0 -0
  144. package/assets/03.agent.page.png +0 -0
  145. package/assets/03.bot.page.png +0 -0
  146. package/assets/link-me.jpg +0 -0
  147. package/assets/register.png +0 -0
  148. package/changelog/v2.2.28.md +0 -70
  149. package/changelog/v2.3.10.md +0 -17
  150. package/changelog/v2.3.11.md +0 -19
  151. package/changelog/v2.3.12.md +0 -25
  152. package/changelog/v2.3.13.md +0 -19
  153. package/changelog/v2.3.14.md +0 -48
  154. package/changelog/v2.3.15.md +0 -15
  155. package/changelog/v2.3.16.md +0 -11
  156. package/changelog/v2.3.18.md +0 -22
  157. package/changelog/v2.3.19.md +0 -73
  158. package/changelog/v2.3.2.md +0 -28
  159. package/changelog/v2.3.26.md +0 -21
  160. package/changelog/v2.3.27.md +0 -33
  161. package/changelog/v2.3.4.md +0 -20
  162. package/changelog/v2.3.9.md +0 -22
  163. package/changelog/v2.4.12.md +0 -37
  164. package/compat-single-account.md +0 -148
  165. package/index.test.ts +0 -38
  166. package/scripts/test-proxy.ts +0 -70
  167. package/scripts/wecom/README.md +0 -123
  168. package/scripts/wecom/menu-click-help.js +0 -59
  169. package/scripts/wecom/menu-click-help.py +0 -55
  170. package/src/accounts.ts +0 -34
  171. package/src/agent/api-client.upload.test.ts +0 -109
  172. package/src/agent/event-router.test.ts +0 -421
  173. package/src/agent/event-router.ts +0 -272
  174. package/src/agent/handler.event-filter.test.ts +0 -135
  175. package/src/agent/handler.ts +0 -1250
  176. package/src/agent/index.ts +0 -12
  177. package/src/agent/script-runner.ts +0 -186
  178. package/src/agent/test-fixtures/invalid-json-script.mjs +0 -1
  179. package/src/agent/test-fixtures/reply-event-script.mjs +0 -29
  180. package/src/agent/test-fixtures/reply-event-script.py +0 -17
  181. package/src/app/account-runtime.ts +0 -276
  182. package/src/app/bootstrap.ts +0 -29
  183. package/src/app/index.ts +0 -192
  184. package/src/capability/agent/delivery-service.ts +0 -87
  185. package/src/capability/agent/fallback-policy.ts +0 -13
  186. package/src/capability/agent/ingress-service.ts +0 -38
  187. package/src/capability/agent/upstream-delivery-service.ts +0 -96
  188. package/src/capability/bot/dispatch-config.ts +0 -47
  189. package/src/capability/bot/fallback-delivery.ts +0 -178
  190. package/src/capability/bot/local-path-delivery.ts +0 -215
  191. package/src/capability/bot/sandbox-media.test.ts +0 -221
  192. package/src/capability/bot/sandbox-media.ts +0 -176
  193. package/src/capability/bot/service.ts +0 -56
  194. package/src/capability/bot/stream-delivery.ts +0 -379
  195. package/src/capability/bot/stream-finalizer.ts +0 -120
  196. package/src/capability/bot/stream-orchestrator.ts +0 -371
  197. package/src/capability/bot/types.ts +0 -8
  198. package/src/capability/calendar/SKILLS_CHECKLIST.md +0 -251
  199. package/src/capability/calendar/tool.ts +0 -417
  200. package/src/capability/calendar/types.ts +0 -309
  201. package/src/capability/doc/tool.ts +0 -1629
  202. package/src/capability/doc/types.ts +0 -792
  203. package/src/capability/mcp/index.ts +0 -10
  204. package/src/capability/mcp/schema.ts +0 -107
  205. package/src/capability/mcp/tool.ts +0 -174
  206. package/src/capability/mcp/transport.ts +0 -394
  207. package/src/channel.config.test.ts +0 -180
  208. package/src/channel.lifecycle.test.ts +0 -255
  209. package/src/channel.meta.test.ts +0 -26
  210. package/src/channel.ts +0 -256
  211. package/src/config/accounts.resolve.test.ts +0 -75
  212. package/src/config/accounts.ts +0 -312
  213. package/src/config/derived-paths.test.ts +0 -111
  214. package/src/config/derived-paths.ts +0 -41
  215. package/src/config/index.ts +0 -22
  216. package/src/config/media.test.ts +0 -113
  217. package/src/config/media.ts +0 -139
  218. package/src/config/network.ts +0 -20
  219. package/src/config/routing.test.ts +0 -88
  220. package/src/config/routing.ts +0 -26
  221. package/src/config/runtime-config.ts +0 -46
  222. package/src/config/schema.ts +0 -144
  223. package/src/context-store.ts +0 -297
  224. package/src/crypto/index.ts +0 -24
  225. package/src/crypto.test.ts +0 -32
  226. package/src/crypto.ts +0 -176
  227. package/src/domain/models.ts +0 -7
  228. package/src/domain/policies.ts +0 -36
  229. package/src/dynamic-agent.account-scope.test.ts +0 -17
  230. package/src/gateway-monitor.ts +0 -181
  231. package/src/http.ts +0 -137
  232. package/src/media.test.ts +0 -82
  233. package/src/monitor/limits.ts +0 -7
  234. package/src/monitor/state.queue.test.ts +0 -185
  235. package/src/monitor/state.ts +0 -34
  236. package/src/monitor.active.test.ts +0 -245
  237. package/src/monitor.inbound-filter.test.ts +0 -63
  238. package/src/monitor.integration.test.ts +0 -208
  239. package/src/monitor.ts +0 -121
  240. package/src/monitor.webhook.test.ts +0 -774
  241. package/src/observability/audit-log.ts +0 -48
  242. package/src/observability/legacy-operational-event-store.ts +0 -36
  243. package/src/observability/raw-envelope-log.ts +0 -28
  244. package/src/observability/status-registry.ts +0 -13
  245. package/src/observability/transport-session-view.ts +0 -14
  246. package/src/onboarding.test.ts +0 -336
  247. package/src/onboarding.ts +0 -704
  248. package/src/outbound.test.ts +0 -1271
  249. package/src/outbound.ts +0 -746
  250. package/src/runtime/dispatcher.ts +0 -71
  251. package/src/runtime/outbound-intent.ts +0 -4
  252. package/src/runtime/reply-orchestrator.test.ts +0 -71
  253. package/src/runtime/reply-orchestrator.ts +0 -67
  254. package/src/runtime/routing-bridge.test.ts +0 -115
  255. package/src/runtime/routing-bridge.ts +0 -44
  256. package/src/runtime/session-manager.test.ts +0 -174
  257. package/src/runtime/session-manager.ts +0 -139
  258. package/src/runtime/source-registry.ts +0 -249
  259. package/src/runtime.ts +0 -14
  260. package/src/shared/command-auth.ts +0 -87
  261. package/src/shared/media-asset.ts +0 -78
  262. package/src/shared/media-service.test.ts +0 -111
  263. package/src/shared/media-service.ts +0 -84
  264. package/src/shared/media-types.ts +0 -5
  265. package/src/shared/xml-parser.test.ts +0 -50
  266. package/src/store/active-reply-store.ts +0 -42
  267. package/src/store/interfaces.ts +0 -11
  268. package/src/store/memory-store.ts +0 -43
  269. package/src/store/stream-batch-store.ts +0 -350
  270. package/src/transport/agent-api/client.ts +0 -277
  271. package/src/transport/agent-api/core.ts +0 -463
  272. package/src/transport/agent-api/delivery.ts +0 -41
  273. package/src/transport/agent-api/media-upload.ts +0 -11
  274. package/src/transport/agent-api/reply.ts +0 -39
  275. package/src/transport/agent-api/upstream-delivery.ts +0 -45
  276. package/src/transport/agent-api/upstream-media-upload.ts +0 -70
  277. package/src/transport/agent-api/upstream-reply.ts +0 -43
  278. package/src/transport/agent-callback/http-handler.ts +0 -47
  279. package/src/transport/agent-callback/inbound.ts +0 -5
  280. package/src/transport/agent-callback/reply.ts +0 -13
  281. package/src/transport/agent-callback/request-handler.ts +0 -244
  282. package/src/transport/agent-callback/session.ts +0 -23
  283. package/src/transport/bot-webhook/active-reply.ts +0 -39
  284. package/src/transport/bot-webhook/http-handler.ts +0 -48
  285. package/src/transport/bot-webhook/inbound-normalizer.ts +0 -371
  286. package/src/transport/bot-webhook/inbound.ts +0 -5
  287. package/src/transport/bot-webhook/message-shape.ts +0 -89
  288. package/src/transport/bot-webhook/protocol.ts +0 -148
  289. package/src/transport/bot-webhook/reply.ts +0 -15
  290. package/src/transport/bot-webhook/request-handler.ts +0 -394
  291. package/src/transport/bot-webhook/session.ts +0 -23
  292. package/src/transport/bot-ws/inbound.test.ts +0 -96
  293. package/src/transport/bot-ws/inbound.ts +0 -116
  294. package/src/transport/bot-ws/media.test.ts +0 -44
  295. package/src/transport/bot-ws/media.ts +0 -321
  296. package/src/transport/bot-ws/reply.test.ts +0 -450
  297. package/src/transport/bot-ws/reply.ts +0 -365
  298. package/src/transport/bot-ws/sdk-adapter.test.ts +0 -187
  299. package/src/transport/bot-ws/sdk-adapter.ts +0 -314
  300. package/src/transport/bot-ws/session.ts +0 -28
  301. package/src/transport/http/common.ts +0 -109
  302. package/src/transport/http/registry.ts +0 -92
  303. package/src/transport/http/request-handler.ts +0 -84
  304. package/src/types/account.ts +0 -72
  305. package/src/types/config.ts +0 -166
  306. package/src/types/constants.ts +0 -31
  307. package/src/types/events.ts +0 -21
  308. package/src/types/global.d.ts +0 -9
  309. package/src/types/index.ts +0 -17
  310. package/src/types/legacy-stream.ts +0 -50
  311. package/src/types/message.ts +0 -187
  312. package/src/types/runtime-context.ts +0 -28
  313. package/src/types/runtime.ts +0 -165
  314. package/src/types.ts +0 -41
  315. package/src/upstream/index.ts +0 -150
  316. package/src/upstream.test.ts +0 -84
  317. package/src/wecom_msg_adapter/markdown_adapter.ts +0 -331
  318. package/tsconfig.json +0 -22
  319. package/vitest.config.ts +0 -26
  320. /package/{src/capability/agent/index.ts → dist/src/capability/agent/index.js} +0 -0
  321. /package/{src/capability/bot/index.ts → dist/src/capability/bot/index.js} +0 -0
  322. /package/{src/capability/calendar/index.ts → dist/src/capability/calendar/index.js} +0 -0
  323. /package/{src/capability/index.ts → dist/src/capability/index.js} +0 -0
@@ -0,0 +1,321 @@
1
+ import crypto from "node:crypto";
2
+ import { wecomFetch } from "../../http.js";
3
+ import { LIMITS } from "../../monitor/state.js";
4
+ import { getActiveReplyUrl, useActiveReplyOnce } from "../../transport/bot-webhook/active-reply.js";
5
+ import { extractLocalImagePathsFromText } from "./fallback-delivery.js";
6
+ import { appendDmContent, buildFallbackPrompt, resolveAgentAccountOrUndefined, sendAgentDmMedia, sendBotFallbackPromptNow, } from "./fallback-delivery.js";
7
+ const STREAM_MAX_BYTES = LIMITS.STREAM_MAX_BYTES;
8
+ const BOT_WINDOW_MS = 6 * 60 * 1000;
9
+ const BOT_SWITCH_MARGIN_MS = 30 * 1000;
10
+ export function createBotReplyDispatcher(params) {
11
+ const { streamStore, target, accountId, config, msg, streamId, rawBody, chatType, userId, core, tableMode, logVerbose, truncateUtf8Bytes, recordBotOperationalEvent, } = params;
12
+ return {
13
+ deliver: async (payload, info) => {
14
+ let text = payload.text ?? "";
15
+ const thinkRegex = /<think>([\s\S]*?)<\/think>/g;
16
+ const thinks = [];
17
+ text = text.replace(thinkRegex, (match) => {
18
+ thinks.push(match);
19
+ return `__THINK_PLACEHOLDER_${thinks.length - 1}__`;
20
+ });
21
+ const trimmedText = text.trim();
22
+ if (trimmedText.startsWith("{") && trimmedText.includes('"template_card"')) {
23
+ try {
24
+ const parsed = JSON.parse(trimmedText);
25
+ if (parsed.template_card) {
26
+ const isSingleChat = msg.chattype !== "group";
27
+ const responseUrl = getActiveReplyUrl(streamId);
28
+ if (responseUrl && isSingleChat) {
29
+ await useActiveReplyOnce(streamId, async ({ responseUrl, proxyUrl }) => {
30
+ const res = await wecomFetch(responseUrl, {
31
+ method: "POST",
32
+ headers: { "Content-Type": "application/json" },
33
+ body: JSON.stringify({
34
+ msgtype: "template_card",
35
+ template_card: parsed.template_card,
36
+ }),
37
+ }, { proxyUrl, timeoutMs: LIMITS.REQUEST_TIMEOUT_MS });
38
+ if (!res.ok) {
39
+ throw new Error(`template_card send failed: ${res.status}`);
40
+ }
41
+ });
42
+ logVerbose(target, `sent template_card: task_id=${parsed.template_card.task_id}`);
43
+ streamStore.updateStream(streamId, (s) => {
44
+ s.finished = true;
45
+ s.content = "[已发送交互卡片]";
46
+ });
47
+ target.touchTransportSession?.({ lastOutboundAt: Date.now(), running: true });
48
+ return;
49
+ }
50
+ logVerbose(target, `template_card fallback to text (group=${!isSingleChat}, hasUrl=${!!responseUrl})`);
51
+ const cardTitle = parsed.template_card.main_title?.title || "交互卡片";
52
+ const cardDesc = parsed.template_card.main_title?.desc || "";
53
+ const buttons = parsed.template_card.button_list?.map((b) => b.text).join(" / ") || "";
54
+ text = `📋 **${cardTitle}**${cardDesc ? `\n${cardDesc}` : ""}${buttons ? `\n\n选项: ${buttons}` : ""}`;
55
+ }
56
+ }
57
+ catch {
58
+ // keep normal text path
59
+ }
60
+ }
61
+ text = core.channel.text.convertMarkdownTables(text, tableMode);
62
+ thinks.forEach((think, i) => {
63
+ text = text.replace(`__THINK_PLACEHOLDER_${i}__`, think);
64
+ });
65
+ const current = streamStore.getStream(streamId);
66
+ if (!current)
67
+ return;
68
+ if (!current.images)
69
+ current.images = [];
70
+ if (!current.agentMediaKeys)
71
+ current.agentMediaKeys = [];
72
+ const deliverKind = info?.kind ?? "block";
73
+ logVerbose(target, `deliver: kind=${deliverKind} chatType=${current.chatType ?? chatType} user=${current.userId ?? userId} textLen=${text.length} mediaCount=${(payload.mediaUrls?.length ?? 0) + (payload.mediaUrl ? 1 : 0)}`);
74
+ if (!payload.mediaUrl && !(payload.mediaUrls?.length ?? 0) && text.includes("/")) {
75
+ const candidates = extractLocalImagePathsFromText({ text, mustAlsoAppearIn: rawBody });
76
+ if (candidates.length > 0) {
77
+ logVerbose(target, `media: 从输出文本推断到本机图片路径(来自用户原消息)count=${candidates.length}`);
78
+ for (const p of candidates) {
79
+ try {
80
+ const fs = await import("node:fs/promises");
81
+ const pathModule = await import("node:path");
82
+ const buf = await fs.readFile(p);
83
+ const ext = pathModule.extname(p).slice(1).toLowerCase();
84
+ const imageExts = {
85
+ jpg: "image/jpeg",
86
+ jpeg: "image/jpeg",
87
+ png: "image/png",
88
+ gif: "image/gif",
89
+ webp: "image/webp",
90
+ bmp: "image/bmp",
91
+ };
92
+ const contentType = imageExts[ext] ?? "application/octet-stream";
93
+ if (!contentType.startsWith("image/"))
94
+ continue;
95
+ const base64 = buf.toString("base64");
96
+ const md5 = crypto.createHash("md5").update(buf).digest("hex");
97
+ current.images.push({ base64, md5 });
98
+ logVerbose(target, `media: 已加载本机图片用于 Bot 交付 path=${p}`);
99
+ }
100
+ catch (err) {
101
+ target.runtime.error?.(`media: 读取本机图片失败 path=${p}: ${String(err)}`);
102
+ }
103
+ }
104
+ }
105
+ }
106
+ if (text.trim()) {
107
+ streamStore.updateStream(streamId, (s) => {
108
+ appendDmContent(s, text);
109
+ });
110
+ }
111
+ const now = Date.now();
112
+ const deadline = current.createdAt + BOT_WINDOW_MS;
113
+ const switchAt = deadline - BOT_SWITCH_MARGIN_MS;
114
+ const nearTimeout = !current.fallbackMode && !current.finished && now >= switchAt;
115
+ if (nearTimeout) {
116
+ const agentCfg = resolveAgentAccountOrUndefined(config, accountId);
117
+ const agentOk = Boolean(agentCfg);
118
+ const prompt = buildFallbackPrompt({
119
+ kind: "timeout",
120
+ agentConfigured: agentOk,
121
+ userId: current.userId,
122
+ chatType: current.chatType,
123
+ });
124
+ logVerbose(target, `fallback(timeout): 触发切换(接近 6 分钟)chatType=${current.chatType} agentConfigured=${agentOk} hasResponseUrl=${Boolean(getActiveReplyUrl(streamId))}`);
125
+ streamStore.updateStream(streamId, (s) => {
126
+ s.fallbackMode = "timeout";
127
+ s.finished = true;
128
+ s.content = prompt;
129
+ s.fallbackPromptSentAt = s.fallbackPromptSentAt ?? Date.now();
130
+ });
131
+ try {
132
+ await sendBotFallbackPromptNow({ streamId, text: prompt });
133
+ logVerbose(target, "fallback(timeout): 群内提示已推送");
134
+ }
135
+ catch (err) {
136
+ target.runtime.error?.(`wecom bot fallback prompt push failed (timeout) streamId=${streamId}: ${String(err)}`);
137
+ recordBotOperationalEvent(target, {
138
+ category: "fallback-delivery-failed",
139
+ summary: `timeout prompt push failed streamId=${streamId}`,
140
+ error: err instanceof Error ? err.message : String(err),
141
+ });
142
+ }
143
+ return;
144
+ }
145
+ const mediaUrls = payload.mediaUrls || (payload.mediaUrl ? [payload.mediaUrl] : []);
146
+ for (const mediaPath of mediaUrls) {
147
+ let contentType;
148
+ let filename = mediaPath.split("/").pop() || "attachment";
149
+ try {
150
+ let buf;
151
+ const looksLikeUrl = /^https?:\/\//i.test(mediaPath);
152
+ if (looksLikeUrl) {
153
+ const loaded = await core.channel.media.fetchRemoteMedia({ url: mediaPath });
154
+ buf = loaded.buffer;
155
+ contentType = loaded.contentType;
156
+ filename = loaded.fileName ?? "attachment";
157
+ }
158
+ else {
159
+ const fs = await import("node:fs/promises");
160
+ const pathModule = await import("node:path");
161
+ buf = await fs.readFile(mediaPath);
162
+ filename = pathModule.basename(mediaPath);
163
+ const ext = pathModule.extname(mediaPath).slice(1).toLowerCase();
164
+ const imageExts = {
165
+ jpg: "image/jpeg",
166
+ jpeg: "image/jpeg",
167
+ png: "image/png",
168
+ gif: "image/gif",
169
+ webp: "image/webp",
170
+ bmp: "image/bmp",
171
+ };
172
+ contentType = imageExts[ext] ?? "application/octet-stream";
173
+ }
174
+ if (contentType?.startsWith("image/")) {
175
+ const base64 = buf.toString("base64");
176
+ const md5 = crypto.createHash("md5").update(buf).digest("hex");
177
+ current.images.push({ base64, md5 });
178
+ logVerbose(target, `media: 识别为图片 contentType=${contentType} filename=${filename}`);
179
+ }
180
+ else {
181
+ const agentCfg = resolveAgentAccountOrUndefined(config, accountId);
182
+ const agentOk = Boolean(agentCfg);
183
+ const alreadySent = current.agentMediaKeys.includes(mediaPath);
184
+ logVerbose(target, `fallback(media): 检测到非图片文件 chatType=${current.chatType} contentType=${contentType ?? "unknown"} filename=${filename} agentConfigured=${agentOk} alreadySent=${alreadySent} hasResponseUrl=${Boolean(getActiveReplyUrl(streamId))}`);
185
+ if (agentCfg && !alreadySent && current.userId) {
186
+ try {
187
+ await sendAgentDmMedia({
188
+ agent: agentCfg,
189
+ userId: current.userId,
190
+ mediaUrlOrPath: mediaPath,
191
+ contentType,
192
+ filename,
193
+ });
194
+ logVerbose(target, `fallback(media): 文件已通过 Agent 私信发送 user=${current.userId}`);
195
+ streamStore.updateStream(streamId, (s) => {
196
+ s.agentMediaKeys = Array.from(new Set([...(s.agentMediaKeys ?? []), mediaPath]));
197
+ });
198
+ }
199
+ catch (err) {
200
+ target.runtime.error?.(`wecom agent dm media failed: ${String(err)}`);
201
+ recordBotOperationalEvent(target, {
202
+ category: "fallback-delivery-failed",
203
+ summary: `agent media dm failed streamId=${streamId} filename=${filename}`,
204
+ error: err instanceof Error ? err.message : String(err),
205
+ });
206
+ }
207
+ }
208
+ if (!current.fallbackMode) {
209
+ const prompt = buildFallbackPrompt({
210
+ kind: "media",
211
+ agentConfigured: agentOk,
212
+ userId: current.userId,
213
+ filename,
214
+ chatType: current.chatType,
215
+ });
216
+ streamStore.updateStream(streamId, (s) => {
217
+ s.fallbackMode = "media";
218
+ s.finished = true;
219
+ s.content = prompt;
220
+ s.fallbackPromptSentAt = s.fallbackPromptSentAt ?? Date.now();
221
+ });
222
+ try {
223
+ await sendBotFallbackPromptNow({ streamId, text: prompt });
224
+ logVerbose(target, "fallback(media): 群内提示已推送");
225
+ }
226
+ catch (err) {
227
+ target.runtime.error?.(`wecom bot fallback prompt push failed (media) streamId=${streamId}: ${String(err)}`);
228
+ recordBotOperationalEvent(target, {
229
+ category: "fallback-delivery-failed",
230
+ summary: `media prompt push failed streamId=${streamId} filename=${filename}`,
231
+ error: err instanceof Error ? err.message : String(err),
232
+ });
233
+ }
234
+ }
235
+ return;
236
+ }
237
+ }
238
+ catch (err) {
239
+ target.runtime.error?.(`Failed to process outbound media: ${mediaPath}: ${String(err)}`);
240
+ recordBotOperationalEvent(target, {
241
+ category: "fallback-delivery-failed",
242
+ summary: `outbound media processing failed streamId=${streamId} media=${mediaPath}`,
243
+ error: err instanceof Error ? err.message : String(err),
244
+ });
245
+ const agentCfg = resolveAgentAccountOrUndefined(config, accountId);
246
+ const agentOk = Boolean(agentCfg);
247
+ const fallbackFilename = filename || mediaPath.split("/").pop() || "attachment";
248
+ if (agentCfg && current.userId && !current.agentMediaKeys.includes(mediaPath)) {
249
+ try {
250
+ await sendAgentDmMedia({
251
+ agent: agentCfg,
252
+ userId: current.userId,
253
+ mediaUrlOrPath: mediaPath,
254
+ contentType,
255
+ filename: fallbackFilename,
256
+ });
257
+ streamStore.updateStream(streamId, (s) => {
258
+ s.agentMediaKeys = Array.from(new Set([...(s.agentMediaKeys ?? []), mediaPath]));
259
+ });
260
+ logVerbose(target, `fallback(error): 媒体处理失败后已通过 Agent 私信发送 user=${current.userId}`);
261
+ }
262
+ catch (sendErr) {
263
+ target.runtime.error?.(`fallback(error): 媒体处理失败后的 Agent 私信发送也失败: ${String(sendErr)}`);
264
+ recordBotOperationalEvent(target, {
265
+ category: "fallback-delivery-failed",
266
+ summary: `fallback error dm failed streamId=${streamId} filename=${fallbackFilename}`,
267
+ error: sendErr instanceof Error ? sendErr.message : String(sendErr),
268
+ });
269
+ }
270
+ }
271
+ if (!current.fallbackMode) {
272
+ const prompt = buildFallbackPrompt({
273
+ kind: "error",
274
+ agentConfigured: agentOk,
275
+ userId: current.userId,
276
+ filename: fallbackFilename,
277
+ chatType: current.chatType,
278
+ });
279
+ streamStore.updateStream(streamId, (s) => {
280
+ s.fallbackMode = "error";
281
+ s.finished = true;
282
+ s.content = prompt;
283
+ s.fallbackPromptSentAt = s.fallbackPromptSentAt ?? Date.now();
284
+ });
285
+ try {
286
+ await sendBotFallbackPromptNow({ streamId, text: prompt });
287
+ logVerbose(target, "fallback(error): 群内提示已推送");
288
+ }
289
+ catch (pushErr) {
290
+ target.runtime.error?.(`wecom bot fallback prompt push failed (error) streamId=${streamId}: ${String(pushErr)}`);
291
+ recordBotOperationalEvent(target, {
292
+ category: "fallback-delivery-failed",
293
+ summary: `fallback error prompt push failed streamId=${streamId} filename=${fallbackFilename}`,
294
+ error: pushErr instanceof Error ? pushErr.message : String(pushErr),
295
+ });
296
+ }
297
+ }
298
+ return;
299
+ }
300
+ }
301
+ const mode = streamStore.getStream(streamId)?.fallbackMode;
302
+ if (mode)
303
+ return;
304
+ const nextText = current.content ? `${current.content}\n\n${text}`.trim() : text.trim();
305
+ streamStore.updateStream(streamId, (s) => {
306
+ s.content = truncateUtf8Bytes(nextText, STREAM_MAX_BYTES);
307
+ if (current.images?.length)
308
+ s.images = current.images;
309
+ });
310
+ target.touchTransportSession?.({ lastOutboundAt: Date.now(), running: true });
311
+ },
312
+ onError: (err, info) => {
313
+ target.runtime.error?.(`[${accountId}] wecom ${info.kind} reply failed: ${String(err)}`);
314
+ recordBotOperationalEvent(target, {
315
+ category: "fallback-delivery-failed",
316
+ summary: `reply dispatcher failed kind=${info.kind} streamId=${streamId}`,
317
+ error: err instanceof Error ? err.message : String(err),
318
+ });
319
+ },
320
+ };
321
+ }
@@ -0,0 +1,81 @@
1
+ import { getActiveReplyUrl } from "../../transport/bot-webhook/active-reply.js";
2
+ import { pushFinalStreamReplyNow, resolveAgentAccountOrUndefined, sendAgentDmText } from "./fallback-delivery.js";
3
+ export async function finalizeBotStream(params) {
4
+ const { streamStore, target, streamId, chatType, core, config, accountId, isResetCommand, resetCommandKind, logInfo, logVerbose, recordBotOperationalEvent, } = params;
5
+ if (isResetCommand) {
6
+ const current = streamStore.getStream(streamId);
7
+ const hasAnyContent = Boolean(current?.content?.trim());
8
+ if (current && !hasAnyContent) {
9
+ const ackText = resetCommandKind === "reset" ? "✅ 已重置会话。" : "✅ 已开启新会话。";
10
+ streamStore.updateStream(streamId, (s) => {
11
+ s.content = ackText;
12
+ s.finished = true;
13
+ });
14
+ }
15
+ }
16
+ streamStore.updateStream(streamId, (s) => {
17
+ if (!s.content.trim() && !(s.images?.length ?? 0)) {
18
+ s.content = "✅ 已处理完成。";
19
+ }
20
+ });
21
+ streamStore.markFinished(streamId);
22
+ const finishedState = streamStore.getStream(streamId);
23
+ if (finishedState?.fallbackMode === "timeout" && !finishedState.finalDeliveredAt) {
24
+ const agentCfg = resolveAgentAccountOrUndefined(config, accountId);
25
+ if (!agentCfg) {
26
+ streamStore.updateStream(streamId, (s) => {
27
+ s.finalDeliveredAt = Date.now();
28
+ });
29
+ }
30
+ else if (finishedState.userId) {
31
+ const dmText = (finishedState.dmContent ?? "").trim();
32
+ if (dmText) {
33
+ try {
34
+ logVerbose(target, `fallback(timeout): 开始通过 Agent 私信发送剩余内容 user=${finishedState.userId} len=${dmText.length}`);
35
+ await sendAgentDmText({ agent: agentCfg, userId: finishedState.userId, text: dmText, core });
36
+ logVerbose(target, `fallback(timeout): Agent 私信发送完成 user=${finishedState.userId}`);
37
+ }
38
+ catch (err) {
39
+ target.runtime.error?.(`wecom agent dm text failed (timeout): ${String(err)}`);
40
+ recordBotOperationalEvent(target, {
41
+ category: "fallback-delivery-failed",
42
+ summary: `timeout final dm failed streamId=${streamId} user=${finishedState.userId}`,
43
+ error: err instanceof Error ? err.message : String(err),
44
+ });
45
+ }
46
+ }
47
+ streamStore.updateStream(streamId, (s) => {
48
+ s.finalDeliveredAt = Date.now();
49
+ });
50
+ }
51
+ }
52
+ const stateAfterFinish = streamStore.getStream(streamId);
53
+ const responseUrl = getActiveReplyUrl(streamId);
54
+ if (stateAfterFinish && responseUrl) {
55
+ try {
56
+ await pushFinalStreamReplyNow({ streamId, state: stateAfterFinish });
57
+ logVerbose(target, `final stream pushed via response_url streamId=${streamId}, chatType=${chatType}, images=${stateAfterFinish.images?.length ?? 0}`);
58
+ }
59
+ catch (err) {
60
+ target.runtime.error?.(`final stream push via response_url failed streamId=${streamId}: ${String(err)}`);
61
+ recordBotOperationalEvent(target, {
62
+ category: "fallback-delivery-failed",
63
+ summary: `final stream push failed streamId=${streamId}`,
64
+ error: err instanceof Error ? err.message : String(err),
65
+ });
66
+ }
67
+ }
68
+ logInfo(target, `queue: 当前批次结束,尝试推进下一批 streamId=${streamId}`);
69
+ const ackStreamIds = streamStore.drainAckStreamsForBatch(streamId);
70
+ if (ackStreamIds.length > 0) {
71
+ const mergedDoneHint = "✅ 已合并处理完成,请查看上一条回复。";
72
+ for (const ackId of ackStreamIds) {
73
+ streamStore.updateStream(ackId, (s) => {
74
+ s.content = mergedDoneHint;
75
+ s.finished = true;
76
+ });
77
+ }
78
+ logInfo(target, `queue: 已更新回执流 count=${ackStreamIds.length} batchStreamId=${streamId}`);
79
+ }
80
+ streamStore.onStreamFinished(streamId);
81
+ }