@yanhaidao/wecom 2.4.160 → 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 (313) hide show
  1. package/dist/index.js +68 -0
  2. package/dist/src/accounts.js +20 -0
  3. package/dist/src/agent/handler.js +895 -0
  4. package/dist/src/agent/index.js +5 -0
  5. package/dist/src/app/account-runtime.js +216 -0
  6. package/dist/src/app/bootstrap.js +19 -0
  7. package/dist/src/app/index.js +118 -0
  8. package/dist/src/capability/agent/delivery-service.js +63 -0
  9. package/dist/src/capability/agent/fallback-policy.js +6 -0
  10. package/dist/src/capability/agent/ingress-service.js +33 -0
  11. package/dist/src/capability/agent/upstream-delivery-service.js +71 -0
  12. package/dist/src/capability/bot/dispatch-config.js +45 -0
  13. package/dist/src/capability/bot/fallback-delivery.js +147 -0
  14. package/dist/src/capability/bot/local-path-delivery.js +178 -0
  15. package/dist/src/capability/bot/sandbox-media.js +138 -0
  16. package/dist/src/capability/bot/service.js +49 -0
  17. package/dist/src/capability/bot/stream-delivery.js +321 -0
  18. package/dist/src/capability/bot/stream-finalizer.js +81 -0
  19. package/dist/src/capability/bot/stream-orchestrator.js +318 -0
  20. package/dist/src/capability/bot/types.js +1 -0
  21. package/{src/capability/calendar/client.ts → dist/src/capability/calendar/client.js} +118 -241
  22. package/{src/capability/calendar/schema.ts → dist/src/capability/calendar/schema.js} +0 -38
  23. package/dist/src/capability/calendar/tool.js +365 -0
  24. package/dist/src/capability/calendar/types.js +12 -0
  25. package/{src/capability/doc/client.ts → dist/src/capability/doc/client.js} +370 -605
  26. package/{src/capability/doc/schema.ts → dist/src/capability/doc/schema.js} +345 -394
  27. package/dist/src/capability/doc/tool.js +1556 -0
  28. package/dist/src/capability/doc/types.js +113 -0
  29. package/dist/src/capability/mcp/index.js +3 -0
  30. package/dist/src/capability/mcp/schema.js +102 -0
  31. package/dist/src/capability/mcp/tool.js +146 -0
  32. package/dist/src/capability/mcp/transport.js +293 -0
  33. package/dist/src/channel.js +224 -0
  34. package/dist/src/config/accounts.js +236 -0
  35. package/dist/src/config/derived-paths.js +31 -0
  36. package/dist/src/config/index.js +7 -0
  37. package/dist/src/config/media.js +110 -0
  38. package/dist/src/config/network.js +32 -0
  39. package/dist/src/config/routing.js +20 -0
  40. package/dist/src/config/runtime-config.js +25 -0
  41. package/dist/src/config/schema.js +4 -0
  42. package/{src/config-schema.ts → dist/src/config-schema.js} +1 -1
  43. package/dist/src/context-store.js +219 -0
  44. package/{src/crypto/aes.ts → dist/src/crypto/aes.js} +11 -28
  45. package/dist/src/crypto/index.js +9 -0
  46. package/{src/crypto/signature.ts → dist/src/crypto/signature.js} +3 -18
  47. package/{src/crypto/xml.ts → dist/src/crypto/xml.js} +3 -11
  48. package/dist/src/crypto.js +145 -0
  49. package/dist/src/domain/models.js +1 -0
  50. package/dist/src/domain/policies.js +32 -0
  51. package/{src/dynamic-agent.ts → dist/src/dynamic-agent.js} +36 -73
  52. package/dist/src/gateway-monitor.js +139 -0
  53. package/dist/src/http.js +114 -0
  54. package/{src/media.ts → dist/src/media.js} +21 -40
  55. package/dist/src/monitor/limits.js +7 -0
  56. package/dist/src/monitor/state.js +28 -0
  57. package/dist/src/monitor.js +84 -0
  58. package/dist/src/observability/audit-log.js +30 -0
  59. package/dist/src/observability/legacy-operational-event-store.js +22 -0
  60. package/dist/src/observability/raw-envelope-log.js +24 -0
  61. package/dist/src/observability/status-registry.js +9 -0
  62. package/dist/src/observability/transport-session-view.js +14 -0
  63. package/dist/src/onboarding.js +546 -0
  64. package/dist/src/outbound.js +557 -0
  65. package/dist/src/runtime/dispatcher.js +57 -0
  66. package/{src/runtime/index.ts → dist/src/runtime/index.js} +0 -1
  67. package/dist/src/runtime/outbound-intent.js +1 -0
  68. package/dist/src/runtime/reply-orchestrator.js +38 -0
  69. package/dist/src/runtime/routing-bridge.js +26 -0
  70. package/dist/src/runtime/session-manager.js +112 -0
  71. package/dist/src/runtime/source-registry.js +174 -0
  72. package/dist/src/runtime.js +1 -0
  73. package/dist/src/shared/command-auth.js +57 -0
  74. package/{src/shared/index.ts → dist/src/shared/index.js} +0 -1
  75. package/dist/src/shared/media-asset.js +65 -0
  76. package/dist/src/shared/media-service.js +59 -0
  77. package/dist/src/shared/media-types.js +1 -0
  78. package/{src/shared/xml-parser.ts → dist/src/shared/xml-parser.js} +72 -63
  79. package/dist/src/store/active-reply-store.js +41 -0
  80. package/dist/src/store/interfaces.js +1 -0
  81. package/dist/src/store/memory-store.js +33 -0
  82. package/dist/src/store/stream-batch-store.js +319 -0
  83. package/{src/target.ts → dist/src/target.js} +15 -48
  84. package/dist/src/transport/agent-api/client.js +168 -0
  85. package/dist/src/transport/agent-api/core.js +337 -0
  86. package/dist/src/transport/agent-api/delivery.js +28 -0
  87. package/dist/src/transport/agent-api/media-upload.js +4 -0
  88. package/dist/src/transport/agent-api/reply.js +24 -0
  89. package/dist/src/transport/agent-api/upstream-delivery.js +30 -0
  90. package/dist/src/transport/agent-api/upstream-media-upload.js +46 -0
  91. package/dist/src/transport/agent-api/upstream-reply.js +26 -0
  92. package/dist/src/transport/agent-callback/http-handler.js +30 -0
  93. package/dist/src/transport/agent-callback/inbound.js +4 -0
  94. package/dist/src/transport/agent-callback/reply.js +8 -0
  95. package/dist/src/transport/agent-callback/request-handler.js +189 -0
  96. package/dist/src/transport/agent-callback/session.js +15 -0
  97. package/dist/src/transport/bot-webhook/active-reply.js +27 -0
  98. package/dist/src/transport/bot-webhook/http-handler.js +31 -0
  99. package/dist/src/transport/bot-webhook/inbound-normalizer.js +496 -0
  100. package/dist/src/transport/bot-webhook/inbound.js +4 -0
  101. package/dist/src/transport/bot-webhook/message-shape.js +98 -0
  102. package/dist/src/transport/bot-webhook/protocol.js +124 -0
  103. package/dist/src/transport/bot-webhook/reply.js +9 -0
  104. package/dist/src/transport/bot-webhook/request-handler.js +285 -0
  105. package/dist/src/transport/bot-webhook/session.js +15 -0
  106. package/dist/src/transport/bot-ws/inbound.js +147 -0
  107. package/dist/src/transport/bot-ws/media.js +236 -0
  108. package/dist/src/transport/bot-ws/reply.js +310 -0
  109. package/dist/src/transport/bot-ws/sdk-adapter.js +257 -0
  110. package/dist/src/transport/bot-ws/session.js +15 -0
  111. package/dist/src/transport/http/common.js +78 -0
  112. package/dist/src/transport/http/registry.js +71 -0
  113. package/dist/src/transport/http/request-handler.js +51 -0
  114. package/{src/transport/index.ts → dist/src/transport/index.js} +2 -10
  115. package/dist/src/types/account.js +1 -0
  116. package/dist/src/types/config.js +1 -0
  117. package/dist/src/types/constants.js +28 -0
  118. package/dist/src/types/events.js +1 -0
  119. package/dist/src/types/index.js +1 -0
  120. package/dist/src/types/legacy-stream.js +1 -0
  121. package/dist/src/types/message.js +5 -0
  122. package/dist/src/types/runtime-context.js +1 -0
  123. package/dist/src/types/runtime.js +1 -0
  124. package/dist/src/types.js +1 -0
  125. package/dist/src/upstream/index.js +111 -0
  126. package/dist/src/wecom_msg_adapter/markdown_adapter.js +280 -0
  127. package/openclaw.plugin.json +15 -0
  128. package/package.json +18 -1
  129. package/.github/workflows/release.yml +0 -143
  130. package/GOVERNANCE.md +0 -26
  131. package/SKILLS_CAL.md +0 -895
  132. package/SKILLS_DOC.md +0 -2288
  133. package/UPSTREAM_CONFIG.md +0 -170
  134. package/UPSTREAM_PLAN.md +0 -175
  135. package/assets/01.bot-add.png +0 -0
  136. package/assets/01.bot-setp2.png +0 -0
  137. package/assets/01.image.jpg +0 -0
  138. package/assets/02.agent.add.png +0 -0
  139. package/assets/02.agent.api-set.png +0 -0
  140. package/assets/02.image.jpg +0 -0
  141. package/assets/03.agent.page.png +0 -0
  142. package/assets/03.bot.page.png +0 -0
  143. package/assets/link-me.jpg +0 -0
  144. package/assets/register.png +0 -0
  145. package/changelog/v2.2.28.md +0 -70
  146. package/changelog/v2.3.10.md +0 -17
  147. package/changelog/v2.3.11.md +0 -19
  148. package/changelog/v2.3.12.md +0 -25
  149. package/changelog/v2.3.13.md +0 -19
  150. package/changelog/v2.3.14.md +0 -48
  151. package/changelog/v2.3.15.md +0 -15
  152. package/changelog/v2.3.16.md +0 -11
  153. package/changelog/v2.3.18.md +0 -22
  154. package/changelog/v2.3.19.md +0 -73
  155. package/changelog/v2.3.2.md +0 -28
  156. package/changelog/v2.3.26.md +0 -21
  157. package/changelog/v2.3.27.md +0 -33
  158. package/changelog/v2.3.4.md +0 -20
  159. package/changelog/v2.3.9.md +0 -22
  160. package/changelog/v2.4.12.md +0 -37
  161. package/changelog/v2.4.16.md +0 -19
  162. package/compat-single-account.md +0 -148
  163. package/index.test.ts +0 -38
  164. package/scripts/test-proxy.ts +0 -70
  165. package/src/accounts.ts +0 -34
  166. package/src/agent/api-client.upload.test.ts +0 -109
  167. package/src/agent/handler.event-filter.test.ts +0 -100
  168. package/src/agent/handler.ts +0 -1105
  169. package/src/agent/index.ts +0 -12
  170. package/src/app/account-runtime.ts +0 -276
  171. package/src/app/bootstrap.ts +0 -29
  172. package/src/app/index.ts +0 -192
  173. package/src/capability/agent/delivery-service.ts +0 -87
  174. package/src/capability/agent/fallback-policy.ts +0 -13
  175. package/src/capability/agent/ingress-service.ts +0 -38
  176. package/src/capability/agent/upstream-delivery-service.ts +0 -96
  177. package/src/capability/bot/dispatch-config.ts +0 -47
  178. package/src/capability/bot/fallback-delivery.ts +0 -178
  179. package/src/capability/bot/local-path-delivery.ts +0 -215
  180. package/src/capability/bot/sandbox-media.test.ts +0 -221
  181. package/src/capability/bot/sandbox-media.ts +0 -176
  182. package/src/capability/bot/service.ts +0 -56
  183. package/src/capability/bot/stream-delivery.ts +0 -379
  184. package/src/capability/bot/stream-finalizer.ts +0 -120
  185. package/src/capability/bot/stream-orchestrator.ts +0 -371
  186. package/src/capability/bot/types.ts +0 -8
  187. package/src/capability/calendar/SKILLS_CHECKLIST.md +0 -251
  188. package/src/capability/calendar/tool.ts +0 -417
  189. package/src/capability/calendar/types.ts +0 -309
  190. package/src/capability/doc/tool.ts +0 -1629
  191. package/src/capability/doc/types.ts +0 -792
  192. package/src/capability/mcp/index.ts +0 -10
  193. package/src/capability/mcp/schema.ts +0 -107
  194. package/src/capability/mcp/tool.ts +0 -174
  195. package/src/capability/mcp/transport.ts +0 -394
  196. package/src/channel.config.test.ts +0 -147
  197. package/src/channel.lifecycle.test.ts +0 -255
  198. package/src/channel.meta.test.ts +0 -26
  199. package/src/channel.ts +0 -256
  200. package/src/config/accounts.resolve.test.ts +0 -75
  201. package/src/config/accounts.ts +0 -296
  202. package/src/config/derived-paths.test.ts +0 -111
  203. package/src/config/derived-paths.ts +0 -41
  204. package/src/config/index.ts +0 -26
  205. package/src/config/media.test.ts +0 -113
  206. package/src/config/media.ts +0 -139
  207. package/src/config/network.ts +0 -53
  208. package/src/config/routing.test.ts +0 -88
  209. package/src/config/routing.ts +0 -26
  210. package/src/config/runtime-config.ts +0 -46
  211. package/src/config/schema.ts +0 -90
  212. package/src/context-store.ts +0 -297
  213. package/src/crypto/index.ts +0 -24
  214. package/src/crypto.test.ts +0 -32
  215. package/src/crypto.ts +0 -176
  216. package/src/domain/models.ts +0 -7
  217. package/src/domain/policies.ts +0 -36
  218. package/src/dynamic-agent.account-scope.test.ts +0 -17
  219. package/src/gateway-monitor.ts +0 -181
  220. package/src/http.ts +0 -145
  221. package/src/media.test.ts +0 -82
  222. package/src/monitor/limits.ts +0 -7
  223. package/src/monitor/state.queue.test.ts +0 -185
  224. package/src/monitor/state.ts +0 -34
  225. package/src/monitor.active.test.ts +0 -245
  226. package/src/monitor.inbound-filter.test.ts +0 -63
  227. package/src/monitor.integration.test.ts +0 -208
  228. package/src/monitor.ts +0 -121
  229. package/src/monitor.webhook.test.ts +0 -774
  230. package/src/observability/audit-log.ts +0 -48
  231. package/src/observability/legacy-operational-event-store.ts +0 -36
  232. package/src/observability/raw-envelope-log.ts +0 -28
  233. package/src/observability/status-registry.ts +0 -13
  234. package/src/observability/transport-session-view.ts +0 -14
  235. package/src/onboarding.test.ts +0 -336
  236. package/src/onboarding.ts +0 -704
  237. package/src/outbound.test.ts +0 -1271
  238. package/src/outbound.ts +0 -746
  239. package/src/runtime/dispatcher.ts +0 -71
  240. package/src/runtime/outbound-intent.ts +0 -4
  241. package/src/runtime/reply-orchestrator.test.ts +0 -71
  242. package/src/runtime/reply-orchestrator.ts +0 -67
  243. package/src/runtime/routing-bridge.test.ts +0 -115
  244. package/src/runtime/routing-bridge.ts +0 -44
  245. package/src/runtime/session-manager.test.ts +0 -174
  246. package/src/runtime/session-manager.ts +0 -139
  247. package/src/runtime/source-registry.ts +0 -249
  248. package/src/runtime.ts +0 -14
  249. package/src/shared/command-auth.ts +0 -87
  250. package/src/shared/media-asset.ts +0 -78
  251. package/src/shared/media-service.test.ts +0 -111
  252. package/src/shared/media-service.ts +0 -84
  253. package/src/shared/media-types.ts +0 -5
  254. package/src/shared/xml-parser.test.ts +0 -50
  255. package/src/store/active-reply-store.ts +0 -42
  256. package/src/store/interfaces.ts +0 -11
  257. package/src/store/memory-store.ts +0 -43
  258. package/src/store/stream-batch-store.ts +0 -350
  259. package/src/transport/agent-api/client.ts +0 -277
  260. package/src/transport/agent-api/core.ts +0 -463
  261. package/src/transport/agent-api/delivery.ts +0 -41
  262. package/src/transport/agent-api/media-upload.ts +0 -11
  263. package/src/transport/agent-api/reply.ts +0 -39
  264. package/src/transport/agent-api/upstream-delivery.ts +0 -45
  265. package/src/transport/agent-api/upstream-media-upload.ts +0 -70
  266. package/src/transport/agent-api/upstream-reply.ts +0 -43
  267. package/src/transport/agent-callback/http-handler.ts +0 -47
  268. package/src/transport/agent-callback/inbound.ts +0 -5
  269. package/src/transport/agent-callback/reply.ts +0 -13
  270. package/src/transport/agent-callback/request-handler.ts +0 -244
  271. package/src/transport/agent-callback/session.ts +0 -23
  272. package/src/transport/bot-webhook/active-reply.ts +0 -39
  273. package/src/transport/bot-webhook/http-handler.ts +0 -48
  274. package/src/transport/bot-webhook/inbound-normalizer.test.ts +0 -433
  275. package/src/transport/bot-webhook/inbound-normalizer.ts +0 -558
  276. package/src/transport/bot-webhook/inbound.ts +0 -5
  277. package/src/transport/bot-webhook/message-shape.ts +0 -92
  278. package/src/transport/bot-webhook/protocol.ts +0 -148
  279. package/src/transport/bot-webhook/reply.ts +0 -15
  280. package/src/transport/bot-webhook/request-handler.ts +0 -394
  281. package/src/transport/bot-webhook/session.ts +0 -23
  282. package/src/transport/bot-ws/inbound.test.ts +0 -290
  283. package/src/transport/bot-ws/inbound.ts +0 -163
  284. package/src/transport/bot-ws/media.test.ts +0 -44
  285. package/src/transport/bot-ws/media.ts +0 -321
  286. package/src/transport/bot-ws/reply.test.ts +0 -450
  287. package/src/transport/bot-ws/reply.ts +0 -365
  288. package/src/transport/bot-ws/sdk-adapter.test.ts +0 -187
  289. package/src/transport/bot-ws/sdk-adapter.ts +0 -314
  290. package/src/transport/bot-ws/session.ts +0 -28
  291. package/src/transport/http/common.ts +0 -109
  292. package/src/transport/http/registry.ts +0 -92
  293. package/src/transport/http/request-handler.ts +0 -84
  294. package/src/types/account.ts +0 -70
  295. package/src/types/config.ts +0 -114
  296. package/src/types/constants.ts +0 -31
  297. package/src/types/events.ts +0 -21
  298. package/src/types/global.d.ts +0 -9
  299. package/src/types/index.ts +0 -17
  300. package/src/types/legacy-stream.ts +0 -50
  301. package/src/types/message.ts +0 -189
  302. package/src/types/runtime-context.ts +0 -28
  303. package/src/types/runtime.ts +0 -165
  304. package/src/types.ts +0 -41
  305. package/src/upstream/index.ts +0 -150
  306. package/src/upstream.test.ts +0 -84
  307. package/src/wecom_msg_adapter/markdown_adapter.ts +0 -331
  308. package/tsconfig.json +0 -22
  309. package/vitest.config.ts +0 -26
  310. /package/{src/capability/agent/index.ts → dist/src/capability/agent/index.js} +0 -0
  311. /package/{src/capability/bot/index.ts → dist/src/capability/bot/index.js} +0 -0
  312. /package/{src/capability/calendar/index.ts → dist/src/capability/calendar/index.js} +0 -0
  313. /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
- });