@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,96 +0,0 @@
1
- import type { ResolvedAgentAccount } from "../../types/index.js";
2
- import { resolveScopedWecomTarget } from "../../target.js";
3
- import { deliverUpstreamAgentApiMedia, deliverUpstreamAgentApiText } from "../../transport/agent-api/upstream-delivery.js";
4
- import { canUseAgentApiDelivery } from "./fallback-policy.js";
5
- import { getWecomRuntime } from "../../runtime.js";
6
-
7
- /**
8
- * 上下游企业消息发送服务
9
- *
10
- * 使用下游企业的 access_token 和 agentId 发送消息
11
- */
12
- export class WecomUpstreamAgentDeliveryService {
13
- constructor(
14
- private readonly upstreamAgent: ResolvedAgentAccount,
15
- private readonly primaryAgent: ResolvedAgentAccount,
16
- ) { }
17
-
18
- assertAvailable(): void {
19
- if (!canUseAgentApiDelivery(this.upstreamAgent)) {
20
- throw new Error(
21
- `WeCom upstream outbound requires channels.wecom.accounts.<accountId>.agent.agentId for upstream corp=${this.upstreamAgent.corpId}.`,
22
- );
23
- }
24
- }
25
-
26
- resolveTargetOrThrow(to: string | undefined) {
27
- const scoped = resolveScopedWecomTarget(to, this.upstreamAgent.accountId);
28
- if (!scoped) {
29
- console.error(`[wecom-upstream-delivery] missing target account=${this.upstreamAgent.accountId}`);
30
- throw new Error("WeCom upstream outbound requires a target (userid, partyid, tagid or chatid).");
31
- }
32
- if (scoped.accountId && scoped.accountId !== this.upstreamAgent.accountId) {
33
- console.error(
34
- `[wecom-upstream-delivery] account mismatch current=${this.upstreamAgent.accountId} targetAccount=${scoped.accountId} raw=${String(to ?? "")}`,
35
- );
36
- throw new Error(
37
- `WeCom upstream outbound account mismatch: target belongs to account=${scoped.accountId}, current account=${this.upstreamAgent.accountId}.`,
38
- );
39
- }
40
- const target = scoped.target;
41
- if (target.chatid) {
42
- console.warn(
43
- `[wecom-upstream-delivery] blocked chat target account=${this.upstreamAgent.accountId} chatId=${target.chatid}`,
44
- );
45
- throw new Error(
46
- `企业微信(WeCom)上下游 Agent 主动发送不支持向群 chatId 发送(chatId=${target.chatid})。` +
47
- `请改为发送给用户(userid / user:xxx)。`,
48
- );
49
- }
50
- return target;
51
- }
52
-
53
- async sendText(params: { to: string | undefined; text: string }): Promise<void> {
54
- this.assertAvailable();
55
- const target = this.resolveTargetOrThrow(params.to);
56
- console.log(
57
- `[wecom-upstream-delivery] sendText account=${this.upstreamAgent.accountId} corpId=${this.upstreamAgent.corpId} to=${String(params.to ?? "")} len=${params.text.length}`,
58
- );
59
-
60
- const runtime = getWecomRuntime();
61
- const chunks = runtime.channel.text.chunkText(params.text, 2048);
62
-
63
- for (const chunk of chunks) {
64
- if (!chunk.trim()) continue;
65
- await deliverUpstreamAgentApiText({
66
- upstreamAgent: this.upstreamAgent,
67
- primaryAgent: this.primaryAgent,
68
- target,
69
- text: chunk,
70
- });
71
- }
72
- }
73
-
74
- async sendMedia(params: {
75
- to: string | undefined;
76
- text?: string;
77
- buffer: Buffer;
78
- filename: string;
79
- contentType: string;
80
- }): Promise<void> {
81
- this.assertAvailable();
82
- const target = this.resolveTargetOrThrow(params.to);
83
- console.log(
84
- `[wecom-upstream-delivery] sendMedia account=${this.upstreamAgent.accountId} corpId=${this.upstreamAgent.corpId} to=${String(params.to ?? "")} filename=${params.filename} contentType=${params.contentType}`,
85
- );
86
- await deliverUpstreamAgentApiMedia({
87
- upstreamAgent: this.upstreamAgent,
88
- primaryAgent: this.primaryAgent,
89
- target,
90
- buffer: params.buffer,
91
- filename: params.filename,
92
- contentType: params.contentType,
93
- text: params.text,
94
- });
95
- }
96
- }
@@ -1,47 +0,0 @@
1
- import type { OpenClawConfig } from "openclaw/plugin-sdk";
2
-
3
- export function buildWecomBotDispatchConfig(config: OpenClawConfig): OpenClawConfig {
4
- const baseAgents = (config as any)?.agents ?? {};
5
- const baseAgentDefaults = (baseAgents as any)?.defaults ?? {};
6
- const baseBlockChunk = (baseAgentDefaults as any)?.blockStreamingChunk ?? {};
7
- const baseBlockCoalesce = (baseAgentDefaults as any)?.blockStreamingCoalesce ?? {};
8
- const baseTools = (config as any)?.tools ?? {};
9
- const baseSandbox = (baseTools as any)?.sandbox ?? {};
10
- const baseSandboxTools = (baseSandbox as any)?.tools ?? {};
11
- const existingTopLevelDeny = Array.isArray((baseTools as any).deny) ? ((baseTools as any).deny as string[]) : [];
12
- const existingSandboxDeny = Array.isArray((baseSandboxTools as any).deny) ? ((baseSandboxTools as any).deny as string[]) : [];
13
- const topLevelDeny = Array.from(new Set([...existingTopLevelDeny, "message"]));
14
- const sandboxDeny = Array.from(new Set([...existingSandboxDeny, "message"]));
15
- return {
16
- ...(config as any),
17
- agents: {
18
- ...baseAgents,
19
- defaults: {
20
- ...baseAgentDefaults,
21
- blockStreamingChunk: {
22
- ...baseBlockChunk,
23
- minChars: baseBlockChunk.minChars ?? 120,
24
- maxChars: baseBlockChunk.maxChars ?? 360,
25
- breakPreference: baseBlockChunk.breakPreference ?? "sentence",
26
- },
27
- blockStreamingCoalesce: {
28
- ...baseBlockCoalesce,
29
- minChars: baseBlockCoalesce.minChars ?? 120,
30
- maxChars: baseBlockCoalesce.maxChars ?? 360,
31
- idleMs: baseBlockCoalesce.idleMs ?? 250,
32
- },
33
- },
34
- },
35
- tools: {
36
- ...baseTools,
37
- deny: topLevelDeny,
38
- sandbox: {
39
- ...baseSandbox,
40
- tools: {
41
- ...baseSandboxTools,
42
- deny: sandboxDeny,
43
- },
44
- },
45
- },
46
- } as OpenClawConfig;
47
- }
@@ -1,178 +0,0 @@
1
- import type { OpenClawConfig, PluginRuntime } from "openclaw/plugin-sdk";
2
-
3
- import { resolveWecomAccount } from "../../config/index.js";
4
- import { wecomFetch } from "../../http.js";
5
- import { LIMITS } from "../../monitor/state.js";
6
- import type { StreamState } from "../../types/legacy-stream.js";
7
- import type { ResolvedAgentAccount } from "../../types/index.js";
8
- import { sendMedia as sendAgentMedia, sendText as sendAgentText, uploadMedia } from "../../transport/agent-api/core.js";
9
- import { buildStreamReplyFromState } from "../../transport/bot-webhook/protocol.js";
10
- import { useActiveReplyOnce } from "../../transport/bot-webhook/active-reply.js";
11
- import { guessContentTypeFromPath } from "../../transport/bot-webhook/inbound-normalizer.js";
12
-
13
- const STREAM_MAX_DM_BYTES = 200_000;
14
-
15
- export function appendDmContent(state: StreamState, text: string): void {
16
- const next = state.dmContent ? `${state.dmContent}\n\n${text}`.trim() : text.trim();
17
- const buf = Buffer.from(next, "utf8");
18
- state.dmContent = buf.length <= STREAM_MAX_DM_BYTES ? next : buf.subarray(buf.length - STREAM_MAX_DM_BYTES).toString("utf8");
19
- }
20
-
21
- export function resolveAgentAccountOrUndefined(cfg: OpenClawConfig, accountId: string): ResolvedAgentAccount | undefined {
22
- const agent = resolveWecomAccount({ cfg, accountId }).agent;
23
- return agent?.configured ? agent : undefined;
24
- }
25
-
26
- export function buildFallbackPrompt(params: {
27
- kind: "media" | "timeout" | "error";
28
- agentConfigured: boolean;
29
- userId?: string;
30
- filename?: string;
31
- chatType?: "group" | "direct";
32
- }): string {
33
- const who = params.userId ? `(${params.userId})` : "";
34
- const scope = params.chatType === "group" ? "群聊" : params.chatType === "direct" ? "私聊" : "会话";
35
- if (!params.agentConfigured) {
36
- return `${scope}中需要通过应用私信发送${params.filename ? `(${params.filename})` : ""},但管理员尚未配置企业微信自建应用(Agent)通道。请联系管理员配置后再试。${who}`.trim();
37
- }
38
- if (!params.userId) {
39
- return `${scope}中需要通过应用私信兜底发送${params.filename ? `(${params.filename})` : ""},但本次回调未能识别触发者 userid(请检查企微回调字段 from.userid / fromuserid)。请联系管理员排查配置。`.trim();
40
- }
41
- if (params.kind === "media") {
42
- return `已生成文件${params.filename ? `(${params.filename})` : ""},将通过应用私信发送给你。${who}`.trim();
43
- }
44
- if (params.kind === "timeout") {
45
- return `内容较长,为避免超时,后续内容将通过应用私信发送给你。${who}`.trim();
46
- }
47
- return `交付出现异常,已尝试通过应用私信发送给你。${who}`.trim();
48
- }
49
-
50
- export async function sendBotFallbackPromptNow(params: { streamId: string; text: string }): Promise<void> {
51
- await useActiveReplyOnce(params.streamId, async ({ responseUrl, proxyUrl }) => {
52
- const payload = {
53
- msgtype: "stream",
54
- stream: {
55
- id: params.streamId,
56
- finish: true,
57
- content: params.text.trim() || "1",
58
- },
59
- };
60
- const res = await wecomFetch(
61
- responseUrl,
62
- {
63
- method: "POST",
64
- headers: { "Content-Type": "application/json" },
65
- body: JSON.stringify(payload),
66
- },
67
- { proxyUrl, timeoutMs: LIMITS.REQUEST_TIMEOUT_MS },
68
- );
69
- if (!res.ok) {
70
- throw new Error(`fallback prompt push failed: ${res.status}`);
71
- }
72
- });
73
- }
74
-
75
- export async function pushFinalStreamReplyNow(params: { streamId: string; state: StreamState }): Promise<void> {
76
- const finalReply = buildStreamReplyFromState(params.state) as unknown as Record<string, unknown>;
77
- await useActiveReplyOnce(params.streamId, async ({ responseUrl, proxyUrl }) => {
78
- const res = await wecomFetch(
79
- responseUrl,
80
- {
81
- method: "POST",
82
- headers: { "Content-Type": "application/json" },
83
- body: JSON.stringify(finalReply),
84
- },
85
- { proxyUrl, timeoutMs: LIMITS.REQUEST_TIMEOUT_MS },
86
- );
87
- if (!res.ok) {
88
- throw new Error(`final stream push failed: ${res.status}`);
89
- }
90
- });
91
- }
92
-
93
- export async function sendAgentDmText(params: {
94
- agent: ResolvedAgentAccount;
95
- userId: string;
96
- text: string;
97
- core: PluginRuntime;
98
- }): Promise<void> {
99
- const chunks = params.core.channel.text.chunkText(params.text, 2048);
100
- for (const chunk of chunks) {
101
- const trimmed = chunk.trim();
102
- if (!trimmed) continue;
103
- await sendAgentText({ agent: params.agent, toUser: params.userId, text: trimmed });
104
- }
105
- }
106
-
107
- export async function sendAgentDmMedia(params: {
108
- agent: ResolvedAgentAccount;
109
- userId: string;
110
- mediaUrlOrPath: string;
111
- contentType?: string;
112
- filename: string;
113
- }): Promise<void> {
114
- let buffer: Buffer;
115
- let inferredContentType = params.contentType;
116
- const looksLikeUrl = /^https?:\/\//i.test(params.mediaUrlOrPath);
117
- if (looksLikeUrl) {
118
- const res = await fetch(params.mediaUrlOrPath, { signal: AbortSignal.timeout(30_000) });
119
- if (!res.ok) throw new Error(`media download failed: ${res.status}`);
120
- buffer = Buffer.from(await res.arrayBuffer());
121
- inferredContentType = inferredContentType || res.headers.get("content-type") || "application/octet-stream";
122
- } else {
123
- const fs = await import("node:fs/promises");
124
- buffer = await fs.readFile(params.mediaUrlOrPath);
125
- }
126
-
127
- let mediaType: "image" | "voice" | "video" | "file" = "file";
128
- const ct = (inferredContentType || "").toLowerCase();
129
- if (ct.startsWith("image/")) mediaType = "image";
130
- else if (ct.startsWith("audio/")) mediaType = "voice";
131
- else if (ct.startsWith("video/")) mediaType = "video";
132
-
133
- const mediaId = await uploadMedia({
134
- agent: params.agent,
135
- type: mediaType,
136
- buffer,
137
- filename: params.filename,
138
- });
139
- await sendAgentMedia({
140
- agent: params.agent,
141
- toUser: params.userId,
142
- mediaId,
143
- mediaType,
144
- });
145
- }
146
-
147
- export function extractLocalImagePathsFromText(params: { text: string; mustAlsoAppearIn: string }): string[] {
148
- const text = params.text;
149
- const mustAlsoAppearIn = params.mustAlsoAppearIn;
150
- if (!text.trim()) return [];
151
- const exts = "(png|jpg|jpeg|gif|webp|bmp)";
152
- const re = new RegExp(String.raw`(\/(?:Users|tmp|root|home)\/[^\s"'<>]+?\.${exts})`, "gi");
153
- const found = new Set<string>();
154
- let m: RegExpExecArray | null;
155
- while ((m = re.exec(text))) {
156
- const p = m[1];
157
- if (!p) continue;
158
- if (!mustAlsoAppearIn.includes(p)) continue;
159
- found.add(p);
160
- }
161
- return Array.from(found);
162
- }
163
-
164
- export function extractLocalFilePathsFromText(text: string): string[] {
165
- if (!text.trim()) return [];
166
- const re = /\/(?:Users|tmp|root|home)\/[^\s"'<>]+/g;
167
- const found = new Set<string>();
168
- let m: RegExpExecArray | null;
169
- while ((m = re.exec(text))) {
170
- const p = m[0]?.trim();
171
- if (p) found.add(p);
172
- }
173
- return Array.from(found);
174
- }
175
-
176
- export function guessLocalPathContentType(filePath: string): string | undefined {
177
- return guessContentTypeFromPath(filePath);
178
- }
@@ -1,215 +0,0 @@
1
- import crypto from "node:crypto";
2
-
3
- import { wecomFetch } from "../../http.js";
4
- import { LIMITS, type StreamStore } from "../../monitor/state.js";
5
- import { getActiveReplyUrl, useActiveReplyOnce } from "../../transport/bot-webhook/active-reply.js";
6
- import type { WecomWebhookTarget } from "../../types/runtime-context.js";
7
- import { buildStreamReplyFromState } from "../../transport/bot-webhook/protocol.js";
8
- import {
9
- buildFallbackPrompt,
10
- extractLocalFilePathsFromText,
11
- guessLocalPathContentType,
12
- resolveAgentAccountOrUndefined,
13
- sendAgentDmMedia,
14
- sendBotFallbackPromptNow,
15
- } from "./fallback-delivery.js";
16
- import type { BotRuntimeLogger } from "./types.js";
17
-
18
- export async function handleDirectLocalPathIntent(params: {
19
- streamStore: StreamStore;
20
- target: WecomWebhookTarget;
21
- streamId: string;
22
- rawBody: string;
23
- userId: string;
24
- chatType: "group" | "direct";
25
- logVerbose: BotRuntimeLogger;
26
- looksLikeSendLocalFileIntent: (rawBody: string) => boolean;
27
- }): Promise<boolean> {
28
- const { streamStore, target, streamId, rawBody, userId, chatType, logVerbose, looksLikeSendLocalFileIntent } = params;
29
- const directLocalPaths = extractLocalFilePathsFromText(rawBody);
30
- if (directLocalPaths.length) {
31
- logVerbose(
32
- target,
33
- `local-path: 检测到用户消息包含本机路径 count=${directLocalPaths.length} intent=${looksLikeSendLocalFileIntent(rawBody)}`,
34
- );
35
- }
36
- if (!directLocalPaths.length || !looksLikeSendLocalFileIntent(rawBody)) {
37
- return false;
38
- }
39
-
40
- const fs = await import("node:fs/promises");
41
- const pathModule = await import("node:path");
42
- const imageExts = new Set(["png", "jpg", "jpeg", "gif", "webp", "bmp"]);
43
-
44
- const imagePaths: string[] = [];
45
- const otherPaths: string[] = [];
46
- for (const p of directLocalPaths) {
47
- const ext = pathModule.extname(p).slice(1).toLowerCase();
48
- if (imageExts.has(ext)) imagePaths.push(p);
49
- else otherPaths.push(p);
50
- }
51
-
52
- if (imagePaths.length > 0 && otherPaths.length === 0) {
53
- const loaded: Array<{ base64: string; md5: string; path: string }> = [];
54
- for (const p of imagePaths) {
55
- try {
56
- const buf = await fs.readFile(p);
57
- const base64 = buf.toString("base64");
58
- const md5 = crypto.createHash("md5").update(buf).digest("hex");
59
- loaded.push({ base64, md5, path: p });
60
- } catch (err) {
61
- target.runtime.error?.(`local-path: 读取图片失败 path=${p}: ${String(err)}`);
62
- }
63
- }
64
-
65
- if (loaded.length > 0) {
66
- streamStore.updateStream(streamId, (s) => {
67
- s.images = loaded.map(({ base64, md5 }) => ({ base64, md5 }));
68
- s.content = loaded.length === 1 ? `已发送图片(${pathModule.basename(loaded[0]!.path)})` : `已发送 ${loaded.length} 张图片`;
69
- s.finished = true;
70
- });
71
-
72
- const responseUrl = getActiveReplyUrl(streamId);
73
- if (responseUrl) {
74
- try {
75
- const finalReply = buildStreamReplyFromState(streamStore.getStream(streamId)!) as unknown as Record<string, unknown>;
76
- await useActiveReplyOnce(streamId, async ({ responseUrl, proxyUrl }) => {
77
- const res = await wecomFetch(
78
- responseUrl,
79
- {
80
- method: "POST",
81
- headers: { "Content-Type": "application/json" },
82
- body: JSON.stringify(finalReply),
83
- },
84
- { proxyUrl, timeoutMs: LIMITS.REQUEST_TIMEOUT_MS },
85
- );
86
- if (!res.ok) throw new Error(`local-path image push failed: ${res.status}`);
87
- });
88
- logVerbose(target, `local-path: 已通过 Bot response_url 推送图片 frames=final images=${loaded.length}`);
89
- } catch (err) {
90
- target.runtime.error?.(`local-path: Bot 主动推送图片失败(将依赖 stream_refresh 拉取): ${String(err)}`);
91
- }
92
- } else {
93
- logVerbose(target, "local-path: 无 response_url,等待 stream_refresh 拉取最终图片");
94
- }
95
- streamStore.onStreamFinished(streamId);
96
- return true;
97
- }
98
-
99
- const agentCfg = resolveAgentAccountOrUndefined(target.config, target.account.accountId);
100
- const agentOk = Boolean(agentCfg);
101
- const fallbackName = imagePaths.length === 1 ? (imagePaths[0]!.split("/").pop() || "image") : `${imagePaths.length} 张图片`;
102
- const prompt = buildFallbackPrompt({
103
- kind: "media",
104
- agentConfigured: agentOk,
105
- userId,
106
- filename: fallbackName,
107
- chatType,
108
- });
109
-
110
- streamStore.updateStream(streamId, (s) => {
111
- s.fallbackMode = "error";
112
- s.finished = true;
113
- s.content = prompt;
114
- s.fallbackPromptSentAt = s.fallbackPromptSentAt ?? Date.now();
115
- });
116
-
117
- try {
118
- await sendBotFallbackPromptNow({ streamId, text: prompt });
119
- logVerbose(target, "local-path: 图片读取失败后已推送兜底提示");
120
- } catch (err) {
121
- target.runtime.error?.(`local-path: 图片读取失败后的兜底提示推送失败: ${String(err)}`);
122
- }
123
-
124
- if (agentCfg && userId && userId !== "unknown") {
125
- for (const p of imagePaths) {
126
- const guessedType = guessLocalPathContentType(p);
127
- try {
128
- await sendAgentDmMedia({
129
- agent: agentCfg,
130
- userId,
131
- mediaUrlOrPath: p,
132
- contentType: guessedType,
133
- filename: p.split("/").pop() || "image",
134
- });
135
- streamStore.updateStream(streamId, (s) => {
136
- s.agentMediaKeys = Array.from(new Set([...(s.agentMediaKeys ?? []), p]));
137
- });
138
- logVerbose(
139
- target,
140
- `local-path: 图片已通过 Agent 私信发送 user=${userId} path=${p} contentType=${guessedType ?? "unknown"}`,
141
- );
142
- } catch (err) {
143
- target.runtime.error?.(`local-path: 图片 Agent 私信兜底失败 path=${p}: ${String(err)}`);
144
- }
145
- }
146
- }
147
- streamStore.onStreamFinished(streamId);
148
- return true;
149
- }
150
-
151
- if (otherPaths.length > 0) {
152
- const agentCfg = resolveAgentAccountOrUndefined(target.config, target.account.accountId);
153
- const agentOk = Boolean(agentCfg);
154
- const filename = otherPaths.length === 1 ? otherPaths[0]!.split("/").pop()! : `${otherPaths.length} 个文件`;
155
- const prompt = buildFallbackPrompt({
156
- kind: "media",
157
- agentConfigured: agentOk,
158
- userId,
159
- filename,
160
- chatType,
161
- });
162
-
163
- streamStore.updateStream(streamId, (s) => {
164
- s.fallbackMode = "media";
165
- s.finished = true;
166
- s.content = prompt;
167
- s.fallbackPromptSentAt = s.fallbackPromptSentAt ?? Date.now();
168
- });
169
-
170
- try {
171
- await sendBotFallbackPromptNow({ streamId, text: prompt });
172
- logVerbose(target, "local-path: 文件兜底提示已推送");
173
- } catch (err) {
174
- target.runtime.error?.(`local-path: 文件兜底提示推送失败: ${String(err)}`);
175
- }
176
-
177
- if (!agentCfg) {
178
- streamStore.onStreamFinished(streamId);
179
- return true;
180
- }
181
- if (!userId || userId === "unknown") {
182
- target.runtime.error?.("local-path: 无法识别触发者 userId,无法 Agent 私信发送文件");
183
- streamStore.onStreamFinished(streamId);
184
- return true;
185
- }
186
-
187
- for (const p of otherPaths) {
188
- const alreadySent = streamStore.getStream(streamId)?.agentMediaKeys?.includes(p);
189
- if (alreadySent) continue;
190
- const guessedType = guessLocalPathContentType(p);
191
- try {
192
- await sendAgentDmMedia({
193
- agent: agentCfg,
194
- userId,
195
- mediaUrlOrPath: p,
196
- contentType: guessedType,
197
- filename: p.split("/").pop() || "file",
198
- });
199
- streamStore.updateStream(streamId, (s) => {
200
- s.agentMediaKeys = Array.from(new Set([...(s.agentMediaKeys ?? []), p]));
201
- });
202
- logVerbose(
203
- target,
204
- `local-path: 文件已通过 Agent 私信发送 user=${userId} path=${p} contentType=${guessedType ?? "unknown"}`,
205
- );
206
- } catch (err) {
207
- target.runtime.error?.(`local-path: Agent 私信发送文件失败 path=${p}: ${String(err)}`);
208
- }
209
- }
210
- streamStore.onStreamFinished(streamId);
211
- return true;
212
- }
213
-
214
- return false;
215
- }