@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,43 +0,0 @@
1
- import type { ResolvedAgentAccount } from "../../types/index.js";
2
- import type { WecomTarget } from "../../target.js";
3
- import { sendUpstreamAgentApiMedia, sendUpstreamAgentApiText } from "./client.js";
4
-
5
- export async function sendUpstreamAgentApiTextReply(params: {
6
- upstreamAgent: ResolvedAgentAccount;
7
- primaryAgent: ResolvedAgentAccount;
8
- target: WecomTarget;
9
- text: string;
10
- }): Promise<void> {
11
- await sendUpstreamAgentApiText({
12
- upstreamAgent: params.upstreamAgent,
13
- primaryAgent: params.primaryAgent,
14
- toUser: params.target.touser,
15
- toParty: params.target.toparty,
16
- toTag: params.target.totag,
17
- chatId: params.target.chatid,
18
- text: params.text,
19
- });
20
- }
21
-
22
- export async function sendUpstreamAgentApiMediaReply(params: {
23
- upstreamAgent: ResolvedAgentAccount;
24
- primaryAgent: ResolvedAgentAccount;
25
- target: WecomTarget;
26
- mediaId: string;
27
- mediaType: "image" | "voice" | "video" | "file";
28
- title?: string;
29
- description?: string;
30
- }): Promise<void> {
31
- await sendUpstreamAgentApiMedia({
32
- upstreamAgent: params.upstreamAgent,
33
- primaryAgent: params.primaryAgent,
34
- toUser: params.target.touser,
35
- toParty: params.target.toparty,
36
- toTag: params.target.totag,
37
- chatId: params.target.chatid,
38
- mediaId: params.mediaId,
39
- mediaType: params.mediaType,
40
- title: params.title,
41
- description: params.description,
42
- });
43
- }
@@ -1,47 +0,0 @@
1
- import type { OpenClawConfig } from "openclaw/plugin-sdk";
2
-
3
- import type { WecomRuntimeEnv } from "../../types/runtime-context.js";
4
- import type { ResolvedAgentAccount } from "../../types/index.js";
5
- import type { WecomAccountRuntime } from "../../app/account-runtime.js";
6
- import { resolveAgentCallbackPaths } from "./inbound.js";
7
- import { createAgentCallbackSessionSnapshot } from "./session.js";
8
- import { registerAgentWebhookTarget } from "../http/registry.js";
9
-
10
- export function startAgentCallbackTransport(params: {
11
- account: ResolvedAgentAccount;
12
- cfg: OpenClawConfig;
13
- runtime: WecomAccountRuntime;
14
- runtimeEnv: WecomRuntimeEnv;
15
- }): { paths: string[]; stop: () => void } {
16
- const paths = resolveAgentCallbackPaths(params.account.accountId);
17
- params.runtime.updateTransportSession(
18
- createAgentCallbackSessionSnapshot({
19
- accountId: params.account.accountId,
20
- running: true,
21
- }),
22
- );
23
- const unregisters = paths.map((path) =>
24
- registerAgentWebhookTarget({
25
- agent: params.account,
26
- config: params.cfg,
27
- runtimeEnv: params.runtimeEnv,
28
- touchTransportSession: (patch) => params.runtime.touchTransportSession("agent-callback", patch),
29
- auditSink: (event) => params.runtime.recordOperationalIssue(event),
30
- path,
31
- }),
32
- );
33
- return {
34
- paths,
35
- stop: () => {
36
- for (const unregister of unregisters) {
37
- unregister();
38
- }
39
- params.runtime.updateTransportSession(
40
- createAgentCallbackSessionSnapshot({
41
- accountId: params.account.accountId,
42
- running: false,
43
- }),
44
- );
45
- },
46
- };
47
- }
@@ -1,5 +0,0 @@
1
- import { resolveDerivedPathSummary } from "../../config/index.js";
2
-
3
- export function resolveAgentCallbackPaths(accountId: string): string[] {
4
- return resolveDerivedPathSummary(accountId).agentCallback;
5
- }
@@ -1,13 +0,0 @@
1
- import type { ReplyContext } from "../../types/index.js";
2
-
3
- export function createAgentCallbackReplyContext(params: {
4
- accountId: string;
5
- raw: ReplyContext["raw"];
6
- }): ReplyContext {
7
- return {
8
- transport: "agent-callback",
9
- accountId: params.accountId,
10
- passiveWindowMs: 5_000,
11
- raw: params.raw,
12
- };
13
- }
@@ -1,244 +0,0 @@
1
- import type { IncomingMessage, ServerResponse } from "node:http";
2
-
3
- import { decryptWecomEncrypted, verifyWecomSignature } from "../../crypto.js";
4
- import { extractEncryptFromXml } from "../../crypto/xml.js";
5
- import { getWecomRuntime } from "../../runtime.js";
6
- import { handleAgentWebhook } from "../../agent/index.js";
7
- import { extractAgentId, parseXml } from "../../shared/xml-parser.js";
8
- import { LIMITS as WECOM_LIMITS } from "../../types/constants.js";
9
- import type { AgentWebhookTarget } from "../http/registry.js";
10
- import {
11
- logRouteFailure,
12
- readTextBody,
13
- resolveQueryParams,
14
- resolveSignatureParam,
15
- type RouteFailureReason,
16
- writeRouteFailure,
17
- } from "../http/common.js";
18
-
19
- const ERROR_HELP = "\n\n遇到问题?联系作者: YanHaidao (微信: YanHaidao)";
20
-
21
- function truncateForLog(raw: string, maxChars = 600): string {
22
- const compact = raw.replace(/\s+/g, " ").trim();
23
- if (compact.length <= maxChars) return compact;
24
- return `${compact.slice(0, maxChars)}...(truncated)`;
25
- }
26
-
27
- function buildParsedAgentSummary(parsed: ReturnType<typeof parseXml>): string {
28
- const data = parsed as Record<string, unknown>;
29
- const msgType = String(data.MsgType ?? "").trim() || "unknown";
30
- const fromUser = String(data.FromUserName ?? "").trim() || "N/A";
31
- const toUser = String(data.ToUserName ?? "").trim() || "N/A";
32
- const event = String(data.Event ?? "").trim() || "N/A";
33
- const msgId = String(data.MsgId ?? "").trim() || "N/A";
34
- const chatId = String(data.ChatId ?? data.chatid ?? "").trim() || "N/A";
35
- const agentId = String(data.AgentID ?? "").trim() || "N/A";
36
- return `msgType=${msgType} from=${fromUser} to=${toUser} event=${event} msgId=${msgId} chatId=${chatId} agentId=${agentId}`;
37
- }
38
-
39
- function normalizeAgentIdValue(value: unknown): number | undefined {
40
- if (typeof value === "number" && Number.isFinite(value)) return value;
41
- const raw = String(value ?? "").trim();
42
- if (!raw) return undefined;
43
- const parsed = Number(raw);
44
- return Number.isFinite(parsed) ? parsed : undefined;
45
- }
46
-
47
- export async function handleAgentCallbackRequest(params: {
48
- req: IncomingMessage;
49
- res: ServerResponse;
50
- path: string;
51
- reqId: string;
52
- targets: AgentWebhookTarget[];
53
- }): Promise<boolean> {
54
- const { req, res, path, reqId, targets } = params;
55
- if (targets.length === 0) {
56
- console.error(
57
- `[wecom] inbound(agent): reqId=${reqId} path=${path} no_registered_target availableTargets=0`,
58
- );
59
- res.statusCode = 404;
60
- res.setHeader("Content-Type", "text/plain; charset=utf-8");
61
- res.end(`agent not configured for path=${path} - Agent 模式未配置或回调路径错误,请运行 openclaw onboarding${ERROR_HELP}`);
62
- return true;
63
- }
64
-
65
- const query = resolveQueryParams(req);
66
- const timestamp = query.get("timestamp") ?? "";
67
- const nonce = query.get("nonce") ?? "";
68
- const signature = resolveSignatureParam(query);
69
- const hasSig = Boolean(signature);
70
- const remote = req.socket?.remoteAddress ?? "unknown";
71
-
72
- if (req.method === "GET") {
73
- const echostr = query.get("echostr") ?? "";
74
- const signatureMatches = targets.filter((target) =>
75
- verifyWecomSignature({
76
- token: target.agent.token,
77
- timestamp,
78
- nonce,
79
- encrypt: echostr,
80
- signature,
81
- }),
82
- );
83
- if (signatureMatches.length !== 1) {
84
- const reason: RouteFailureReason =
85
- signatureMatches.length === 0 ? "wecom_account_not_found" : "wecom_account_conflict";
86
- const candidateIds = (signatureMatches.length > 0 ? signatureMatches : targets).map((target) => target.agent.accountId);
87
- logRouteFailure({
88
- reqId,
89
- path,
90
- method: "GET",
91
- reason,
92
- candidateAccountIds: candidateIds,
93
- });
94
- writeRouteFailure(
95
- res,
96
- reason,
97
- reason === "wecom_account_conflict"
98
- ? "Agent callback account conflict: multiple accounts matched signature."
99
- : "Agent callback account not found: signature verification failed.",
100
- );
101
- return true;
102
- }
103
- const selected = signatureMatches[0]!;
104
- try {
105
- const plain = decryptWecomEncrypted({
106
- encodingAESKey: selected.agent.encodingAESKey,
107
- receiveId: selected.agent.corpId,
108
- encrypt: echostr,
109
- });
110
- res.statusCode = 200;
111
- res.setHeader("Content-Type", "text/plain; charset=utf-8");
112
- res.end(plain);
113
- return true;
114
- } catch {
115
- res.statusCode = 400;
116
- res.setHeader("Content-Type", "text/plain; charset=utf-8");
117
- res.end(`decrypt failed - 解密失败,请检查 EncodingAESKey${ERROR_HELP}`);
118
- return true;
119
- }
120
- }
121
-
122
- if (req.method !== "POST") {
123
- return false;
124
- }
125
-
126
- const rawBody = await readTextBody(req, WECOM_LIMITS.MAX_REQUEST_BODY_SIZE);
127
- if (!rawBody.ok) {
128
- res.statusCode = 400;
129
- res.setHeader("Content-Type", "text/plain; charset=utf-8");
130
- res.end(rawBody.error || "invalid payload");
131
- return true;
132
- }
133
-
134
- console.log(
135
- `[wecom] inbound(agent): reqId=${reqId} path=${path} rawXmlBytes=${Buffer.byteLength(rawBody.value, "utf8")} rawPreview=${JSON.stringify(truncateForLog(rawBody.value))}`,
136
- );
137
-
138
- let encrypted = "";
139
- try {
140
- encrypted = extractEncryptFromXml(rawBody.value);
141
- } catch {
142
- res.statusCode = 400;
143
- res.setHeader("Content-Type", "text/plain; charset=utf-8");
144
- res.end(`invalid xml - 缺少 Encrypt 字段${ERROR_HELP}`);
145
- return true;
146
- }
147
-
148
- console.log(
149
- `[wecom] inbound(agent): reqId=${reqId} path=${path} encryptedLen=${encrypted.length}`,
150
- );
151
-
152
- const signatureMatches = targets.filter((target) =>
153
- verifyWecomSignature({
154
- token: target.agent.token,
155
- timestamp,
156
- nonce,
157
- encrypt: encrypted,
158
- signature,
159
- }),
160
- );
161
- if (signatureMatches.length !== 1) {
162
- const reason: RouteFailureReason =
163
- signatureMatches.length === 0 ? "wecom_account_not_found" : "wecom_account_conflict";
164
- const candidateIds = (signatureMatches.length > 0 ? signatureMatches : targets).map((target) => target.agent.accountId);
165
- logRouteFailure({
166
- reqId,
167
- path,
168
- method: "POST",
169
- reason,
170
- candidateAccountIds: candidateIds,
171
- });
172
- writeRouteFailure(
173
- res,
174
- reason,
175
- reason === "wecom_account_conflict"
176
- ? "Agent callback account conflict: multiple accounts matched signature."
177
- : "Agent callback account not found: signature verification failed.",
178
- );
179
- return true;
180
- }
181
-
182
- const selected = signatureMatches[0]!;
183
- let decrypted = "";
184
- let parsed: ReturnType<typeof parseXml> | null = null;
185
- try {
186
- decrypted = decryptWecomEncrypted({
187
- encodingAESKey: selected.agent.encodingAESKey,
188
- receiveId: selected.agent.corpId,
189
- encrypt: encrypted,
190
- });
191
- parsed = parseXml(decrypted);
192
- } catch {
193
- res.statusCode = 400;
194
- res.setHeader("Content-Type", "text/plain; charset=utf-8");
195
- res.end(`decrypt failed - 解密失败,请检查 EncodingAESKey${ERROR_HELP}`);
196
- return true;
197
- }
198
- if (!parsed) {
199
- res.statusCode = 400;
200
- res.setHeader("Content-Type", "text/plain; charset=utf-8");
201
- res.end(`invalid xml - XML 解析失败${ERROR_HELP}`);
202
- return true;
203
- }
204
-
205
- selected.runtimeEnv.log?.(
206
- `[wecom] inbound(agent): reqId=${reqId} accountId=${selected.agent.accountId} decryptedBytes=${Buffer.byteLength(decrypted, "utf8")} parsed=${buildParsedAgentSummary(parsed)} decryptedPreview=${JSON.stringify(truncateForLog(decrypted))}`,
207
- );
208
-
209
- const inboundAgentId = normalizeAgentIdValue(extractAgentId(parsed));
210
- if (
211
- inboundAgentId !== undefined &&
212
- selected.agent.agentId !== undefined &&
213
- inboundAgentId !== selected.agent.agentId
214
- ) {
215
- selected.runtimeEnv.error?.(
216
- `[wecom] inbound(agent): reqId=${reqId} accountId=${selected.agent.accountId} agentId_mismatch expected=${selected.agent.agentId} actual=${inboundAgentId}`,
217
- );
218
- }
219
-
220
- const core = getWecomRuntime();
221
- selected.runtimeEnv.log?.(
222
- `[wecom] inbound(agent): reqId=${reqId} method=${req.method ?? "UNKNOWN"} remote=${remote} timestamp=${timestamp ? "yes" : "no"} nonce=${nonce ? "yes" : "no"} msg_signature=${hasSig ? "yes" : "no"} accountId=${selected.agent.accountId}`,
223
- );
224
- selected.touchTransportSession?.({ lastInboundAt: Date.now(), running: true });
225
- return handleAgentWebhook({
226
- req,
227
- res,
228
- verifiedPost: {
229
- timestamp,
230
- nonce,
231
- signature,
232
- encrypted,
233
- decrypted,
234
- parsed,
235
- },
236
- agent: selected.agent,
237
- config: selected.config,
238
- core,
239
- log: selected.runtimeEnv.log,
240
- error: selected.runtimeEnv.error,
241
- auditSink: selected.auditSink,
242
- touchTransportSession: selected.touchTransportSession,
243
- });
244
- }
@@ -1,23 +0,0 @@
1
- import type { TransportSessionSnapshot } from "../../types/index.js";
2
-
3
- export function createAgentCallbackSessionSnapshot(params: {
4
- accountId: string;
5
- running: boolean;
6
- lastInboundAt?: number;
7
- lastOutboundAt?: number;
8
- lastError?: string;
9
- }): TransportSessionSnapshot {
10
- return {
11
- accountId: params.accountId,
12
- transport: "agent-callback",
13
- running: params.running,
14
- ownerId: `${params.accountId}:agent-callback`,
15
- connected: params.running,
16
- authenticated: true,
17
- lastConnectedAt: params.running ? Date.now() : undefined,
18
- lastDisconnectedAt: params.running ? undefined : Date.now(),
19
- lastInboundAt: params.lastInboundAt,
20
- lastOutboundAt: params.lastOutboundAt,
21
- lastError: params.lastError,
22
- };
23
- }
@@ -1,39 +0,0 @@
1
- import { wecomFetch } from "../../http.js";
2
- import { LIMITS, monitorState } from "../../monitor/state.js";
3
-
4
- const activeReplyStore = monitorState.activeReplyStore;
5
-
6
- export function storeActiveReply(streamId: string, responseUrl?: string, proxyUrl?: string): void {
7
- activeReplyStore.store(streamId, responseUrl, proxyUrl);
8
- }
9
-
10
- export function getActiveReplyUrl(streamId: string): string | undefined {
11
- return activeReplyStore.getUrl(streamId);
12
- }
13
-
14
- export async function useActiveReplyOnce(
15
- streamId: string,
16
- fn: (params: { responseUrl: string; proxyUrl?: string }) => Promise<void>,
17
- ): Promise<void> {
18
- return activeReplyStore.use(streamId, async (params) => {
19
- await new Promise((resolve) => setTimeout(resolve, 1000));
20
- await fn(params);
21
- });
22
- }
23
-
24
- export async function sendActiveMessage(streamId: string, content: string): Promise<void> {
25
- await useActiveReplyOnce(streamId, async ({ responseUrl, proxyUrl }) => {
26
- const res = await wecomFetch(
27
- responseUrl,
28
- {
29
- method: "POST",
30
- headers: { "Content-Type": "application/json" },
31
- body: JSON.stringify({ msgtype: "text", text: { content } }),
32
- },
33
- { proxyUrl, timeoutMs: LIMITS.REQUEST_TIMEOUT_MS },
34
- );
35
- if (!res.ok) {
36
- throw new Error(`active send failed: ${res.status}`);
37
- }
38
- });
39
- }
@@ -1,48 +0,0 @@
1
- import type { OpenClawConfig } from "openclaw/plugin-sdk";
2
-
3
- import type { WecomRuntimeEnv } from "../../types/runtime-context.js";
4
- import type { ResolvedBotAccount } from "../../types/index.js";
5
- import type { WecomAccountRuntime } from "../../app/account-runtime.js";
6
- import { resolveBotWebhookPaths } from "./inbound.js";
7
- import { createBotWebhookSessionSnapshot } from "./session.js";
8
- import { registerWecomWebhookTarget } from "../http/registry.js";
9
-
10
- export function startBotWebhookTransport(params: {
11
- account: ResolvedBotAccount;
12
- cfg: OpenClawConfig;
13
- runtime: WecomAccountRuntime;
14
- runtimeEnv: WecomRuntimeEnv;
15
- }): { paths: string[]; stop: () => void } {
16
- const paths = resolveBotWebhookPaths(params.account.accountId);
17
- params.runtime.updateTransportSession(
18
- createBotWebhookSessionSnapshot({
19
- accountId: params.account.accountId,
20
- running: true,
21
- }),
22
- );
23
- const unregisters = paths.map((path) =>
24
- registerWecomWebhookTarget({
25
- account: params.account,
26
- config: params.cfg,
27
- runtime: params.runtimeEnv,
28
- core: params.runtime.core,
29
- path,
30
- touchTransportSession: (patch) => params.runtime.touchTransportSession("bot-webhook", patch),
31
- auditSink: (event) => params.runtime.recordOperationalIssue(event),
32
- }),
33
- );
34
- return {
35
- paths,
36
- stop: () => {
37
- for (const unregister of unregisters) {
38
- unregister();
39
- }
40
- params.runtime.updateTransportSession(
41
- createBotWebhookSessionSnapshot({
42
- accountId: params.account.accountId,
43
- running: false,
44
- }),
45
- );
46
- },
47
- };
48
- }