@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,59 +0,0 @@
1
- #!/usr/bin/env node
2
-
3
- // 从 stdin 读取平台注入的事件 envelope(JSON)
4
- let raw = "";
5
- process.stdin.setEncoding("utf8");
6
-
7
- process.stdin.on("data", (chunk) => {
8
- raw += chunk;
9
- });
10
-
11
- process.stdin.on("end", () => {
12
- let payload = {};
13
- try {
14
- payload = JSON.parse(raw || "{}");
15
- } catch {
16
- // 输入异常时返回可读错误,避免脚本抛异常导致无输出
17
- process.stdout.write(
18
- JSON.stringify({
19
- ok: false,
20
- action: "reply_text",
21
- reply: {
22
- text: "事件脚本解析输入失败,请联系管理员。",
23
- },
24
- chainToAgent: true,
25
- }),
26
- );
27
- return;
28
- }
29
-
30
- const message = payload && typeof payload === "object" ? payload.message ?? {} : {};
31
- const eventType = String(message.eventType ?? "unknown");
32
- const eventKey = String(message.eventKey ?? "");
33
- const fromUser = String(message.fromUser ?? "");
34
-
35
- // 这里给出联调用的回显文本,业务可改为真正逻辑
36
- const text = [
37
- "已收到菜单事件",
38
- `eventType=${eventType}`,
39
- eventKey ? `eventKey=${eventKey}` : "eventKey=<empty>",
40
- fromUser ? `fromUser=${fromUser}` : "",
41
- "\n你可以在 openclaw 配置里把这个 eventKey 路由到具体业务脚本。",
42
- ]
43
- .filter(Boolean)
44
- .join("\n");
45
-
46
- process.stdout.write(
47
- JSON.stringify({
48
- ok: true,
49
- action: "reply_text",
50
- reply: {
51
- text,
52
- },
53
- chainToAgent: false,
54
- audit: {
55
- tags: ["menu", "click", eventType],
56
- },
57
- }),
58
- );
59
- });
@@ -1,55 +0,0 @@
1
- #!/usr/bin/env python3
2
- import json
3
- import sys
4
-
5
-
6
- def main() -> int:
7
- # 从 stdin 读取平台注入的事件 envelope(JSON)
8
- try:
9
- payload = json.load(sys.stdin)
10
- except Exception:
11
- # 输入异常时返回可读错误,避免脚本抛异常导致无输出
12
- json.dump(
13
- {
14
- "ok": False,
15
- "action": "reply_text",
16
- "reply": {"text": "事件脚本解析输入失败,请联系管理员。"},
17
- "chainToAgent": True,
18
- },
19
- sys.stdout,
20
- ensure_ascii=False,
21
- )
22
- return 0
23
-
24
- message = payload.get("message", {}) if isinstance(payload, dict) else {}
25
- event_type = str(message.get("eventType") or "unknown")
26
- event_key = str(message.get("eventKey") or "")
27
- from_user = str(message.get("fromUser") or "")
28
-
29
- # 这里给出联调用的回显文本,业务可改为真正逻辑
30
- lines = [
31
- "已收到菜单事件",
32
- f"eventType={event_type}",
33
- f"eventKey={event_key or '<empty>'}",
34
- ]
35
- if from_user:
36
- lines.append(f"fromUser={from_user}")
37
- lines.append("")
38
- lines.append("你可以在 openclaw 配置里把这个 eventKey 路由到具体业务脚本。")
39
-
40
- json.dump(
41
- {
42
- "ok": True,
43
- "action": "reply_text",
44
- "reply": {"text": "\n".join(lines)},
45
- "chainToAgent": False,
46
- "audit": {"tags": ["menu", "click", event_type]},
47
- },
48
- sys.stdout,
49
- ensure_ascii=False,
50
- )
51
- return 0
52
-
53
-
54
- if __name__ == "__main__":
55
- raise SystemExit(main())
package/src/accounts.ts DELETED
@@ -1,34 +0,0 @@
1
- import type { OpenClawConfig } from "openclaw/plugin-sdk";
2
-
3
- import type { ResolvedWecomAccount } from "./types/index.js";
4
- import {
5
- listWecomAccountIds as listWecomAccountIdsFromConfig,
6
- resolveDefaultWecomAccountId as resolveDefaultWecomAccountIdFromConfig,
7
- resolveWecomAccount as resolveWecomAccountFromConfig,
8
- } from "./config/accounts.js";
9
-
10
- /**
11
- * Backward-compatible re-export layer.
12
- * Keep this file as a thin wrapper so older imports continue to work,
13
- * while all account logic stays single-sourced in `src/config/accounts.ts`.
14
- */
15
- export function listWecomAccountIds(cfg: OpenClawConfig): string[] {
16
- return listWecomAccountIdsFromConfig(cfg);
17
- }
18
-
19
- export function resolveDefaultWecomAccountId(cfg: OpenClawConfig): string {
20
- return resolveDefaultWecomAccountIdFromConfig(cfg);
21
- }
22
-
23
- export function resolveWecomAccount(params: {
24
- cfg: OpenClawConfig;
25
- accountId?: string | null;
26
- }): ResolvedWecomAccount {
27
- return resolveWecomAccountFromConfig(params);
28
- }
29
-
30
- export function listEnabledWecomAccounts(cfg: OpenClawConfig): ResolvedWecomAccount[] {
31
- return listWecomAccountIdsFromConfig(cfg)
32
- .map((accountId) => resolveWecomAccountFromConfig({ cfg, accountId }))
33
- .filter((account) => account.enabled);
34
- }
@@ -1,109 +0,0 @@
1
- import { beforeEach, describe, expect, it, vi } from "vitest";
2
- import type { ResolvedAgentAccount } from "../types/index.js";
3
-
4
- const { wecomFetchMock, resolveProxyMock } = vi.hoisted(() => ({
5
- wecomFetchMock: vi.fn(),
6
- resolveProxyMock: vi.fn(() => undefined),
7
- }));
8
-
9
- vi.mock("../http.js", () => ({
10
- wecomFetch: wecomFetchMock,
11
- readResponseBodyAsBuffer: vi.fn(),
12
- }));
13
-
14
- vi.mock("../config/index.js", () => ({
15
- resolveWecomEgressProxyUrlFromNetwork: resolveProxyMock,
16
- }));
17
-
18
- import { uploadMedia } from "../transport/agent-api/core.js";
19
-
20
- function createAgent(agentId: number): ResolvedAgentAccount {
21
- return {
22
- accountId: `acct-${agentId}`,
23
- configured: true,
24
- corpId: "corp",
25
- corpSecret: "secret",
26
- agentId,
27
- token: "token",
28
- encodingAESKey: "aes",
29
- config: {} as any,
30
- };
31
- }
32
-
33
- function jsonResponse(body: unknown): Response {
34
- return new Response(JSON.stringify(body), {
35
- status: 200,
36
- headers: { "Content-Type": "application/json" },
37
- });
38
- }
39
-
40
- describe("wecom agent uploadMedia", () => {
41
- beforeEach(() => {
42
- wecomFetchMock.mockReset();
43
- resolveProxyMock.mockReset();
44
- resolveProxyMock.mockReturnValue(undefined);
45
- });
46
-
47
- it("uses text/plain for .txt uploads", async () => {
48
- wecomFetchMock.mockResolvedValueOnce(jsonResponse({ access_token: "token-1", expires_in: 7200 }));
49
- wecomFetchMock.mockResolvedValueOnce(jsonResponse({ errcode: 0, errmsg: "ok", media_id: "m-1" }));
50
-
51
- const mediaId = await uploadMedia({
52
- agent: createAgent(10001),
53
- type: "file",
54
- buffer: Buffer.from("hello txt"),
55
- filename: "note.txt",
56
- });
57
-
58
- expect(mediaId).toBe("m-1");
59
- const [, init] = wecomFetchMock.mock.calls[1] as [string, RequestInit];
60
- const body = init.body as Buffer;
61
- const bodyText = body.toString("utf8");
62
- expect(bodyText).toContain('filename="note.txt"');
63
- expect(bodyText).toContain("Content-Type: text/plain");
64
- });
65
-
66
- it("uses docx mime and normalizes non-ascii filename", async () => {
67
- wecomFetchMock.mockResolvedValueOnce(jsonResponse({ access_token: "token-2", expires_in: 7200 }));
68
- wecomFetchMock.mockResolvedValueOnce(jsonResponse({ errcode: 0, errmsg: "ok", media_id: "m-2" }));
69
-
70
- const mediaId = await uploadMedia({
71
- agent: createAgent(10002),
72
- type: "file",
73
- buffer: Buffer.from("docx bytes"),
74
- filename: "需求文档.docx",
75
- });
76
-
77
- expect(mediaId).toBe("m-2");
78
- const [, init] = wecomFetchMock.mock.calls[1] as [string, RequestInit];
79
- const body = init.body as Buffer;
80
- const bodyText = body.toString("utf8");
81
- expect(bodyText).toContain('filename="file.docx"');
82
- expect(bodyText).toContain(
83
- "Content-Type: application/vnd.openxmlformats-officedocument.wordprocessingml.document",
84
- );
85
- });
86
-
87
- it("retries with octet-stream when preferred mime upload fails", async () => {
88
- wecomFetchMock.mockResolvedValueOnce(jsonResponse({ access_token: "token-3", expires_in: 7200 }));
89
- wecomFetchMock.mockResolvedValueOnce(jsonResponse({ errcode: 40005, errmsg: "invalid media type" }));
90
- wecomFetchMock.mockResolvedValueOnce(jsonResponse({ errcode: 0, errmsg: "ok", media_id: "m-3" }));
91
-
92
- const mediaId = await uploadMedia({
93
- agent: createAgent(10003),
94
- type: "file",
95
- buffer: Buffer.from("yaml bytes"),
96
- filename: "config.yaml",
97
- });
98
-
99
- expect(mediaId).toBe("m-3");
100
- expect(wecomFetchMock).toHaveBeenCalledTimes(3);
101
-
102
- const [, firstUploadInit] = wecomFetchMock.mock.calls[1] as [string, RequestInit];
103
- const [, retryUploadInit] = wecomFetchMock.mock.calls[2] as [string, RequestInit];
104
- const firstUploadBody = (firstUploadInit.body as Buffer).toString("utf8");
105
- const retryUploadBody = (retryUploadInit.body as Buffer).toString("utf8");
106
- expect(firstUploadBody).toContain("Content-Type: application/yaml");
107
- expect(retryUploadBody).toContain("Content-Type: application/octet-stream");
108
- });
109
- });
@@ -1,421 +0,0 @@
1
- import path from "node:path";
2
- import { describe, expect, it } from "vitest";
3
-
4
- import { routeAgentInboundEvent } from "./event-router.js";
5
- import type { ResolvedAgentAccount, WecomAgentInboundMessage } from "../types/index.js";
6
- import type { WecomRuntimeAuditEvent } from "../types/runtime-context.js";
7
-
8
- function createAgent(overrides?: Partial<ResolvedAgentAccount>): ResolvedAgentAccount {
9
- return {
10
- accountId: "default",
11
- configured: true,
12
- callbackConfigured: true,
13
- apiConfigured: true,
14
- corpId: "corp-1",
15
- corpSecret: "secret",
16
- agentId: 1001,
17
- token: "token",
18
- encodingAESKey: "aes",
19
- eventEnabled: true,
20
- allowedEventTypes: ["click", "change_contact"],
21
- config: {
22
- corpId: "corp-1",
23
- corpSecret: "secret",
24
- agentId: 1001,
25
- token: "token",
26
- encodingAESKey: "aes",
27
- },
28
- ...overrides,
29
- };
30
- }
31
-
32
- describe("routeAgentInboundEvent", () => {
33
- it("ignores unmatched events when unmatchedAction is ignore", async () => {
34
- const agent = createAgent({
35
- config: {
36
- corpId: "corp-1",
37
- corpSecret: "secret",
38
- agentId: 1001,
39
- token: "token",
40
- encodingAESKey: "aes",
41
- eventRouting: {
42
- unmatchedAction: "ignore",
43
- routes: [],
44
- },
45
- },
46
- });
47
-
48
- const result = await routeAgentInboundEvent({
49
- agent,
50
- msgType: "event",
51
- eventType: "click",
52
- fromUser: "zhangsan",
53
- msg: { MsgType: "event", Event: "click", EventKey: "MENU_X" },
54
- });
55
-
56
- expect(result.handled).toBe(true);
57
- expect(result.chainToAgent).toBe(false);
58
- expect(result.reason).toBe("unmatched_event_ignored");
59
- });
60
-
61
- it("proxies unmatched events to agent when unmatchedAction is forwardToAgent", async () => {
62
- const agent = createAgent({
63
- config: {
64
- corpId: "corp-1",
65
- corpSecret: "secret",
66
- agentId: 1001,
67
- token: "token",
68
- encodingAESKey: "aes",
69
- eventRouting: {
70
- unmatchedAction: "forwardToAgent",
71
- routes: [],
72
- },
73
- },
74
- });
75
-
76
- const result = await routeAgentInboundEvent({
77
- agent,
78
- msgType: "event",
79
- eventType: "click",
80
- fromUser: "zhangsan",
81
- msg: { MsgType: "event", Event: "click", EventKey: "MENU_X" },
82
- });
83
-
84
- expect(result.handled).toBe(false);
85
- expect(result.chainToAgent).toBe(true);
86
- });
87
-
88
- it("matches builtin echo routes using eventKey", async () => {
89
- const agent = createAgent({
90
- config: {
91
- corpId: "corp-1",
92
- corpSecret: "secret",
93
- agentId: 1001,
94
- token: "token",
95
- encodingAESKey: "aes",
96
- eventRouting: {
97
- routes: [
98
- {
99
- id: "menu-help",
100
- when: { eventType: "click", eventKey: "MENU_HELP" },
101
- handler: { type: "builtin", name: "echo" },
102
- },
103
- ],
104
- },
105
- },
106
- });
107
-
108
- const result = await routeAgentInboundEvent({
109
- agent,
110
- msgType: "event",
111
- eventType: "click",
112
- fromUser: "zhangsan",
113
- msg: { MsgType: "event", Event: "click", EventKey: "MENU_HELP" },
114
- });
115
-
116
- expect(result.handled).toBe(true);
117
- expect(result.replyText).toContain("event=click");
118
- expect(result.replyText).toContain("eventKey=MENU_HELP");
119
- expect(result.matchedRouteId).toBe("menu-help");
120
- });
121
-
122
- it("matches change_contact routes using changeType", async () => {
123
- const agent = createAgent({
124
- config: {
125
- corpId: "corp-1",
126
- corpSecret: "secret",
127
- agentId: 1001,
128
- token: "token",
129
- encodingAESKey: "aes",
130
- eventRouting: {
131
- routes: [
132
- {
133
- id: "contact-create-user",
134
- when: { eventType: "change_contact", changeType: "create_user" },
135
- handler: { type: "builtin", name: "echo" },
136
- },
137
- ],
138
- },
139
- },
140
- });
141
-
142
- const msg: WecomAgentInboundMessage = {
143
- MsgType: "event",
144
- Event: "change_contact",
145
- ChangeType: "create_user",
146
- };
147
- const result = await routeAgentInboundEvent({
148
- agent,
149
- msgType: "event",
150
- eventType: "change_contact",
151
- fromUser: "sys",
152
- msg,
153
- });
154
-
155
- expect(result.handled).toBe(true);
156
- expect(result.replyText).toContain("changeType=create_user");
157
- });
158
-
159
- it("passes full event params to node scripts", async () => {
160
- const fixturePath = path.resolve("src/agent/test-fixtures/reply-event-script.mjs");
161
- const agent = createAgent({
162
- config: {
163
- corpId: "corp-1",
164
- corpSecret: "secret",
165
- agentId: 1001,
166
- token: "token",
167
- encodingAESKey: "aes",
168
- eventRouting: {
169
- routes: [
170
- {
171
- id: "script-click",
172
- when: { eventType: "click", eventKeyPrefix: "MENU_" },
173
- handler: { type: "node_script", entry: fixturePath },
174
- },
175
- ],
176
- },
177
- scriptRuntime: {
178
- enabled: true,
179
- allowPaths: [path.resolve("src/agent/test-fixtures")],
180
- nodeCommand: process.execPath,
181
- },
182
- },
183
- });
184
-
185
- const result = await routeAgentInboundEvent({
186
- agent,
187
- msgType: "event",
188
- eventType: "click",
189
- fromUser: "zhangsan",
190
- msg: {
191
- ToUserName: "corp-1",
192
- FromUserName: "zhangsan",
193
- MsgType: "event",
194
- Event: "click",
195
- EventKey: "MENU_HELP",
196
- AgentID: 1001,
197
- },
198
- });
199
-
200
- expect(result.handled).toBe(true);
201
- expect(result.replyText).toBe("script:click:MENU_HELP:");
202
- });
203
-
204
- it("supports scripts that explicitly continue the default pipeline", async () => {
205
- const fixturePath = path.resolve("src/agent/test-fixtures/reply-event-script.mjs");
206
- const agent = createAgent({
207
- config: {
208
- corpId: "corp-1",
209
- corpSecret: "secret",
210
- agentId: 1001,
211
- token: "token",
212
- encodingAESKey: "aes",
213
- eventRouting: {
214
- routes: [
215
- {
216
- id: "script-click-pass",
217
- when: { eventType: "click", eventKey: "PASS_TO_DEFAULT" },
218
- handler: { type: "node_script", entry: fixturePath },
219
- },
220
- ],
221
- },
222
- scriptRuntime: {
223
- enabled: true,
224
- allowPaths: [path.resolve("src/agent/test-fixtures")],
225
- nodeCommand: process.execPath,
226
- },
227
- },
228
- });
229
-
230
- const result = await routeAgentInboundEvent({
231
- agent,
232
- msgType: "event",
233
- eventType: "click",
234
- fromUser: "zhangsan",
235
- msg: {
236
- MsgType: "event",
237
- Event: "click",
238
- EventKey: "PASS_TO_DEFAULT",
239
- },
240
- });
241
-
242
- expect(result.handled).toBe(true);
243
- expect(result.chainToAgent).toBe(true);
244
- expect(result.replyText).toBeUndefined();
245
- });
246
-
247
- it("passes full event params to python scripts", async () => {
248
- const fixturePath = path.resolve("src/agent/test-fixtures/reply-event-script.py");
249
- const agent = createAgent({
250
- config: {
251
- corpId: "corp-1",
252
- corpSecret: "secret",
253
- agentId: 1001,
254
- token: "token",
255
- encodingAESKey: "aes",
256
- eventRouting: {
257
- routes: [
258
- {
259
- id: "script-python-click",
260
- when: { eventType: "click", eventKey: "MENU_PY" },
261
- handler: { type: "python_script", entry: fixturePath },
262
- },
263
- ],
264
- },
265
- scriptRuntime: {
266
- enabled: true,
267
- allowPaths: [path.resolve("src/agent/test-fixtures")],
268
- pythonCommand: "python3",
269
- },
270
- },
271
- });
272
-
273
- const result = await routeAgentInboundEvent({
274
- agent,
275
- msgType: "event",
276
- eventType: "click",
277
- fromUser: "zhangsan",
278
- msg: {
279
- MsgType: "event",
280
- Event: "click",
281
- EventKey: "MENU_PY",
282
- },
283
- });
284
-
285
- expect(result.handled).toBe(true);
286
- expect(result.replyText).toBe("python:click:MENU_PY:");
287
- });
288
-
289
- it("records audit events for successful script execution", async () => {
290
- const fixturePath = path.resolve("src/agent/test-fixtures/reply-event-script.mjs");
291
- const auditEvents: WecomRuntimeAuditEvent[] = [];
292
- const agent = createAgent({
293
- config: {
294
- corpId: "corp-1",
295
- corpSecret: "secret",
296
- agentId: 1001,
297
- token: "token",
298
- encodingAESKey: "aes",
299
- eventRouting: {
300
- routes: [
301
- {
302
- id: "script-audit-success",
303
- when: { eventType: "click", eventKey: "MENU_AUDIT" },
304
- handler: { type: "node_script", entry: fixturePath },
305
- },
306
- ],
307
- },
308
- scriptRuntime: {
309
- enabled: true,
310
- allowPaths: [path.resolve("src/agent/test-fixtures")],
311
- nodeCommand: process.execPath,
312
- },
313
- },
314
- });
315
-
316
- await routeAgentInboundEvent({
317
- agent,
318
- msgType: "event",
319
- eventType: "click",
320
- fromUser: "zhangsan",
321
- msg: {
322
- MsgType: "event",
323
- Event: "click",
324
- EventKey: "MENU_AUDIT",
325
- },
326
- auditSink: (event) => auditEvents.push(event),
327
- });
328
-
329
- expect(auditEvents).toHaveLength(1);
330
- expect(auditEvents[0]?.category).toBe("inbound");
331
- expect(auditEvents[0]?.summary).toContain("event route script ok");
332
- });
333
-
334
- it("captures invalid script output as a routed error and audits it", async () => {
335
- const fixturePath = path.resolve("src/agent/test-fixtures/invalid-json-script.mjs");
336
- const auditEvents: WecomRuntimeAuditEvent[] = [];
337
- const agent = createAgent({
338
- config: {
339
- corpId: "corp-1",
340
- corpSecret: "secret",
341
- agentId: 1001,
342
- token: "token",
343
- encodingAESKey: "aes",
344
- eventRouting: {
345
- routes: [
346
- {
347
- id: "script-invalid-json",
348
- when: { eventType: "click", eventKey: "MENU_BAD_JSON" },
349
- handler: { type: "node_script", entry: fixturePath },
350
- },
351
- ],
352
- },
353
- scriptRuntime: {
354
- enabled: true,
355
- allowPaths: [path.resolve("src/agent/test-fixtures")],
356
- nodeCommand: process.execPath,
357
- },
358
- },
359
- });
360
-
361
- const result = await routeAgentInboundEvent({
362
- agent,
363
- msgType: "event",
364
- eventType: "click",
365
- fromUser: "zhangsan",
366
- msg: {
367
- MsgType: "event",
368
- Event: "click",
369
- EventKey: "MENU_BAD_JSON",
370
- },
371
- auditSink: (event) => auditEvents.push(event),
372
- });
373
-
374
- expect(result.handled).toBe(true);
375
- expect(result.chainToAgent).toBe(false);
376
- expect(result.reason).toBe("script_node_script_error");
377
- expect(result.error).toContain("not valid JSON");
378
- expect(auditEvents).toHaveLength(1);
379
- expect(auditEvents[0]?.category).toBe("runtime-error");
380
- });
381
-
382
- it("treats invalid eventKeyPattern as non-match instead of throwing", async () => {
383
- const auditEvents: WecomRuntimeAuditEvent[] = [];
384
- const agent = createAgent({
385
- config: {
386
- corpId: "corp-1",
387
- corpSecret: "secret",
388
- agentId: 1001,
389
- token: "token",
390
- encodingAESKey: "aes",
391
- eventRouting: {
392
- unmatchedAction: "ignore",
393
- routes: [
394
- {
395
- id: "invalid-pattern",
396
- when: { eventType: "click", eventKeyPattern: "(*" },
397
- handler: { type: "builtin", name: "echo" },
398
- },
399
- ],
400
- },
401
- },
402
- });
403
-
404
- const result = await routeAgentInboundEvent({
405
- agent,
406
- msgType: "event",
407
- eventType: "click",
408
- fromUser: "zhangsan",
409
- msg: { MsgType: "event", Event: "click", EventKey: "MENU_X" },
410
- auditSink: (event) => auditEvents.push(event),
411
- });
412
-
413
- expect(result.handled).toBe(true);
414
- expect(result.chainToAgent).toBe(false);
415
- expect(result.reason).toBe("unmatched_event_ignored");
416
- expect(auditEvents).toHaveLength(1);
417
- expect(auditEvents[0]?.category).toBe("runtime-error");
418
- expect(auditEvents[0]?.summary).toContain("invalid route eventKeyPattern");
419
- expect(auditEvents[0]?.error).toContain("routeId=invalid-pattern");
420
- });
421
- });