@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,75 +0,0 @@
1
- import type { OpenClawConfig } from "openclaw/plugin-sdk";
2
- import { describe, expect, it } from "vitest";
3
-
4
- import { resolveWecomAccount } from "./accounts.js";
5
-
6
- describe("resolveWecomAccount", () => {
7
- const cfg: OpenClawConfig = {
8
- channels: {
9
- wecom: {
10
- enabled: true,
11
- defaultAccount: "acct-a",
12
- accounts: {
13
- "acct-a": {
14
- enabled: true,
15
- bot: {
16
- primaryTransport: "webhook",
17
- webhook: {
18
- token: "token-a",
19
- encodingAESKey: "aes-a",
20
- },
21
- },
22
- },
23
- },
24
- },
25
- },
26
- } as OpenClawConfig;
27
-
28
- it("does not fall back when explicit accountId does not exist", () => {
29
- const account = resolveWecomAccount({ cfg, accountId: "missing" });
30
- expect(account.accountId).toBe("missing");
31
- expect(account.enabled).toBe(false);
32
- expect(account.configured).toBe(false);
33
- });
34
-
35
- it("uses configured default account when accountId is omitted", () => {
36
- const account = resolveWecomAccount({ cfg });
37
- expect(account.accountId).toBe("acct-a");
38
- expect(account.enabled).toBe(true);
39
- expect(account.configured).toBe(true);
40
- });
41
-
42
- it("treats literal default as an alias for configured default account", () => {
43
- const account = resolveWecomAccount({ cfg, accountId: "default" });
44
- expect(account.accountId).toBe("acct-a");
45
- expect(account.enabled).toBe(true);
46
- expect(account.configured).toBe(true);
47
- });
48
-
49
- it("accepts agentSecret for fresh configs and normalizes it for runtime use", () => {
50
- const agentCfg: OpenClawConfig = {
51
- channels: {
52
- wecom: {
53
- enabled: true,
54
- defaultAccount: "acct-agent",
55
- accounts: {
56
- "acct-agent": {
57
- enabled: true,
58
- agent: {
59
- corpId: "corp-id",
60
- agentSecret: "agent-secret",
61
- agentId: 1000001,
62
- token: "token",
63
- encodingAESKey: "1234567890123456789012345678901234567890123",
64
- },
65
- },
66
- },
67
- },
68
- },
69
- } as OpenClawConfig;
70
-
71
- const account = resolveWecomAccount({ cfg: agentCfg });
72
- expect(account.agent?.apiConfigured).toBe(true);
73
- expect(account.agent?.corpSecret).toBe("agent-secret");
74
- });
75
- });
@@ -1,296 +0,0 @@
1
- import type { OpenClawConfig } from "openclaw/plugin-sdk";
2
-
3
- import type {
4
- ResolvedAgentAccount,
5
- ResolvedBotAccount,
6
- ResolvedMode,
7
- ResolvedWecomAccount,
8
- ResolvedWecomAccounts,
9
- WecomAccountConfig,
10
- WecomAgentConfig,
11
- WecomBotConfig,
12
- WecomConfig,
13
- WecomNetworkConfig,
14
- } from "../types/index.js";
15
-
16
- export const DEFAULT_ACCOUNT_ID = "default";
17
-
18
- export type WecomAccountConflict = {
19
- type: "duplicate_bot_id" | "duplicate_agent_id";
20
- accountId: string;
21
- ownerAccountId: string;
22
- message: string;
23
- };
24
-
25
- function toNumber(value: number | string | undefined): number | undefined {
26
- if (value == null) return undefined;
27
- const parsed = typeof value === "number" ? value : Number(value);
28
- return Number.isFinite(parsed) ? parsed : undefined;
29
- }
30
-
31
- function resolveBotAccount(
32
- accountId: string,
33
- config: WecomBotConfig,
34
- network?: WecomNetworkConfig,
35
- ): ResolvedBotAccount {
36
- const primaryTransport = config.primaryTransport ?? (config.ws ? "ws" : "webhook");
37
- const wsConfigured = Boolean(config.ws?.botId && config.ws?.secret);
38
- const webhookConfigured = Boolean(config.webhook?.token && config.webhook?.encodingAESKey);
39
- const configured = primaryTransport === "ws" ? wsConfigured : webhookConfigured;
40
- return {
41
- accountId,
42
- configured,
43
- primaryTransport,
44
- wsConfigured,
45
- webhookConfigured,
46
- config,
47
- network,
48
- ws: config.ws
49
- ? {
50
- botId: config.ws.botId,
51
- secret: config.ws.secret,
52
- }
53
- : undefined,
54
- webhook: config.webhook
55
- ? {
56
- token: config.webhook.token,
57
- encodingAESKey: config.webhook.encodingAESKey,
58
- receiveId: config.webhook.receiveId?.trim() ?? "",
59
- }
60
- : undefined,
61
- token: config.webhook?.token ?? "",
62
- encodingAESKey: config.webhook?.encodingAESKey ?? "",
63
- receiveId: config.webhook?.receiveId?.trim() ?? "",
64
- botId: config.ws?.botId ?? "",
65
- secret: config.ws?.secret ?? "",
66
- };
67
- }
68
-
69
- function resolveAgentAccount(
70
- accountId: string,
71
- config: WecomAgentConfig,
72
- network?: WecomNetworkConfig,
73
- ): ResolvedAgentAccount {
74
- const agentId = toNumber(config.agentId);
75
- const callbackConfigured = Boolean(config.token && config.encodingAESKey);
76
- const normalizedAgentSecret = config.agentSecret?.trim() || config.corpSecret?.trim() || "";
77
- const apiConfigured = Boolean(config.corpId && normalizedAgentSecret && agentId);
78
- return {
79
- accountId,
80
- configured: callbackConfigured || apiConfigured,
81
- callbackConfigured,
82
- apiConfigured,
83
- corpId: config.corpId,
84
- corpSecret: normalizedAgentSecret,
85
- agentId,
86
- token: config.token,
87
- encodingAESKey: config.encodingAESKey,
88
- config,
89
- network,
90
- };
91
- }
92
-
93
- function toResolvedAccount(params: {
94
- accountId: string;
95
- enabled: boolean;
96
- name?: string;
97
- config: WecomAccountConfig;
98
- network?: WecomNetworkConfig;
99
- }): ResolvedWecomAccount {
100
- const bot = params.config.bot
101
- ? resolveBotAccount(params.accountId, params.config.bot, params.network)
102
- : undefined;
103
- const agent = params.config.agent
104
- ? resolveAgentAccount(params.accountId, params.config.agent, params.network)
105
- : undefined;
106
- return {
107
- accountId: params.accountId,
108
- name: params.name,
109
- enabled: params.enabled,
110
- configured: Boolean(bot?.configured || agent?.configured),
111
- config: params.config,
112
- bot,
113
- agent,
114
- };
115
- }
116
-
117
- function createMissingResolvedAccount(accountId: string): ResolvedWecomAccount {
118
- return {
119
- accountId,
120
- enabled: false,
121
- configured: false,
122
- config: {},
123
- };
124
- }
125
-
126
- export function detectMode(config: WecomConfig | undefined): ResolvedMode {
127
- if (!config || config.enabled === false) return "disabled";
128
- if (config.accounts && Object.keys(config.accounts).length > 0) {
129
- return "matrix";
130
- }
131
- if (config.bot || config.agent) {
132
- return "legacy";
133
- }
134
- return "disabled";
135
- }
136
-
137
- function resolveMatrixAccounts(wecom: WecomConfig): Record<string, ResolvedWecomAccount> {
138
- const resolved: Record<string, ResolvedWecomAccount> = {};
139
- for (const [rawId, entry] of Object.entries(wecom.accounts ?? {})) {
140
- const accountId = rawId.trim();
141
- if (!accountId || !entry) continue;
142
- resolved[accountId] = toResolvedAccount({
143
- accountId,
144
- enabled: wecom.enabled !== false && entry.enabled !== false,
145
- name: entry.name,
146
- config: entry,
147
- network: wecom.network,
148
- });
149
- }
150
- return resolved;
151
- }
152
-
153
- function resolveLegacyAccounts(wecom: WecomConfig): Record<string, ResolvedWecomAccount> {
154
- const config: WecomAccountConfig = {
155
- bot: wecom.bot,
156
- agent: wecom.agent,
157
- };
158
- return {
159
- [DEFAULT_ACCOUNT_ID]: toResolvedAccount({
160
- accountId: DEFAULT_ACCOUNT_ID,
161
- enabled: wecom.enabled !== false,
162
- config,
163
- network: wecom.network,
164
- }),
165
- };
166
- }
167
-
168
- function normalizeKey(value: string): string {
169
- return value.trim().toLowerCase();
170
- }
171
-
172
- function collectWecomAccountConflicts(cfg: OpenClawConfig): Map<string, WecomAccountConflict> {
173
- const resolved = resolveWecomAccounts(cfg);
174
- const conflicts = new Map<string, WecomAccountConflict>();
175
- const botOwners = new Map<string, string>();
176
- const agentOwners = new Map<string, string>();
177
-
178
- for (const accountId of Object.keys(resolved.accounts).sort((a, b) => a.localeCompare(b))) {
179
- const account = resolved.accounts[accountId];
180
- if (!account || account.enabled === false) continue;
181
-
182
- const botId = account.bot?.botId?.trim();
183
- if (botId) {
184
- const key = normalizeKey(botId);
185
- const owner = botOwners.get(key);
186
- if (owner && owner !== accountId) {
187
- conflicts.set(accountId, {
188
- type: "duplicate_bot_id",
189
- accountId,
190
- ownerAccountId: owner,
191
- message:
192
- `Duplicate WeCom botId: account "${accountId}" shares botId with account "${owner}". ` +
193
- "Keep one owner account per botId.",
194
- });
195
- } else {
196
- botOwners.set(key, accountId);
197
- }
198
- }
199
-
200
- const corpId = account.agent?.corpId?.trim();
201
- const agentId = account.agent?.agentId;
202
- if (corpId && typeof agentId === "number") {
203
- const key = `${normalizeKey(corpId)}:${agentId}`;
204
- const owner = agentOwners.get(key);
205
- if (owner && owner !== accountId) {
206
- conflicts.set(accountId, {
207
- type: "duplicate_agent_id",
208
- accountId,
209
- ownerAccountId: owner,
210
- message:
211
- `Duplicate WeCom agent identity: account "${accountId}" shares corpId/agentId (${corpId}/${agentId}) with account "${owner}". ` +
212
- "Keep one owner account per corpId/agentId pair.",
213
- });
214
- } else {
215
- agentOwners.set(key, accountId);
216
- }
217
- }
218
- }
219
-
220
- return conflicts;
221
- }
222
-
223
- export function resolveWecomAccountConflict(params: {
224
- cfg: OpenClawConfig;
225
- accountId: string;
226
- }): WecomAccountConflict | undefined {
227
- return collectWecomAccountConflicts(params.cfg).get(params.accountId);
228
- }
229
-
230
- export function listWecomAccountIds(cfg: OpenClawConfig): string[] {
231
- const wecom = cfg.channels?.wecom as WecomConfig | undefined;
232
- const mode = detectMode(wecom);
233
- if (mode === "matrix") {
234
- return Object.keys(wecom?.accounts ?? {})
235
- .map((value) => value.trim())
236
- .filter(Boolean)
237
- .sort((a, b) => a.localeCompare(b));
238
- }
239
- if (mode === "legacy") {
240
- return [DEFAULT_ACCOUNT_ID];
241
- }
242
- return [];
243
- }
244
-
245
- export function resolveDefaultWecomAccountId(cfg: OpenClawConfig): string {
246
- const wecom = cfg.channels?.wecom as WecomConfig | undefined;
247
- const ids = listWecomAccountIds(cfg);
248
- if (wecom?.defaultAccount && ids.includes(wecom.defaultAccount)) {
249
- return wecom.defaultAccount;
250
- }
251
- return ids[0] ?? DEFAULT_ACCOUNT_ID;
252
- }
253
-
254
- export function resolveWecomAccounts(cfg: OpenClawConfig): ResolvedWecomAccounts {
255
- const wecom = (cfg.channels?.wecom as WecomConfig | undefined) ?? {};
256
- const mode = detectMode(wecom);
257
- const accounts = mode === "matrix" ? resolveMatrixAccounts(wecom) : mode === "legacy" ? resolveLegacyAccounts(wecom) : {};
258
- const defaultAccountId = resolveDefaultWecomAccountId(cfg);
259
- return {
260
- mode,
261
- defaultAccountId,
262
- accounts,
263
- bot: accounts[defaultAccountId]?.bot,
264
- agent: accounts[defaultAccountId]?.agent,
265
- };
266
- }
267
-
268
- export function resolveWecomAccount(params: {
269
- cfg: OpenClawConfig;
270
- accountId?: string | null;
271
- }): ResolvedWecomAccount {
272
- const resolved = resolveWecomAccounts(params.cfg);
273
- const explicitAccountId = params.accountId?.trim();
274
- const accountId = explicitAccountId || resolved.defaultAccountId;
275
- const direct = resolved.accounts[accountId];
276
- if (direct) {
277
- return direct;
278
- }
279
-
280
- // Treat the literal "default" as an alias for the configured default account.
281
- // This keeps generic onboarding flows working even when the first WeCom account
282
- // was created under a custom id like "haidao" instead of a literal "default".
283
- if (explicitAccountId === DEFAULT_ACCOUNT_ID) {
284
- const fallback = resolved.accounts[resolved.defaultAccountId];
285
- if (fallback) {
286
- return fallback;
287
- }
288
- }
289
-
290
- return createMissingResolvedAccount(accountId);
291
- }
292
-
293
- export function isWecomEnabled(cfg: OpenClawConfig): boolean {
294
- const resolved = resolveWecomAccounts(cfg);
295
- return Object.values(resolved.accounts).some((account) => account.enabled && account.configured);
296
- }
@@ -1,111 +0,0 @@
1
- import { describe, expect, it } from "vitest";
2
-
3
- import type { OpenClawConfig, PluginRuntime } from "openclaw/plugin-sdk";
4
-
5
- import { resolveDerivedPathSummary } from "./derived-paths.js";
6
- import {
7
- hasMatrixExplicitRoutesRegistered,
8
- registerAgentWebhookTarget,
9
- registerWecomWebhookTarget,
10
- } from "../transport/http/registry.js";
11
- import type { ResolvedAgentAccount, ResolvedBotAccount } from "../types/index.js";
12
-
13
- function createBotAccount(accountId: string): ResolvedBotAccount {
14
- return {
15
- accountId,
16
- configured: true,
17
- primaryTransport: "webhook",
18
- wsConfigured: false,
19
- webhookConfigured: true,
20
- config: {} as ResolvedBotAccount["config"],
21
- token: "token",
22
- encodingAESKey: "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG",
23
- receiveId: "",
24
- botId: "",
25
- secret: "",
26
- webhook: {
27
- token: "token",
28
- encodingAESKey: "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG",
29
- receiveId: "",
30
- },
31
- };
32
- }
33
-
34
- function createAgentAccount(accountId: string): ResolvedAgentAccount {
35
- return {
36
- accountId,
37
- configured: true,
38
- callbackConfigured: true,
39
- apiConfigured: true,
40
- corpId: `corp-${accountId}`,
41
- corpSecret: `secret-${accountId}`,
42
- agentId: 1001,
43
- token: "token",
44
- encodingAESKey: "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG",
45
- config: {} as ResolvedAgentAccount["config"],
46
- };
47
- }
48
-
49
- const emptyConfig = {} as OpenClawConfig;
50
- const emptyCore = {} as PluginRuntime;
51
-
52
- describe("resolveDerivedPathSummary", () => {
53
- it("registers scoped aliases first for the default account", () => {
54
- expect(resolveDerivedPathSummary("default")).toEqual({
55
- botWebhook: [
56
- "/plugins/wecom/bot/default",
57
- "/wecom/bot/default",
58
- "/plugins/wecom/bot",
59
- "/wecom",
60
- "/wecom/bot",
61
- ],
62
- agentCallback: [
63
- "/plugins/wecom/agent/default",
64
- "/wecom/agent/default",
65
- "/plugins/wecom/agent",
66
- "/wecom/agent",
67
- ],
68
- });
69
- });
70
- });
71
-
72
- describe("hasMatrixExplicitRoutesRegistered", () => {
73
- it("ignores default-account scoped aliases", () => {
74
- const unregisterBot = registerWecomWebhookTarget({
75
- account: createBotAccount("default"),
76
- config: emptyConfig,
77
- runtime: {},
78
- core: emptyCore,
79
- path: "/plugins/wecom/bot/default",
80
- });
81
- const unregisterAgent = registerAgentWebhookTarget({
82
- agent: createAgentAccount("default"),
83
- config: emptyConfig,
84
- runtimeEnv: {},
85
- path: "/plugins/wecom/agent/default",
86
- });
87
-
88
- try {
89
- expect(hasMatrixExplicitRoutesRegistered()).toBe(false);
90
- } finally {
91
- unregisterAgent();
92
- unregisterBot();
93
- }
94
- });
95
-
96
- it("detects non-default explicit account routes", () => {
97
- const unregister = registerWecomWebhookTarget({
98
- account: createBotAccount("acct-a"),
99
- config: emptyConfig,
100
- runtime: {},
101
- core: emptyCore,
102
- path: "/plugins/wecom/bot/acct-a",
103
- });
104
-
105
- try {
106
- expect(hasMatrixExplicitRoutesRegistered()).toBe(true);
107
- } finally {
108
- unregister();
109
- }
110
- });
111
- });
@@ -1,41 +0,0 @@
1
- import { DEFAULT_ACCOUNT_ID } from "./accounts.js";
2
- import type { WecomTransportKind } from "../types/runtime.js";
3
- import { WEBHOOK_PATHS } from "../types/constants.js";
4
-
5
- export function resolveDerivedPath(params: {
6
- accountId: string;
7
- transport: Extract<WecomTransportKind, "bot-webhook" | "agent-callback">;
8
- includeLegacy?: boolean;
9
- }): string[] {
10
- const accountId = params.accountId.trim() || DEFAULT_ACCOUNT_ID;
11
- const isDefault = accountId === DEFAULT_ACCOUNT_ID;
12
- if (params.transport === "bot-webhook") {
13
- return isDefault
14
- ? [
15
- `${WEBHOOK_PATHS.BOT_PLUGIN}/${accountId}`,
16
- `${WEBHOOK_PATHS.BOT}/${accountId}`,
17
- WEBHOOK_PATHS.BOT_PLUGIN,
18
- WEBHOOK_PATHS.BOT_ALT,
19
- WEBHOOK_PATHS.BOT,
20
- ]
21
- : [`${WEBHOOK_PATHS.BOT_PLUGIN}/${accountId}`];
22
- }
23
- return isDefault
24
- ? [
25
- `${WEBHOOK_PATHS.AGENT_PLUGIN}/${accountId}`,
26
- `${WEBHOOK_PATHS.AGENT}/${accountId}`,
27
- WEBHOOK_PATHS.AGENT_PLUGIN,
28
- WEBHOOK_PATHS.AGENT,
29
- ]
30
- : [`${WEBHOOK_PATHS.AGENT_PLUGIN}/${accountId}`];
31
- }
32
-
33
- export function resolveDerivedPathSummary(accountId: string): {
34
- botWebhook: string[];
35
- agentCallback: string[];
36
- } {
37
- return {
38
- botWebhook: resolveDerivedPath({ accountId, transport: "bot-webhook" }),
39
- agentCallback: resolveDerivedPath({ accountId, transport: "agent-callback" }),
40
- };
41
- }
@@ -1,26 +0,0 @@
1
- export { WecomConfigSchema, type WecomConfigInput } from "./schema.js";
2
- export {
3
- DEFAULT_ACCOUNT_ID,
4
- detectMode,
5
- listWecomAccountIds,
6
- resolveDefaultWecomAccountId,
7
- resolveWecomAccount,
8
- resolveWecomAccountConflict,
9
- resolveWecomAccounts,
10
- isWecomEnabled,
11
- } from "./accounts.js";
12
- export { resolveWecomRuntimeAccount, resolveWecomRuntimeConfig, type ResolvedRuntimeAccount, type ResolvedRuntimeConfig } from "./runtime-config.js";
13
- export { resolveDerivedPath, resolveDerivedPathSummary } from "./derived-paths.js";
14
- export {
15
- resolveWecomEgressProxyUrl,
16
- resolveWecomEgressProxyUrlFromNetwork,
17
- resolveWecomMediaDownloadTimeoutMs,
18
- } from "./network.js";
19
- export {
20
- DEFAULT_WECOM_MEDIA_MAX_BYTES,
21
- getWecomDefaultMediaLocalRoots,
22
- resolveWecomConfiguredMediaLocalRoots,
23
- resolveWecomMediaMaxBytes,
24
- resolveWecomMergedMediaLocalRoots,
25
- } from "./media.js";
26
- export { resolveWecomFailClosedOnDefaultRoute, shouldRejectWecomDefaultRoute } from "./routing.js";
@@ -1,113 +0,0 @@
1
- import os from "node:os";
2
- import path from "node:path";
3
- import { afterEach, describe, expect, it, vi } from "vitest";
4
- import { resolvePreferredOpenClawTmpDir } from "openclaw/plugin-sdk/infra-runtime";
5
- import { resolveWecomMediaMaxBytes, resolveWecomMergedMediaLocalRoots } from "./media.js";
6
-
7
- describe("resolveWecomMergedMediaLocalRoots", () => {
8
- afterEach(() => {
9
- vi.unstubAllEnvs();
10
- });
11
-
12
- it("merges defaults with configured local roots", () => {
13
- vi.stubEnv("OPENCLAW_STATE_DIR", "/tmp/wecom-state");
14
-
15
- const roots = resolveWecomMergedMediaLocalRoots({
16
- cfg: {
17
- channels: {
18
- wecom: {
19
- media: {
20
- localRoots: ["~/Downloads", "/tmp/custom-root"],
21
- },
22
- },
23
- },
24
- } as never,
25
- });
26
-
27
- expect(roots).toEqual(
28
- expect.arrayContaining([
29
- path.resolve(resolvePreferredOpenClawTmpDir()),
30
- "/tmp/wecom-state",
31
- "/tmp/wecom-state/media",
32
- "/tmp/wecom-state/agents",
33
- "/tmp/wecom-state/workspace",
34
- "/tmp/wecom-state/sandboxes",
35
- path.resolve(os.homedir(), "Desktop"),
36
- path.resolve(os.homedir(), "Documents"),
37
- path.resolve(os.homedir(), "Downloads"),
38
- path.resolve(os.homedir(), "Movies"),
39
- path.resolve(os.homedir(), "Pictures"),
40
- "/tmp/custom-root",
41
- ]),
42
- );
43
- });
44
-
45
- it("keeps defaults, base roots, and configured roots without duplicates", () => {
46
- vi.stubEnv("OPENCLAW_STATE_DIR", "/tmp/wecom-state");
47
-
48
- const roots = resolveWecomMergedMediaLocalRoots({
49
- cfg: {
50
- channels: {
51
- wecom: {
52
- media: {
53
- localRoots: ["/tmp/agent-root", "/tmp/downloads"],
54
- },
55
- },
56
- },
57
- } as never,
58
- baseRoots: ["/tmp/agent-root", "/tmp/workspace-agent"],
59
- });
60
-
61
- expect(roots).toEqual(
62
- expect.arrayContaining([
63
- path.resolve(resolvePreferredOpenClawTmpDir()),
64
- "/tmp/wecom-state",
65
- "/tmp/workspace-agent",
66
- "/tmp/agent-root",
67
- "/tmp/downloads",
68
- ]),
69
- );
70
- expect(roots.filter((root) => root === "/tmp/agent-root")).toHaveLength(1);
71
- });
72
- });
73
-
74
- describe("resolveWecomMediaMaxBytes", () => {
75
- it("prefers account mediaMaxMb over channel and agent defaults", () => {
76
- expect(
77
- resolveWecomMediaMaxBytes(
78
- {
79
- agents: {
80
- defaults: {
81
- mediaMaxMb: 12,
82
- },
83
- },
84
- channels: {
85
- wecom: {
86
- mediaMaxMb: 24,
87
- accounts: {
88
- ops: {
89
- mediaMaxMb: 32,
90
- },
91
- },
92
- },
93
- },
94
- } as never,
95
- "ops",
96
- ),
97
- ).toBe(32 * 1024 * 1024);
98
- });
99
-
100
- it("falls back to legacy channels.wecom.media.maxBytes when mediaMaxMb is unset", () => {
101
- expect(
102
- resolveWecomMediaMaxBytes({
103
- channels: {
104
- wecom: {
105
- media: {
106
- maxBytes: 15 * 1024 * 1024,
107
- },
108
- },
109
- },
110
- } as never),
111
- ).toBe(15 * 1024 * 1024);
112
- });
113
- });