@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
@@ -1,450 +0,0 @@
1
- import os from "node:os";
2
- import path from "node:path";
3
- import type { WSClient } from "@wecom/aibot-node-sdk";
4
- import { describe, expect, it, vi, beforeEach, afterEach } from "vitest";
5
- import { resolvePreferredOpenClawTmpDir } from "openclaw/plugin-sdk/infra-runtime";
6
- import { uploadAndSendBotWsMedia } from "./media.js";
7
- import { createBotWsReplyHandle } from "./reply.js";
8
-
9
- vi.mock("./media.js", () => ({
10
- uploadAndSendBotWsMedia: vi.fn(),
11
- }));
12
-
13
- type ReplyHandleParams = Parameters<typeof createBotWsReplyHandle>[0];
14
-
15
- describe("createBotWsReplyHandle", () => {
16
- let mockClient: import("vitest").Mocked<WSClient>;
17
- const uploadAndSendBotWsMediaMock = vi.mocked(uploadAndSendBotWsMedia);
18
-
19
- beforeEach(async () => {
20
- vi.useFakeTimers();
21
- vi.stubEnv("OPENCLAW_STATE_DIR", "/tmp/wecom-reply-state");
22
- mockClient = {
23
- replyStream: vi.fn(),
24
- sendMessage: vi.fn(),
25
- replyWelcome: vi.fn(),
26
- } as unknown as import("vitest").Mocked<WSClient>;
27
- mockClient.replyStream.mockResolvedValue({} as any);
28
- mockClient.sendMessage.mockResolvedValue({} as any);
29
- mockClient.replyWelcome.mockResolvedValue({} as any);
30
- uploadAndSendBotWsMediaMock.mockReset();
31
- uploadAndSendBotWsMediaMock.mockResolvedValue({ ok: true, messageId: "media-1" } as any);
32
- const runtime = await import("../../runtime.js");
33
- runtime.setWecomRuntime({
34
- config: {
35
- loadConfig: () => ({
36
- channels: {
37
- wecom: {},
38
- },
39
- }),
40
- },
41
- } as any);
42
- });
43
-
44
- afterEach(() => {
45
- vi.clearAllTimers();
46
- vi.useRealTimers();
47
- vi.restoreAllMocks();
48
- vi.unstubAllEnvs();
49
- });
50
-
51
- it("uses configured placeholder content for immediate ws ack", async () => {
52
- createBotWsReplyHandle({
53
- client: mockClient,
54
- frame: {
55
- headers: { req_id: "req-1" },
56
- body: { chatid: "123", chattype: "group" },
57
- cmd: "aibot_msg_callback",
58
- } as unknown as ReplyHandleParams["frame"],
59
- accountId: "default",
60
- inboundKind: "text",
61
- placeholderContent: "正在思考...",
62
- });
63
-
64
- vi.advanceTimersByTime(3000);
65
- // Let promises flush
66
- await Promise.resolve();
67
-
68
- expect(mockClient.replyStream).toHaveBeenCalledWith(
69
- expect.objectContaining({
70
- headers: { req_id: "req-1" },
71
- }),
72
- expect.any(String),
73
- "正在思考...",
74
- false,
75
- );
76
- });
77
-
78
- it("keeps placeholder alive until the first real ws chunk arrives", async () => {
79
- const handle = createBotWsReplyHandle({
80
- client: mockClient,
81
- frame: {
82
- headers: { req_id: "req-keepalive" },
83
- body: {},
84
- } as unknown as ReplyHandleParams["frame"],
85
- accountId: "default",
86
- inboundKind: "text",
87
- placeholderContent: "正在思考...",
88
- });
89
-
90
- vi.advanceTimersByTime(3000);
91
- // Flush the microtasks so `placeholderInFlight` becomes false
92
- for (let i = 0; i < 10; i++) await Promise.resolve();
93
-
94
- // Now trigger the next timer
95
- vi.advanceTimersByTime(3000);
96
- for (let i = 0; i < 10; i++) await Promise.resolve();
97
- expect(mockClient.replyStream).toHaveBeenCalledTimes(2);
98
-
99
- handle.deliver({ text: "最终回复", isReasoning: false }, { kind: "final" });
100
- await Promise.resolve();
101
-
102
- expect(mockClient.replyStream).toHaveBeenCalledWith(
103
- expect.objectContaining({
104
- headers: { req_id: "req-keepalive" },
105
- }),
106
- expect.any(String),
107
- "最终回复",
108
- true,
109
- );
110
-
111
- // Ensure interval is cleared
112
- vi.advanceTimersByTime(6000);
113
- await Promise.resolve();
114
- expect(mockClient.replyStream).toHaveBeenCalledTimes(3);
115
- });
116
-
117
- it("does not auto-send placeholder when disabled", async () => {
118
- createBotWsReplyHandle({
119
- client: mockClient,
120
- frame: {
121
- headers: { req_id: "req-2" },
122
- body: {},
123
- } as unknown as ReplyHandleParams["frame"],
124
- accountId: "default",
125
- inboundKind: "text",
126
- autoSendPlaceholder: false,
127
- });
128
-
129
- vi.advanceTimersByTime(3000);
130
- await Promise.resolve();
131
- expect(mockClient.replyStream).not.toHaveBeenCalled();
132
- });
133
-
134
- it("sends cumulative content for block streaming updates", async () => {
135
- const handle = createBotWsReplyHandle({
136
- client: mockClient,
137
- frame: {
138
- headers: { req_id: "req-blocks" },
139
- body: {},
140
- } as unknown as ReplyHandleParams["frame"],
141
- accountId: "default",
142
- inboundKind: "text",
143
- autoSendPlaceholder: false,
144
- });
145
-
146
- await handle.deliver({ text: "第一段", isReasoning: false }, { kind: "block" });
147
- await handle.deliver({ text: "第二段", isReasoning: false }, { kind: "block" });
148
- await handle.deliver({ text: "收尾", isReasoning: false }, { kind: "final" });
149
-
150
- expect(mockClient.replyStream).toHaveBeenNthCalledWith(
151
- 1,
152
- expect.objectContaining({ headers: { req_id: "req-blocks" } }),
153
- expect.any(String),
154
- "第一段",
155
- false,
156
- );
157
- expect(mockClient.replyStream).toHaveBeenNthCalledWith(
158
- 2,
159
- expect.objectContaining({ headers: { req_id: "req-blocks" } }),
160
- expect.any(String),
161
- "第一段\n第二段",
162
- false,
163
- );
164
- expect(mockClient.replyStream).toHaveBeenNthCalledWith(
165
- 3,
166
- expect.objectContaining({ headers: { req_id: "req-blocks" } }),
167
- expect.any(String),
168
- "第一段\n第二段\n收尾",
169
- true,
170
- );
171
- });
172
-
173
- it("streams block text even when media is deferred to final", async () => {
174
- const handle = createBotWsReplyHandle({
175
- client: mockClient,
176
- frame: {
177
- headers: { req_id: "req-block-media" },
178
- body: {},
179
- } as unknown as ReplyHandleParams["frame"],
180
- accountId: "default",
181
- inboundKind: "text",
182
- autoSendPlaceholder: false,
183
- });
184
-
185
- await handle.deliver(
186
- {
187
- text: "正文先发",
188
- mediaUrls: ["/tmp/a.png", "/tmp/b.png"],
189
- isReasoning: false,
190
- },
191
- { kind: "block" },
192
- );
193
-
194
- expect(mockClient.replyStream).toHaveBeenCalledWith(
195
- expect.objectContaining({ headers: { req_id: "req-block-media" } }),
196
- expect.any(String),
197
- "正文先发",
198
- false,
199
- );
200
- });
201
-
202
- it("includes default global media local roots for final media sends", async () => {
203
- const runtime = await import("../../runtime.js");
204
- runtime.setWecomRuntime({
205
- config: {
206
- loadConfig: () => ({}),
207
- },
208
- } as any);
209
-
210
- const handle = createBotWsReplyHandle({
211
- client: mockClient,
212
- frame: {
213
- headers: { req_id: "req-final-media-roots" },
214
- body: {
215
- from: { userid: "hidao" },
216
- chattype: "single",
217
- },
218
- } as unknown as ReplyHandleParams["frame"],
219
- accountId: "default",
220
- inboundKind: "text",
221
- autoSendPlaceholder: false,
222
- });
223
-
224
- await handle.deliver(
225
- {
226
- mediaUrls: ["/Users/YanHaidao/Downloads/01.png"],
227
- isReasoning: false,
228
- },
229
- { kind: "final" },
230
- );
231
-
232
- expect(uploadAndSendBotWsMediaMock).toHaveBeenCalledWith(
233
- expect.objectContaining({
234
- chatId: "hidao",
235
- maxBytes: 80 * 1024 * 1024,
236
- mediaUrl: "/Users/YanHaidao/Downloads/01.png",
237
- mediaLocalRoots: expect.arrayContaining([
238
- path.resolve(resolvePreferredOpenClawTmpDir()),
239
- "/tmp/wecom-reply-state",
240
- "/tmp/wecom-reply-state/media",
241
- path.resolve(os.homedir(), "Desktop"),
242
- path.resolve(os.homedir(), "Documents"),
243
- path.resolve(os.homedir(), "Downloads"),
244
- ]),
245
- }),
246
- );
247
- expect(mockClient.replyStream).toHaveBeenCalledWith(
248
- expect.objectContaining({ headers: { req_id: "req-final-media-roots" } }),
249
- expect.any(String),
250
- "文件已发送。",
251
- true,
252
- );
253
- });
254
-
255
- it("passes configured mediaMaxMb to final media sends", async () => {
256
- const runtime = await import("../../runtime.js");
257
- runtime.setWecomRuntime({
258
- config: {
259
- loadConfig: () => ({
260
- agents: {
261
- defaults: {
262
- mediaMaxMb: 12,
263
- },
264
- },
265
- channels: {
266
- wecom: {
267
- mediaMaxMb: 24,
268
- accounts: {
269
- default: {
270
- mediaMaxMb: 40,
271
- },
272
- },
273
- },
274
- },
275
- }),
276
- },
277
- } as any);
278
-
279
- const handle = createBotWsReplyHandle({
280
- client: mockClient,
281
- frame: {
282
- headers: { req_id: "req-final-media-max-bytes" },
283
- body: {
284
- from: { userid: "hidao" },
285
- chattype: "single",
286
- },
287
- } as unknown as ReplyHandleParams["frame"],
288
- accountId: "default",
289
- inboundKind: "text",
290
- autoSendPlaceholder: false,
291
- });
292
-
293
- await handle.deliver(
294
- {
295
- mediaUrls: ["/Users/YanHaidao/Downloads/01.png"],
296
- isReasoning: false,
297
- },
298
- { kind: "final" },
299
- );
300
-
301
- expect(uploadAndSendBotWsMediaMock).toHaveBeenCalledWith(
302
- expect.objectContaining({
303
- chatId: "hidao",
304
- maxBytes: 40 * 1024 * 1024,
305
- }),
306
- );
307
- });
308
-
309
- it("stops placeholder keepalive when the first block contains media", async () => {
310
- const handle = createBotWsReplyHandle({
311
- client: mockClient,
312
- frame: {
313
- headers: { req_id: "req-placeholder-media" },
314
- body: {},
315
- } as unknown as ReplyHandleParams["frame"],
316
- accountId: "default",
317
- inboundKind: "text",
318
- placeholderContent: "正在思考...",
319
- });
320
-
321
- vi.advanceTimersByTime(3000);
322
- for (let i = 0; i < 10; i++) await Promise.resolve();
323
- expect(mockClient.replyStream).toHaveBeenCalledTimes(1);
324
-
325
- await handle.deliver(
326
- {
327
- text: "正文先发",
328
- mediaUrls: ["/tmp/a.png"],
329
- isReasoning: false,
330
- },
331
- { kind: "block" },
332
- );
333
-
334
- vi.advanceTimersByTime(6000);
335
- for (let i = 0; i < 10; i++) await Promise.resolve();
336
-
337
- expect(mockClient.replyStream).toHaveBeenCalledTimes(2);
338
- expect(mockClient.replyStream).toHaveBeenNthCalledWith(
339
- 2,
340
- expect.objectContaining({ headers: { req_id: "req-placeholder-media" } }),
341
- expect.any(String),
342
- "正文先发",
343
- false,
344
- );
345
- });
346
-
347
- it("swallows expired stream update errors during delivery", async () => {
348
- const expiredError = {
349
- headers: { req_id: "req-expired" },
350
- errcode: 846608,
351
- errmsg: "stream message update expired (>6 minutes), cannot update",
352
- };
353
- mockClient.replyStream.mockRejectedValueOnce(expiredError);
354
- const onFail = vi.fn();
355
-
356
- const handle = createBotWsReplyHandle({
357
- client: mockClient,
358
- frame: {
359
- headers: { req_id: "req-expired" },
360
- body: {},
361
- } as unknown as ReplyHandleParams["frame"],
362
- accountId: "default",
363
- inboundKind: "text",
364
- autoSendPlaceholder: false,
365
- onFail,
366
- });
367
-
368
- await handle.deliver({ text: "最终回复", isReasoning: false }, { kind: "final" });
369
-
370
- expect(mockClient.replyStream).toHaveBeenCalledTimes(1);
371
- expect(onFail).toHaveBeenCalledWith(expiredError);
372
- });
373
-
374
- it.each([
375
- [{ headers: { req_id: "req-invalid" }, errcode: 846605, errmsg: "invalid req_id" }],
376
- [
377
- {
378
- headers: { req_id: "req-expired" },
379
- errcode: 846608,
380
- errmsg: "stream message update expired (>6 minutes), cannot update",
381
- },
382
- ],
383
- ])("does not retry error reply when the ws reply window is already closed", async (error) => {
384
- const onFail = vi.fn();
385
- const handle = createBotWsReplyHandle({
386
- client: mockClient,
387
- frame: {
388
- headers: { req_id: String(error.headers.req_id) },
389
- body: {},
390
- } as unknown as ReplyHandleParams["frame"],
391
- accountId: "default",
392
- inboundKind: "text",
393
- autoSendPlaceholder: false,
394
- onFail,
395
- });
396
-
397
- await handle.fail?.(error);
398
-
399
- expect(mockClient.replyStream).not.toHaveBeenCalled();
400
- expect(onFail).toHaveBeenCalledTimes(1);
401
- });
402
-
403
- it("sends simple fallback message for ordinary events without placeholders", async () => {
404
- const handle = createBotWsReplyHandle({
405
- client: mockClient,
406
- frame: {
407
- headers: { req_id: "event_req" },
408
- body: { chattype: "single", from: { userid: "alice" } },
409
- } as unknown as ReplyHandleParams["frame"],
410
- accountId: "default",
411
- inboundKind: "event",
412
- });
413
-
414
- vi.advanceTimersByTime(3000);
415
- await Promise.resolve();
416
- // Events should not send stream placeholders
417
- expect(mockClient.replyStream).not.toHaveBeenCalled();
418
-
419
- handle.deliver({ text: "Event Reply", isReasoning: false }, { kind: "final" });
420
- await Promise.resolve();
421
-
422
- expect(mockClient.sendMessage).toHaveBeenCalledWith("alice", {
423
- msgtype: "markdown",
424
- markdown: { content: "Event Reply" },
425
- });
426
- });
427
-
428
- it("sends replyWelcome for welcome events", async () => {
429
- const handle = createBotWsReplyHandle({
430
- client: mockClient,
431
- frame: {
432
- headers: { req_id: "welcome_req" },
433
- body: { chattype: "single", from: { userid: "bob" } },
434
- } as unknown as ReplyHandleParams["frame"],
435
- accountId: "default",
436
- inboundKind: "welcome",
437
- });
438
-
439
- handle.deliver({ text: "Hello Bob", isReasoning: false }, { kind: "final" });
440
- await Promise.resolve();
441
-
442
- expect(mockClient.replyWelcome).toHaveBeenCalledWith(
443
- expect.objectContaining({ headers: { req_id: "welcome_req" } }),
444
- {
445
- msgtype: "text",
446
- text: { content: "Hello Bob" },
447
- },
448
- );
449
- });
450
- });