@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.
- package/dist/index.js +68 -0
- package/dist/src/accounts.js +20 -0
- package/dist/src/agent/handler.js +895 -0
- package/dist/src/agent/index.js +5 -0
- package/dist/src/app/account-runtime.js +216 -0
- package/dist/src/app/bootstrap.js +19 -0
- package/dist/src/app/index.js +118 -0
- package/dist/src/capability/agent/delivery-service.js +63 -0
- package/dist/src/capability/agent/fallback-policy.js +6 -0
- package/dist/src/capability/agent/ingress-service.js +33 -0
- package/dist/src/capability/agent/upstream-delivery-service.js +71 -0
- package/dist/src/capability/bot/dispatch-config.js +45 -0
- package/dist/src/capability/bot/fallback-delivery.js +147 -0
- package/dist/src/capability/bot/local-path-delivery.js +178 -0
- package/dist/src/capability/bot/sandbox-media.js +138 -0
- package/dist/src/capability/bot/service.js +49 -0
- package/dist/src/capability/bot/stream-delivery.js +321 -0
- package/dist/src/capability/bot/stream-finalizer.js +81 -0
- package/dist/src/capability/bot/stream-orchestrator.js +318 -0
- package/dist/src/capability/bot/types.js +1 -0
- package/{src/capability/calendar/client.ts → dist/src/capability/calendar/client.js} +118 -241
- package/{src/capability/calendar/schema.ts → dist/src/capability/calendar/schema.js} +0 -38
- package/dist/src/capability/calendar/tool.js +365 -0
- package/dist/src/capability/calendar/types.js +12 -0
- package/{src/capability/doc/client.ts → dist/src/capability/doc/client.js} +370 -605
- package/{src/capability/doc/schema.ts → dist/src/capability/doc/schema.js} +345 -394
- package/dist/src/capability/doc/tool.js +1556 -0
- package/dist/src/capability/doc/types.js +113 -0
- package/dist/src/capability/mcp/index.js +3 -0
- package/dist/src/capability/mcp/schema.js +102 -0
- package/dist/src/capability/mcp/tool.js +146 -0
- package/dist/src/capability/mcp/transport.js +293 -0
- package/dist/src/channel.js +224 -0
- package/dist/src/config/accounts.js +236 -0
- package/dist/src/config/derived-paths.js +31 -0
- package/dist/src/config/index.js +7 -0
- package/dist/src/config/media.js +110 -0
- package/dist/src/config/network.js +32 -0
- package/dist/src/config/routing.js +20 -0
- package/dist/src/config/runtime-config.js +25 -0
- package/dist/src/config/schema.js +4 -0
- package/{src/config-schema.ts → dist/src/config-schema.js} +1 -1
- package/dist/src/context-store.js +219 -0
- package/{src/crypto/aes.ts → dist/src/crypto/aes.js} +11 -28
- package/dist/src/crypto/index.js +9 -0
- package/{src/crypto/signature.ts → dist/src/crypto/signature.js} +3 -18
- package/{src/crypto/xml.ts → dist/src/crypto/xml.js} +3 -11
- package/dist/src/crypto.js +145 -0
- package/dist/src/domain/models.js +1 -0
- package/dist/src/domain/policies.js +32 -0
- package/{src/dynamic-agent.ts → dist/src/dynamic-agent.js} +36 -73
- package/dist/src/gateway-monitor.js +139 -0
- package/dist/src/http.js +114 -0
- package/{src/media.ts → dist/src/media.js} +21 -40
- package/dist/src/monitor/limits.js +7 -0
- package/dist/src/monitor/state.js +28 -0
- package/dist/src/monitor.js +84 -0
- package/dist/src/observability/audit-log.js +30 -0
- package/dist/src/observability/legacy-operational-event-store.js +22 -0
- package/dist/src/observability/raw-envelope-log.js +24 -0
- package/dist/src/observability/status-registry.js +9 -0
- package/dist/src/observability/transport-session-view.js +14 -0
- package/dist/src/onboarding.js +546 -0
- package/dist/src/outbound.js +557 -0
- package/dist/src/runtime/dispatcher.js +57 -0
- package/{src/runtime/index.ts → dist/src/runtime/index.js} +0 -1
- package/dist/src/runtime/outbound-intent.js +1 -0
- package/dist/src/runtime/reply-orchestrator.js +38 -0
- package/dist/src/runtime/routing-bridge.js +26 -0
- package/dist/src/runtime/session-manager.js +112 -0
- package/dist/src/runtime/source-registry.js +174 -0
- package/dist/src/runtime.js +1 -0
- package/dist/src/shared/command-auth.js +57 -0
- package/{src/shared/index.ts → dist/src/shared/index.js} +0 -1
- package/dist/src/shared/media-asset.js +65 -0
- package/dist/src/shared/media-service.js +59 -0
- package/dist/src/shared/media-types.js +1 -0
- package/{src/shared/xml-parser.ts → dist/src/shared/xml-parser.js} +72 -63
- package/dist/src/store/active-reply-store.js +41 -0
- package/dist/src/store/interfaces.js +1 -0
- package/dist/src/store/memory-store.js +33 -0
- package/dist/src/store/stream-batch-store.js +319 -0
- package/{src/target.ts → dist/src/target.js} +15 -48
- package/dist/src/transport/agent-api/client.js +168 -0
- package/dist/src/transport/agent-api/core.js +337 -0
- package/dist/src/transport/agent-api/delivery.js +28 -0
- package/dist/src/transport/agent-api/media-upload.js +4 -0
- package/dist/src/transport/agent-api/reply.js +24 -0
- package/dist/src/transport/agent-api/upstream-delivery.js +30 -0
- package/dist/src/transport/agent-api/upstream-media-upload.js +46 -0
- package/dist/src/transport/agent-api/upstream-reply.js +26 -0
- package/dist/src/transport/agent-callback/http-handler.js +30 -0
- package/dist/src/transport/agent-callback/inbound.js +4 -0
- package/dist/src/transport/agent-callback/reply.js +8 -0
- package/dist/src/transport/agent-callback/request-handler.js +189 -0
- package/dist/src/transport/agent-callback/session.js +15 -0
- package/dist/src/transport/bot-webhook/active-reply.js +27 -0
- package/dist/src/transport/bot-webhook/http-handler.js +31 -0
- package/dist/src/transport/bot-webhook/inbound-normalizer.js +496 -0
- package/dist/src/transport/bot-webhook/inbound.js +4 -0
- package/dist/src/transport/bot-webhook/message-shape.js +98 -0
- package/dist/src/transport/bot-webhook/protocol.js +124 -0
- package/dist/src/transport/bot-webhook/reply.js +9 -0
- package/dist/src/transport/bot-webhook/request-handler.js +285 -0
- package/dist/src/transport/bot-webhook/session.js +15 -0
- package/dist/src/transport/bot-ws/inbound.js +147 -0
- package/dist/src/transport/bot-ws/media.js +236 -0
- package/dist/src/transport/bot-ws/reply.js +310 -0
- package/dist/src/transport/bot-ws/sdk-adapter.js +257 -0
- package/dist/src/transport/bot-ws/session.js +15 -0
- package/dist/src/transport/http/common.js +78 -0
- package/dist/src/transport/http/registry.js +71 -0
- package/dist/src/transport/http/request-handler.js +51 -0
- package/{src/transport/index.ts → dist/src/transport/index.js} +2 -10
- package/dist/src/types/account.js +1 -0
- package/dist/src/types/config.js +1 -0
- package/dist/src/types/constants.js +28 -0
- package/dist/src/types/events.js +1 -0
- package/dist/src/types/index.js +1 -0
- package/dist/src/types/legacy-stream.js +1 -0
- package/dist/src/types/message.js +5 -0
- package/dist/src/types/runtime-context.js +1 -0
- package/dist/src/types/runtime.js +1 -0
- package/dist/src/types.js +1 -0
- package/dist/src/upstream/index.js +111 -0
- package/dist/src/wecom_msg_adapter/markdown_adapter.js +280 -0
- package/openclaw.plugin.json +15 -0
- package/package.json +18 -1
- package/.github/workflows/release.yml +0 -143
- package/GOVERNANCE.md +0 -26
- package/SKILLS_CAL.md +0 -895
- package/SKILLS_DOC.md +0 -2288
- package/UPSTREAM_CONFIG.md +0 -170
- package/UPSTREAM_PLAN.md +0 -175
- package/assets/01.bot-add.png +0 -0
- package/assets/01.bot-setp2.png +0 -0
- package/assets/01.image.jpg +0 -0
- package/assets/02.agent.add.png +0 -0
- package/assets/02.agent.api-set.png +0 -0
- package/assets/02.image.jpg +0 -0
- package/assets/03.agent.page.png +0 -0
- package/assets/03.bot.page.png +0 -0
- package/assets/link-me.jpg +0 -0
- package/assets/register.png +0 -0
- package/changelog/v2.2.28.md +0 -70
- package/changelog/v2.3.10.md +0 -17
- package/changelog/v2.3.11.md +0 -19
- package/changelog/v2.3.12.md +0 -25
- package/changelog/v2.3.13.md +0 -19
- package/changelog/v2.3.14.md +0 -48
- package/changelog/v2.3.15.md +0 -15
- package/changelog/v2.3.16.md +0 -11
- package/changelog/v2.3.18.md +0 -22
- package/changelog/v2.3.19.md +0 -73
- package/changelog/v2.3.2.md +0 -28
- package/changelog/v2.3.26.md +0 -21
- package/changelog/v2.3.27.md +0 -33
- package/changelog/v2.3.4.md +0 -20
- package/changelog/v2.3.9.md +0 -22
- package/changelog/v2.4.12.md +0 -37
- package/changelog/v2.4.16.md +0 -19
- package/compat-single-account.md +0 -148
- package/index.test.ts +0 -38
- package/scripts/test-proxy.ts +0 -70
- package/src/accounts.ts +0 -34
- package/src/agent/api-client.upload.test.ts +0 -109
- package/src/agent/handler.event-filter.test.ts +0 -100
- package/src/agent/handler.ts +0 -1105
- package/src/agent/index.ts +0 -12
- package/src/app/account-runtime.ts +0 -276
- package/src/app/bootstrap.ts +0 -29
- package/src/app/index.ts +0 -192
- package/src/capability/agent/delivery-service.ts +0 -87
- package/src/capability/agent/fallback-policy.ts +0 -13
- package/src/capability/agent/ingress-service.ts +0 -38
- package/src/capability/agent/upstream-delivery-service.ts +0 -96
- package/src/capability/bot/dispatch-config.ts +0 -47
- package/src/capability/bot/fallback-delivery.ts +0 -178
- package/src/capability/bot/local-path-delivery.ts +0 -215
- package/src/capability/bot/sandbox-media.test.ts +0 -221
- package/src/capability/bot/sandbox-media.ts +0 -176
- package/src/capability/bot/service.ts +0 -56
- package/src/capability/bot/stream-delivery.ts +0 -379
- package/src/capability/bot/stream-finalizer.ts +0 -120
- package/src/capability/bot/stream-orchestrator.ts +0 -371
- package/src/capability/bot/types.ts +0 -8
- package/src/capability/calendar/SKILLS_CHECKLIST.md +0 -251
- package/src/capability/calendar/tool.ts +0 -417
- package/src/capability/calendar/types.ts +0 -309
- package/src/capability/doc/tool.ts +0 -1629
- package/src/capability/doc/types.ts +0 -792
- package/src/capability/mcp/index.ts +0 -10
- package/src/capability/mcp/schema.ts +0 -107
- package/src/capability/mcp/tool.ts +0 -174
- package/src/capability/mcp/transport.ts +0 -394
- package/src/channel.config.test.ts +0 -147
- package/src/channel.lifecycle.test.ts +0 -255
- package/src/channel.meta.test.ts +0 -26
- package/src/channel.ts +0 -256
- package/src/config/accounts.resolve.test.ts +0 -75
- package/src/config/accounts.ts +0 -296
- package/src/config/derived-paths.test.ts +0 -111
- package/src/config/derived-paths.ts +0 -41
- package/src/config/index.ts +0 -26
- package/src/config/media.test.ts +0 -113
- package/src/config/media.ts +0 -139
- package/src/config/network.ts +0 -53
- package/src/config/routing.test.ts +0 -88
- package/src/config/routing.ts +0 -26
- package/src/config/runtime-config.ts +0 -46
- package/src/config/schema.ts +0 -90
- package/src/context-store.ts +0 -297
- package/src/crypto/index.ts +0 -24
- package/src/crypto.test.ts +0 -32
- package/src/crypto.ts +0 -176
- package/src/domain/models.ts +0 -7
- package/src/domain/policies.ts +0 -36
- package/src/dynamic-agent.account-scope.test.ts +0 -17
- package/src/gateway-monitor.ts +0 -181
- package/src/http.ts +0 -145
- package/src/media.test.ts +0 -82
- package/src/monitor/limits.ts +0 -7
- package/src/monitor/state.queue.test.ts +0 -185
- package/src/monitor/state.ts +0 -34
- package/src/monitor.active.test.ts +0 -245
- package/src/monitor.inbound-filter.test.ts +0 -63
- package/src/monitor.integration.test.ts +0 -208
- package/src/monitor.ts +0 -121
- package/src/monitor.webhook.test.ts +0 -774
- package/src/observability/audit-log.ts +0 -48
- package/src/observability/legacy-operational-event-store.ts +0 -36
- package/src/observability/raw-envelope-log.ts +0 -28
- package/src/observability/status-registry.ts +0 -13
- package/src/observability/transport-session-view.ts +0 -14
- package/src/onboarding.test.ts +0 -336
- package/src/onboarding.ts +0 -704
- package/src/outbound.test.ts +0 -1271
- package/src/outbound.ts +0 -746
- package/src/runtime/dispatcher.ts +0 -71
- package/src/runtime/outbound-intent.ts +0 -4
- package/src/runtime/reply-orchestrator.test.ts +0 -71
- package/src/runtime/reply-orchestrator.ts +0 -67
- package/src/runtime/routing-bridge.test.ts +0 -115
- package/src/runtime/routing-bridge.ts +0 -44
- package/src/runtime/session-manager.test.ts +0 -174
- package/src/runtime/session-manager.ts +0 -139
- package/src/runtime/source-registry.ts +0 -249
- package/src/runtime.ts +0 -14
- package/src/shared/command-auth.ts +0 -87
- package/src/shared/media-asset.ts +0 -78
- package/src/shared/media-service.test.ts +0 -111
- package/src/shared/media-service.ts +0 -84
- package/src/shared/media-types.ts +0 -5
- package/src/shared/xml-parser.test.ts +0 -50
- package/src/store/active-reply-store.ts +0 -42
- package/src/store/interfaces.ts +0 -11
- package/src/store/memory-store.ts +0 -43
- package/src/store/stream-batch-store.ts +0 -350
- package/src/transport/agent-api/client.ts +0 -277
- package/src/transport/agent-api/core.ts +0 -463
- package/src/transport/agent-api/delivery.ts +0 -41
- package/src/transport/agent-api/media-upload.ts +0 -11
- package/src/transport/agent-api/reply.ts +0 -39
- package/src/transport/agent-api/upstream-delivery.ts +0 -45
- package/src/transport/agent-api/upstream-media-upload.ts +0 -70
- package/src/transport/agent-api/upstream-reply.ts +0 -43
- package/src/transport/agent-callback/http-handler.ts +0 -47
- package/src/transport/agent-callback/inbound.ts +0 -5
- package/src/transport/agent-callback/reply.ts +0 -13
- package/src/transport/agent-callback/request-handler.ts +0 -244
- package/src/transport/agent-callback/session.ts +0 -23
- package/src/transport/bot-webhook/active-reply.ts +0 -39
- package/src/transport/bot-webhook/http-handler.ts +0 -48
- package/src/transport/bot-webhook/inbound-normalizer.test.ts +0 -433
- package/src/transport/bot-webhook/inbound-normalizer.ts +0 -558
- package/src/transport/bot-webhook/inbound.ts +0 -5
- package/src/transport/bot-webhook/message-shape.ts +0 -92
- package/src/transport/bot-webhook/protocol.ts +0 -148
- package/src/transport/bot-webhook/reply.ts +0 -15
- package/src/transport/bot-webhook/request-handler.ts +0 -394
- package/src/transport/bot-webhook/session.ts +0 -23
- package/src/transport/bot-ws/inbound.test.ts +0 -290
- package/src/transport/bot-ws/inbound.ts +0 -163
- package/src/transport/bot-ws/media.test.ts +0 -44
- package/src/transport/bot-ws/media.ts +0 -321
- package/src/transport/bot-ws/reply.test.ts +0 -450
- package/src/transport/bot-ws/reply.ts +0 -365
- package/src/transport/bot-ws/sdk-adapter.test.ts +0 -187
- package/src/transport/bot-ws/sdk-adapter.ts +0 -314
- package/src/transport/bot-ws/session.ts +0 -28
- package/src/transport/http/common.ts +0 -109
- package/src/transport/http/registry.ts +0 -92
- package/src/transport/http/request-handler.ts +0 -84
- package/src/types/account.ts +0 -70
- package/src/types/config.ts +0 -114
- package/src/types/constants.ts +0 -31
- package/src/types/events.ts +0 -21
- package/src/types/global.d.ts +0 -9
- package/src/types/index.ts +0 -17
- package/src/types/legacy-stream.ts +0 -50
- package/src/types/message.ts +0 -189
- package/src/types/runtime-context.ts +0 -28
- package/src/types/runtime.ts +0 -165
- package/src/types.ts +0 -41
- package/src/upstream/index.ts +0 -150
- package/src/upstream.test.ts +0 -84
- package/src/wecom_msg_adapter/markdown_adapter.ts +0 -331
- package/tsconfig.json +0 -22
- package/vitest.config.ts +0 -26
- /package/{src/capability/agent/index.ts → dist/src/capability/agent/index.js} +0 -0
- /package/{src/capability/bot/index.ts → dist/src/capability/bot/index.js} +0 -0
- /package/{src/capability/calendar/index.ts → dist/src/capability/calendar/index.js} +0 -0
- /package/{src/capability/index.ts → dist/src/capability/index.js} +0 -0
|
@@ -0,0 +1,895 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* WeCom Agent Webhook 处理器
|
|
3
|
+
* 处理 XML 格式回调
|
|
4
|
+
*/
|
|
5
|
+
import path from "node:path";
|
|
6
|
+
import { pathToFileURL } from "node:url";
|
|
7
|
+
import { resolveWecomMediaMaxBytes, shouldRejectWecomDefaultRoute } from "../config/index.js";
|
|
8
|
+
import { buildAgentSessionTarget, generateAgentId, shouldUseDynamicAgent, ensureDynamicAgentListed, } from "../dynamic-agent.js";
|
|
9
|
+
import { setPeerContext } from "../context-store.js";
|
|
10
|
+
import { registerWecomSourceSnapshot } from "../runtime/source-registry.js";
|
|
11
|
+
import { buildWecomUnauthorizedCommandPrompt, resolveWecomCommandAuthorization, } from "../shared/command-auth.js";
|
|
12
|
+
import { extractMsgType, extractFromUser, extractContent, extractChatId, extractMediaId, extractMsgId, extractFileName, extractAgentId, extractToUser, } from "../shared/xml-parser.js";
|
|
13
|
+
import { resolveOutboundMediaAsset } from "../shared/media-asset.js";
|
|
14
|
+
import { downloadAgentApiMedia, downloadUpstreamAgentApiMedia, sendAgentApiText, sendUpstreamAgentApiText, } from "../transport/agent-api/client.js";
|
|
15
|
+
import { deliverAgentApiMedia } from "../transport/agent-api/delivery.js";
|
|
16
|
+
import { deliverUpstreamAgentApiMedia } from "../transport/agent-api/upstream-delivery.js";
|
|
17
|
+
import { detectUpstreamUser, createUpstreamAgentConfig, resolveUpstreamCorpConfig } from "../upstream/index.js";
|
|
18
|
+
/** 错误提示信息 */
|
|
19
|
+
const ERROR_HELP = "\n\n遇到问题?联系作者: YanHaidao (微信: YanHaidao)";
|
|
20
|
+
// Agent webhook 幂等去重池(防止企微回调重试导致重复回复)
|
|
21
|
+
// 注意:这是进程内内存去重,重启会清空;但足以覆盖企微的短周期重试。
|
|
22
|
+
const RECENT_MSGID_TTL_MS = 10 * 60 * 1000;
|
|
23
|
+
const recentAgentMsgIds = new Map();
|
|
24
|
+
// Event deduplication (e.g. for ENTER_AGENT/subscribe welcome messages)
|
|
25
|
+
// We only want to send a welcome message once every 5 minutes per user
|
|
26
|
+
const RECENT_EVENT_TTL_MS = 3 * 60 * 1000;
|
|
27
|
+
const recentAgentEvents = new Map();
|
|
28
|
+
function rememberAgentEvent(key) {
|
|
29
|
+
const now = Date.now();
|
|
30
|
+
const existing = recentAgentEvents.get(key);
|
|
31
|
+
if (existing && now - existing < RECENT_EVENT_TTL_MS)
|
|
32
|
+
return false;
|
|
33
|
+
recentAgentEvents.set(key, now);
|
|
34
|
+
// Prune expired
|
|
35
|
+
for (const [k, ts] of recentAgentEvents) {
|
|
36
|
+
if (now - ts >= RECENT_EVENT_TTL_MS)
|
|
37
|
+
recentAgentEvents.delete(k);
|
|
38
|
+
}
|
|
39
|
+
return true;
|
|
40
|
+
}
|
|
41
|
+
function rememberAgentMsgId(msgId) {
|
|
42
|
+
const now = Date.now();
|
|
43
|
+
const existing = recentAgentMsgIds.get(msgId);
|
|
44
|
+
if (existing && now - existing < RECENT_MSGID_TTL_MS)
|
|
45
|
+
return false;
|
|
46
|
+
recentAgentMsgIds.set(msgId, now);
|
|
47
|
+
// 简单清理:只在写入时做一次线性 prune,避免无界增长
|
|
48
|
+
for (const [k, ts] of recentAgentMsgIds) {
|
|
49
|
+
if (now - ts >= RECENT_MSGID_TTL_MS)
|
|
50
|
+
recentAgentMsgIds.delete(k);
|
|
51
|
+
}
|
|
52
|
+
return true;
|
|
53
|
+
}
|
|
54
|
+
function looksLikeTextFile(buffer) {
|
|
55
|
+
const sampleSize = Math.min(buffer.length, 4096);
|
|
56
|
+
if (sampleSize === 0)
|
|
57
|
+
return true;
|
|
58
|
+
let bad = 0;
|
|
59
|
+
for (let i = 0; i < sampleSize; i++) {
|
|
60
|
+
const b = buffer[i];
|
|
61
|
+
const isWhitespace = b === 0x09 || b === 0x0a || b === 0x0d; // \t \n \r
|
|
62
|
+
const isPrintable = b >= 0x20 && b !== 0x7f;
|
|
63
|
+
if (!isWhitespace && !isPrintable)
|
|
64
|
+
bad++;
|
|
65
|
+
}
|
|
66
|
+
// 非可打印字符占比太高,基本可判断为二进制
|
|
67
|
+
return bad / sampleSize <= 0.02;
|
|
68
|
+
}
|
|
69
|
+
function analyzeTextHeuristic(buffer) {
|
|
70
|
+
const sampleSize = Math.min(buffer.length, 4096);
|
|
71
|
+
if (sampleSize === 0)
|
|
72
|
+
return { sampleSize: 0, badCount: 0, badRatio: 0 };
|
|
73
|
+
let badCount = 0;
|
|
74
|
+
for (let i = 0; i < sampleSize; i++) {
|
|
75
|
+
const b = buffer[i];
|
|
76
|
+
const isWhitespace = b === 0x09 || b === 0x0a || b === 0x0d;
|
|
77
|
+
const isPrintable = b >= 0x20 && b !== 0x7f;
|
|
78
|
+
if (!isWhitespace && !isPrintable)
|
|
79
|
+
badCount++;
|
|
80
|
+
}
|
|
81
|
+
return { sampleSize, badCount, badRatio: badCount / sampleSize };
|
|
82
|
+
}
|
|
83
|
+
function previewHex(buffer, maxBytes = 32) {
|
|
84
|
+
const n = Math.min(buffer.length, maxBytes);
|
|
85
|
+
if (n <= 0)
|
|
86
|
+
return "";
|
|
87
|
+
return buffer.subarray(0, n).toString("hex").replace(/(..)/g, "$1 ").trim();
|
|
88
|
+
}
|
|
89
|
+
function buildTextFilePreview(buffer, maxChars) {
|
|
90
|
+
if (!looksLikeTextFile(buffer))
|
|
91
|
+
return undefined;
|
|
92
|
+
const text = buffer.toString("utf8");
|
|
93
|
+
if (!text.trim())
|
|
94
|
+
return undefined;
|
|
95
|
+
const truncated = text.length > maxChars ? `${text.slice(0, maxChars)}\n…(已截断)` : text;
|
|
96
|
+
return truncated;
|
|
97
|
+
}
|
|
98
|
+
function readContextSessionId(ctx) {
|
|
99
|
+
const sessionId = "SessionId" in ctx ? ctx.SessionId : undefined;
|
|
100
|
+
return typeof sessionId === "string" && sessionId.trim() ? sessionId.trim() : undefined;
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* 仅允许“用户意图消息”进入 AI 会话。
|
|
104
|
+
* - event 回调(如 enter_agent/subscribe)不应触发会话与自动回复
|
|
105
|
+
* - 系统发送者(sys)不应触发会话与自动回复
|
|
106
|
+
* - 缺失发送者时默认丢弃,避免写入异常会话
|
|
107
|
+
*/
|
|
108
|
+
export function shouldProcessAgentInboundMessage(params) {
|
|
109
|
+
const msgType = String(params.msgType ?? "")
|
|
110
|
+
.trim()
|
|
111
|
+
.toLowerCase();
|
|
112
|
+
const fromUser = String(params.fromUser ?? "").trim();
|
|
113
|
+
const chatId = String(params.chatId ?? "").trim();
|
|
114
|
+
const normalizedFromUser = fromUser.toLowerCase();
|
|
115
|
+
const eventType = String(params.eventType ?? "")
|
|
116
|
+
.trim()
|
|
117
|
+
.toLowerCase();
|
|
118
|
+
if (msgType === "event") {
|
|
119
|
+
const allowedEvents = [
|
|
120
|
+
"subscribe",
|
|
121
|
+
"enter_agent",
|
|
122
|
+
"batch_job_result",
|
|
123
|
+
// WeCom Doc events
|
|
124
|
+
"doc_create",
|
|
125
|
+
"doc_delete",
|
|
126
|
+
"doc_content_change",
|
|
127
|
+
"doc_member_change",
|
|
128
|
+
// WeCom Form events
|
|
129
|
+
"wedoc_collect_submit",
|
|
130
|
+
// SmartSheet events
|
|
131
|
+
"smartsheet_record_change",
|
|
132
|
+
"smartsheet_field_change",
|
|
133
|
+
"smartsheet_view_change",
|
|
134
|
+
];
|
|
135
|
+
if (allowedEvents.includes(eventType) ||
|
|
136
|
+
eventType.startsWith("doc_") ||
|
|
137
|
+
eventType.startsWith("wedoc_") ||
|
|
138
|
+
eventType.startsWith("smartsheet_")) {
|
|
139
|
+
return {
|
|
140
|
+
shouldProcess: true,
|
|
141
|
+
reason: `allowed_event:${eventType}`,
|
|
142
|
+
};
|
|
143
|
+
}
|
|
144
|
+
return {
|
|
145
|
+
shouldProcess: false,
|
|
146
|
+
reason: `event:${eventType || "unknown"}`,
|
|
147
|
+
};
|
|
148
|
+
}
|
|
149
|
+
if (!fromUser) {
|
|
150
|
+
if (chatId) {
|
|
151
|
+
return {
|
|
152
|
+
shouldProcess: true,
|
|
153
|
+
reason: "missing_sender_but_group_chat",
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
return {
|
|
157
|
+
shouldProcess: false,
|
|
158
|
+
reason: "missing_sender",
|
|
159
|
+
};
|
|
160
|
+
}
|
|
161
|
+
if (normalizedFromUser === "sys") {
|
|
162
|
+
return {
|
|
163
|
+
shouldProcess: false,
|
|
164
|
+
reason: "system_sender",
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
return {
|
|
168
|
+
shouldProcess: true,
|
|
169
|
+
reason: "user_message",
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
export function shouldSuppressAgentReplyText(params) {
|
|
173
|
+
return params.mediaReplySeen && Boolean(params.text.trim());
|
|
174
|
+
}
|
|
175
|
+
function normalizeAgentId(value) {
|
|
176
|
+
if (typeof value === "number" && Number.isFinite(value))
|
|
177
|
+
return value;
|
|
178
|
+
const raw = String(value ?? "").trim();
|
|
179
|
+
if (!raw)
|
|
180
|
+
return undefined;
|
|
181
|
+
const parsed = Number(raw);
|
|
182
|
+
return Number.isFinite(parsed) ? parsed : undefined;
|
|
183
|
+
}
|
|
184
|
+
/**
|
|
185
|
+
* **resolveQueryParams (解析查询参数)**
|
|
186
|
+
*
|
|
187
|
+
* 辅助函数:从 IncomingMessage 中解析 URL 查询字符串,用于获取签名、时间戳等参数。
|
|
188
|
+
*/
|
|
189
|
+
function resolveQueryParams(req) {
|
|
190
|
+
const url = new URL(req.url ?? "/", "http://localhost");
|
|
191
|
+
return url.searchParams;
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* 处理消息回调 (POST)
|
|
195
|
+
*/
|
|
196
|
+
async function handleMessageCallback(params) {
|
|
197
|
+
const { req, res, verifiedPost, agent, config, core, log, error, auditSink } = params;
|
|
198
|
+
try {
|
|
199
|
+
if (!verifiedPost) {
|
|
200
|
+
error?.("[wecom-agent] inbound: missing preverified envelope");
|
|
201
|
+
res.statusCode = 400;
|
|
202
|
+
res.setHeader("Content-Type", "text/plain; charset=utf-8");
|
|
203
|
+
res.end(`invalid request - 缺少上游验签结果${ERROR_HELP}`);
|
|
204
|
+
return true;
|
|
205
|
+
}
|
|
206
|
+
log?.(`[wecom-agent] inbound: method=${req.method ?? "UNKNOWN"} remote=${req.socket?.remoteAddress ?? "unknown"}`);
|
|
207
|
+
const query = resolveQueryParams(req);
|
|
208
|
+
const querySignature = query.get("msg_signature") ?? "";
|
|
209
|
+
const encrypted = verifiedPost.encrypted;
|
|
210
|
+
const decrypted = verifiedPost.decrypted;
|
|
211
|
+
const msg = verifiedPost.parsed;
|
|
212
|
+
const timestamp = verifiedPost.timestamp;
|
|
213
|
+
const nonce = verifiedPost.nonce;
|
|
214
|
+
const signature = verifiedPost.signature || querySignature;
|
|
215
|
+
log?.(`[wecom-agent] inbound: using preverified envelope timestamp=${timestamp ? "yes" : "no"} nonce=${nonce ? "yes" : "no"} msg_signature=${signature ? "yes" : "no"} encryptLen=${encrypted.length}`);
|
|
216
|
+
log?.(`[wecom-agent] inbound: decryptedBytes=${Buffer.byteLength(decrypted, "utf8")}`);
|
|
217
|
+
const inboundAgentId = normalizeAgentId(extractAgentId(msg));
|
|
218
|
+
if (inboundAgentId !== undefined &&
|
|
219
|
+
typeof agent.agentId === "number" &&
|
|
220
|
+
Number.isFinite(agent.agentId) &&
|
|
221
|
+
inboundAgentId !== agent.agentId) {
|
|
222
|
+
error?.(`[wecom-agent] inbound: agentId mismatch ignored expectedAgentId=${agent.agentId} actualAgentId=${String(extractAgentId(msg) ?? "")}`);
|
|
223
|
+
}
|
|
224
|
+
const msgType = extractMsgType(msg);
|
|
225
|
+
const fromUser = extractFromUser(msg);
|
|
226
|
+
const chatId = extractChatId(msg);
|
|
227
|
+
const msgId = extractMsgId(msg);
|
|
228
|
+
const eventType = String(msg.Event ?? "")
|
|
229
|
+
.trim()
|
|
230
|
+
.toLowerCase();
|
|
231
|
+
if (msgId) {
|
|
232
|
+
const ok = rememberAgentMsgId(msgId);
|
|
233
|
+
if (!ok) {
|
|
234
|
+
log?.(`[wecom-agent] duplicate msgId=${msgId} from=${fromUser} chatId=${chatId ?? "N/A"} type=${msgType}; skipped`);
|
|
235
|
+
auditSink?.({
|
|
236
|
+
transport: "agent-callback",
|
|
237
|
+
category: "duplicate-reply",
|
|
238
|
+
messageId: msgId,
|
|
239
|
+
summary: `duplicate agent callback from=${fromUser} chatId=${chatId ?? "N/A"} type=${msgType}`,
|
|
240
|
+
raw: {
|
|
241
|
+
transport: "agent-callback",
|
|
242
|
+
envelopeType: "xml",
|
|
243
|
+
body: msg,
|
|
244
|
+
},
|
|
245
|
+
});
|
|
246
|
+
res.statusCode = 200;
|
|
247
|
+
res.setHeader("Content-Type", "text/plain; charset=utf-8");
|
|
248
|
+
res.end("success");
|
|
249
|
+
return true;
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
// Agent 模式下 enter_agent / subscribe 不做任何处理,静默回 success
|
|
253
|
+
if (msgType === "event" && (eventType === "enter_agent" || eventType === "subscribe")) {
|
|
254
|
+
log?.(`[wecom-agent] ignoring ${eventType} from=${fromUser}; agent does not handle welcome events`);
|
|
255
|
+
res.statusCode = 200;
|
|
256
|
+
res.setHeader("Content-Type", "text/plain; charset=utf-8");
|
|
257
|
+
res.end("success");
|
|
258
|
+
return true;
|
|
259
|
+
}
|
|
260
|
+
const content = String(extractContent(msg) ?? "");
|
|
261
|
+
const preview = content.length > 100 ? `${content.slice(0, 100)}…` : content;
|
|
262
|
+
log?.(`[wecom-agent] ${msgType} from=${fromUser} chatId=${chatId ?? "N/A"} msgId=${msgId ?? "N/A"} content=${preview}`);
|
|
263
|
+
// 先返回 success (Agent 模式使用 API 发送回复,不用被动回复)
|
|
264
|
+
res.statusCode = 200;
|
|
265
|
+
res.setHeader("Content-Type", "text/plain; charset=utf-8");
|
|
266
|
+
res.end("success");
|
|
267
|
+
const decision = shouldProcessAgentInboundMessage({
|
|
268
|
+
msgType,
|
|
269
|
+
fromUser,
|
|
270
|
+
chatId,
|
|
271
|
+
eventType,
|
|
272
|
+
});
|
|
273
|
+
if (!decision.shouldProcess) {
|
|
274
|
+
log?.(`[wecom-agent] skip processing: type=${msgType || "unknown"} event=${eventType || "N/A"} from=${fromUser || "N/A"} reason=${decision.reason}`);
|
|
275
|
+
return true;
|
|
276
|
+
}
|
|
277
|
+
// 异步处理消息
|
|
278
|
+
processAgentMessage({
|
|
279
|
+
agent,
|
|
280
|
+
config,
|
|
281
|
+
core,
|
|
282
|
+
fromUser,
|
|
283
|
+
chatId,
|
|
284
|
+
msgType,
|
|
285
|
+
content,
|
|
286
|
+
msg,
|
|
287
|
+
log,
|
|
288
|
+
error,
|
|
289
|
+
auditSink,
|
|
290
|
+
touchTransportSession: params.touchTransportSession,
|
|
291
|
+
}).catch((err) => {
|
|
292
|
+
error?.(`[wecom-agent] process failed: ${String(err)}`);
|
|
293
|
+
});
|
|
294
|
+
return true;
|
|
295
|
+
}
|
|
296
|
+
catch (err) {
|
|
297
|
+
error?.(`[wecom-agent] callback failed: ${String(err)}`);
|
|
298
|
+
res.statusCode = 400;
|
|
299
|
+
res.setHeader("Content-Type", "text/plain; charset=utf-8");
|
|
300
|
+
res.end(`error - 回调处理失败${ERROR_HELP}`);
|
|
301
|
+
return true;
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
/**
|
|
305
|
+
* **processAgentMessage (处理 Agent 消息)**
|
|
306
|
+
*
|
|
307
|
+
* 异步处理解密后的消息内容,并触发 OpenClaw Agent。
|
|
308
|
+
* 流程:
|
|
309
|
+
* 1. 路由解析:根据 userid或群ID 确定 Agent 路由。
|
|
310
|
+
* 2. 媒体处理:如果是图片/文件等,下载资源。
|
|
311
|
+
* 3. 上下文构建:创建 Inbound Context。
|
|
312
|
+
* 4. 会话记录:更新 Session 状态。
|
|
313
|
+
* 5. 调度回复:将 Agent 的响应通过 `api-client` 发送回企业微信。
|
|
314
|
+
*/
|
|
315
|
+
async function processAgentMessage(params) {
|
|
316
|
+
const { agent, config, core, fromUser, chatId, content, msg, msgType, log, error, auditSink, touchTransportSession, } = params;
|
|
317
|
+
const isGroup = Boolean(chatId);
|
|
318
|
+
const peerId = isGroup ? chatId : fromUser;
|
|
319
|
+
const replyTarget = isGroup
|
|
320
|
+
? { toUser: undefined, chatId: peerId }
|
|
321
|
+
: { toUser: fromUser, chatId: undefined };
|
|
322
|
+
let upstreamAgent;
|
|
323
|
+
let upstreamReplyTarget;
|
|
324
|
+
let primaryAgentForUpstream;
|
|
325
|
+
const eventType = String(msg.Event ?? "")
|
|
326
|
+
.trim()
|
|
327
|
+
.toLowerCase();
|
|
328
|
+
// 检测是否是上下游用户
|
|
329
|
+
const toUserName = extractToUser(msg);
|
|
330
|
+
const isUpstreamUser = detectUpstreamUser({
|
|
331
|
+
messageToUserName: toUserName,
|
|
332
|
+
primaryCorpId: agent.corpId,
|
|
333
|
+
});
|
|
334
|
+
if (isUpstreamUser) {
|
|
335
|
+
log?.(`[wecom-agent] detected upstream user: from=${fromUser} toCorpId=${toUserName}`);
|
|
336
|
+
// 查找上下游配置,构建上游 Agent 配置
|
|
337
|
+
const upstreamConfig = resolveUpstreamCorpConfig({
|
|
338
|
+
upstreamCorpId: toUserName,
|
|
339
|
+
upstreamCorps: agent.config.upstreamCorps,
|
|
340
|
+
});
|
|
341
|
+
if (upstreamConfig) {
|
|
342
|
+
upstreamAgent = createUpstreamAgentConfig({
|
|
343
|
+
baseAgent: agent,
|
|
344
|
+
upstreamCorpId: toUserName,
|
|
345
|
+
upstreamAgentId: upstreamConfig.agentId,
|
|
346
|
+
});
|
|
347
|
+
primaryAgentForUpstream = agent;
|
|
348
|
+
// 上下游的 replyTarget 与普通 DM 一致(toUser = fromUser)
|
|
349
|
+
upstreamReplyTarget = isGroup
|
|
350
|
+
? { toUser: undefined, chatId: peerId }
|
|
351
|
+
: { toUser: fromUser, chatId: undefined };
|
|
352
|
+
}
|
|
353
|
+
else {
|
|
354
|
+
error?.(`[wecom-agent] upstream user detected but no upstream config for corpId=${toUserName}; fallback to primary agent target`);
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
const resolveInboundKind = () => {
|
|
358
|
+
if (msgType === "event") {
|
|
359
|
+
if (eventType === "subscribe" || eventType === "enter_agent")
|
|
360
|
+
return "welcome";
|
|
361
|
+
return "event";
|
|
362
|
+
}
|
|
363
|
+
if (msgType === "image")
|
|
364
|
+
return "image";
|
|
365
|
+
if (msgType === "voice")
|
|
366
|
+
return "voice";
|
|
367
|
+
if (msgType === "video")
|
|
368
|
+
return "video";
|
|
369
|
+
if (msgType === "file")
|
|
370
|
+
return "file";
|
|
371
|
+
if (msgType === "location")
|
|
372
|
+
return "location";
|
|
373
|
+
if (msgType === "link")
|
|
374
|
+
return "link";
|
|
375
|
+
return "text";
|
|
376
|
+
};
|
|
377
|
+
const inboundKind = resolveInboundKind();
|
|
378
|
+
const resolveEventText = () => {
|
|
379
|
+
if (inboundKind === "welcome" && agent.config.welcomeText) {
|
|
380
|
+
return agent.config.welcomeText;
|
|
381
|
+
}
|
|
382
|
+
if (msgType === "event") {
|
|
383
|
+
return `[event:${eventType || "unknown"}]`;
|
|
384
|
+
}
|
|
385
|
+
return content;
|
|
386
|
+
};
|
|
387
|
+
// BUG FIX: 真正调用 resolveEventText() 获取欢迎语或事件描述
|
|
388
|
+
const resolvedContent = resolveEventText();
|
|
389
|
+
let finalContent = resolvedContent;
|
|
390
|
+
const mediaMaxBytes = resolveWecomMediaMaxBytes(config, agent.accountId);
|
|
391
|
+
// 处理媒体文件
|
|
392
|
+
const attachments = [];
|
|
393
|
+
let mediaPath;
|
|
394
|
+
let mediaType;
|
|
395
|
+
if (["image", "voice", "video", "file"].includes(msgType)) {
|
|
396
|
+
const mediaId = extractMediaId(msg);
|
|
397
|
+
if (mediaId) {
|
|
398
|
+
try {
|
|
399
|
+
log?.(`[wecom-agent] downloading media: ${mediaId} (${msgType})`);
|
|
400
|
+
const { buffer, contentType, filename: headerFileName, } = upstreamAgent && primaryAgentForUpstream
|
|
401
|
+
? await downloadUpstreamAgentApiMedia({
|
|
402
|
+
upstreamAgent,
|
|
403
|
+
primaryAgent: primaryAgentForUpstream,
|
|
404
|
+
mediaId,
|
|
405
|
+
maxBytes: mediaMaxBytes,
|
|
406
|
+
})
|
|
407
|
+
: await downloadAgentApiMedia({ agent, mediaId, maxBytes: mediaMaxBytes });
|
|
408
|
+
const xmlFileName = extractFileName(msg);
|
|
409
|
+
const originalFileName = (xmlFileName || headerFileName || `${mediaId}.bin`).trim();
|
|
410
|
+
const heuristic = analyzeTextHeuristic(buffer);
|
|
411
|
+
// 推断文件名后缀
|
|
412
|
+
const extMap = {
|
|
413
|
+
"image/jpeg": "jpg",
|
|
414
|
+
"image/png": "png",
|
|
415
|
+
"image/gif": "gif",
|
|
416
|
+
"audio/amr": "amr",
|
|
417
|
+
"audio/speex": "speex",
|
|
418
|
+
"video/mp4": "mp4",
|
|
419
|
+
};
|
|
420
|
+
const textPreview = msgType === "file" ? buildTextFilePreview(buffer, 12_000) : undefined;
|
|
421
|
+
const looksText = Boolean(textPreview);
|
|
422
|
+
const originalExt = path.extname(originalFileName).toLowerCase();
|
|
423
|
+
const normalizedContentType = looksText && originalExt === ".md"
|
|
424
|
+
? "text/markdown"
|
|
425
|
+
: looksText && (!contentType || contentType === "application/octet-stream")
|
|
426
|
+
? "text/plain; charset=utf-8"
|
|
427
|
+
: contentType;
|
|
428
|
+
const ext = extMap[normalizedContentType] || (looksText ? "txt" : "bin");
|
|
429
|
+
const filename = `${mediaId}.${ext}`;
|
|
430
|
+
log?.(`[wecom-agent] file meta: msgType=${msgType} mediaId=${mediaId} size=${buffer.length} maxBytes=${mediaMaxBytes} ` +
|
|
431
|
+
`contentType=${contentType} normalizedContentType=${normalizedContentType} originalFileName=${originalFileName} ` +
|
|
432
|
+
`xmlFileName=${xmlFileName ?? "N/A"} headerFileName=${headerFileName ?? "N/A"} ` +
|
|
433
|
+
`textHeuristic(sample=${heuristic.sampleSize}, bad=${heuristic.badCount}, ratio=${heuristic.badRatio.toFixed(4)}) ` +
|
|
434
|
+
`headHex="${previewHex(buffer)}"`);
|
|
435
|
+
// 使用 Core SDK 保存媒体文件
|
|
436
|
+
const saved = await core.channel.media.saveMediaBuffer(buffer, normalizedContentType, "inbound", // context/scope
|
|
437
|
+
mediaMaxBytes, // limit
|
|
438
|
+
originalFileName);
|
|
439
|
+
log?.(`[wecom-agent] media saved to: ${saved.path}`);
|
|
440
|
+
mediaPath = saved.path;
|
|
441
|
+
mediaType = normalizedContentType;
|
|
442
|
+
// 构建附件
|
|
443
|
+
attachments.push({
|
|
444
|
+
name: originalFileName,
|
|
445
|
+
contentType: normalizedContentType,
|
|
446
|
+
remoteUrl: pathToFileURL(saved.path).href, // 使用跨平台安全的文件 URL
|
|
447
|
+
});
|
|
448
|
+
// 更新文本提示
|
|
449
|
+
if (textPreview) {
|
|
450
|
+
finalContent = [
|
|
451
|
+
content,
|
|
452
|
+
"",
|
|
453
|
+
"文件内容预览:",
|
|
454
|
+
"```",
|
|
455
|
+
textPreview,
|
|
456
|
+
"```",
|
|
457
|
+
`(已下载 ${buffer.length} 字节)`,
|
|
458
|
+
].join("\n");
|
|
459
|
+
}
|
|
460
|
+
else {
|
|
461
|
+
if (msgType === "file") {
|
|
462
|
+
finalContent = [
|
|
463
|
+
content,
|
|
464
|
+
"",
|
|
465
|
+
`已收到文件:${originalFileName}`,
|
|
466
|
+
`文件类型:${normalizedContentType || contentType || "未知"}`,
|
|
467
|
+
"提示:当前仅对文本/Markdown/JSON/CSV/HTML/PDF(可选)做内容抽取;其他二进制格式请转为 PDF 或复制文本内容。",
|
|
468
|
+
`(已下载 ${buffer.length} 字节)`,
|
|
469
|
+
].join("\n");
|
|
470
|
+
}
|
|
471
|
+
else {
|
|
472
|
+
finalContent = `${content} (已下载 ${buffer.length} 字节)`;
|
|
473
|
+
}
|
|
474
|
+
}
|
|
475
|
+
log?.(`[wecom-agent] file preview: enabled=${looksText} finalContentLen=${finalContent.length} attachments=${attachments.length}`);
|
|
476
|
+
}
|
|
477
|
+
catch (err) {
|
|
478
|
+
error?.(`[wecom-agent] media processing failed: ${String(err)}`);
|
|
479
|
+
auditSink?.({
|
|
480
|
+
transport: "agent-callback",
|
|
481
|
+
category: "runtime-error",
|
|
482
|
+
messageId: extractMsgId(msg) ?? undefined,
|
|
483
|
+
summary: `agent media processing failed mediaId=${mediaId}`,
|
|
484
|
+
raw: {
|
|
485
|
+
transport: "agent-callback",
|
|
486
|
+
envelopeType: "xml",
|
|
487
|
+
body: msg,
|
|
488
|
+
},
|
|
489
|
+
error: err instanceof Error ? err.message : String(err),
|
|
490
|
+
});
|
|
491
|
+
finalContent = [
|
|
492
|
+
content,
|
|
493
|
+
"",
|
|
494
|
+
`媒体处理失败:${String(err)}`,
|
|
495
|
+
`提示:可在 OpenClaw 配置中提高 channels.wecom.mediaMaxMb(当前=${Math.round(mediaMaxBytes / (1024 * 1024))}MB)`,
|
|
496
|
+
"例如:openclaw config set channels.wecom.mediaMaxMb 50",
|
|
497
|
+
].join("\n");
|
|
498
|
+
}
|
|
499
|
+
}
|
|
500
|
+
else {
|
|
501
|
+
const keys = Object.keys(msg ?? {})
|
|
502
|
+
.slice(0, 50)
|
|
503
|
+
.join(",");
|
|
504
|
+
error?.(`[wecom-agent] mediaId not found for ${msgType}; keys=${keys}`);
|
|
505
|
+
}
|
|
506
|
+
}
|
|
507
|
+
// 解析路由
|
|
508
|
+
const route = core.channel.routing.resolveAgentRoute({
|
|
509
|
+
cfg: config,
|
|
510
|
+
channel: "wecom",
|
|
511
|
+
accountId: agent.accountId,
|
|
512
|
+
peer: { kind: isGroup ? "group" : "direct", id: peerId },
|
|
513
|
+
});
|
|
514
|
+
// ===== 动态 Agent 路由注入 =====
|
|
515
|
+
const useDynamicAgent = shouldUseDynamicAgent({
|
|
516
|
+
chatType: isGroup ? "group" : "dm",
|
|
517
|
+
senderId: fromUser,
|
|
518
|
+
config,
|
|
519
|
+
});
|
|
520
|
+
if (shouldRejectWecomDefaultRoute({ cfg: config, matchedBy: route.matchedBy, useDynamicAgent })) {
|
|
521
|
+
const prompt = `当前账号(${agent.accountId})未绑定 OpenClaw Agent,已拒绝回退到默认主智能体。` +
|
|
522
|
+
`请在 bindings 中添加:{"agentId":"你的Agent","match":{"channel":"wecom","accountId":"${agent.accountId}"}}`;
|
|
523
|
+
error?.(`[wecom-agent] routing guard: blocked default fallback accountId=${agent.accountId} matchedBy=${route.matchedBy} from=${fromUser}`);
|
|
524
|
+
try {
|
|
525
|
+
if (upstreamAgent) {
|
|
526
|
+
await sendUpstreamAgentApiText({
|
|
527
|
+
upstreamAgent,
|
|
528
|
+
primaryAgent: primaryAgentForUpstream,
|
|
529
|
+
...(upstreamReplyTarget ?? replyTarget),
|
|
530
|
+
text: prompt,
|
|
531
|
+
});
|
|
532
|
+
}
|
|
533
|
+
else {
|
|
534
|
+
await sendAgentApiText({ agent, ...replyTarget, text: prompt });
|
|
535
|
+
}
|
|
536
|
+
touchTransportSession?.({ lastOutboundAt: Date.now(), running: true });
|
|
537
|
+
log?.(`[wecom-agent] routing guard prompt delivered to ${fromUser}`);
|
|
538
|
+
}
|
|
539
|
+
catch (err) {
|
|
540
|
+
error?.(`[wecom-agent] routing guard prompt failed: ${String(err)}`);
|
|
541
|
+
auditSink?.({
|
|
542
|
+
transport: "agent-callback",
|
|
543
|
+
category: "fallback-delivery-failed",
|
|
544
|
+
summary: `routing guard prompt failed user=${fromUser}`,
|
|
545
|
+
raw: {
|
|
546
|
+
transport: "agent-callback",
|
|
547
|
+
envelopeType: "xml",
|
|
548
|
+
body: msg,
|
|
549
|
+
},
|
|
550
|
+
error: err instanceof Error ? err.message : String(err),
|
|
551
|
+
});
|
|
552
|
+
}
|
|
553
|
+
return;
|
|
554
|
+
}
|
|
555
|
+
if (useDynamicAgent) {
|
|
556
|
+
const targetAgentId = generateAgentId(isGroup ? "group" : "dm", peerId, agent.accountId);
|
|
557
|
+
route.agentId = targetAgentId;
|
|
558
|
+
route.sessionKey = `agent:${targetAgentId}:wecom:${agent.accountId}:${isGroup ? "group" : "dm"}:${peerId}`;
|
|
559
|
+
// 异步添加到 agents.list(不阻塞)
|
|
560
|
+
ensureDynamicAgentListed(targetAgentId, core).catch(() => { });
|
|
561
|
+
log?.(`[wecom-agent] dynamic agent routing: ${targetAgentId}, sessionKey=${route.sessionKey}`);
|
|
562
|
+
}
|
|
563
|
+
// ===== 动态 Agent 路由注入结束 =====
|
|
564
|
+
registerWecomSourceSnapshot({
|
|
565
|
+
accountId: agent.accountId,
|
|
566
|
+
source: "agent-callback",
|
|
567
|
+
messageId: extractMsgId(msg) ?? undefined,
|
|
568
|
+
sessionKey: route.sessionKey,
|
|
569
|
+
peerKind: isGroup ? "group" : "direct",
|
|
570
|
+
peerId,
|
|
571
|
+
});
|
|
572
|
+
// 构建上下文
|
|
573
|
+
const fromLabel = isGroup ? `group:${peerId}` : `user:${fromUser}`;
|
|
574
|
+
const storePath = core.channel.session.resolveStorePath(config.session?.store, {
|
|
575
|
+
agentId: route.agentId,
|
|
576
|
+
});
|
|
577
|
+
const envelopeOptions = core.channel.reply.resolveEnvelopeFormatOptions(config);
|
|
578
|
+
const previousTimestamp = core.channel.session.readSessionUpdatedAt({
|
|
579
|
+
storePath,
|
|
580
|
+
sessionKey: route.sessionKey,
|
|
581
|
+
});
|
|
582
|
+
const body = core.channel.reply.formatAgentEnvelope({
|
|
583
|
+
channel: "WeCom",
|
|
584
|
+
from: fromLabel,
|
|
585
|
+
previousTimestamp,
|
|
586
|
+
envelope: envelopeOptions,
|
|
587
|
+
body: finalContent,
|
|
588
|
+
});
|
|
589
|
+
const authz = await resolveWecomCommandAuthorization({
|
|
590
|
+
core,
|
|
591
|
+
cfg: config,
|
|
592
|
+
// Agent 门禁应读取 channels.wecom.agent.dm(即 agent.config.dm),而不是 channels.wecom.dm(不存在)
|
|
593
|
+
accountConfig: agent.config,
|
|
594
|
+
rawBody: finalContent,
|
|
595
|
+
senderUserId: fromUser,
|
|
596
|
+
});
|
|
597
|
+
log?.(`[wecom-agent] authz: dmPolicy=${authz.dmPolicy} shouldCompute=${authz.shouldComputeAuth} sender=${fromUser.toLowerCase()} senderAllowed=${authz.senderAllowed} authorizerConfigured=${authz.authorizerConfigured} commandAuthorized=${String(authz.commandAuthorized)}`);
|
|
598
|
+
// 命令门禁:未授权时必须明确回复(Agent 侧用私信提示)
|
|
599
|
+
if (authz.shouldComputeAuth && authz.commandAuthorized !== true) {
|
|
600
|
+
const prompt = buildWecomUnauthorizedCommandPrompt({
|
|
601
|
+
senderUserId: fromUser,
|
|
602
|
+
dmPolicy: authz.dmPolicy,
|
|
603
|
+
scope: "agent",
|
|
604
|
+
});
|
|
605
|
+
try {
|
|
606
|
+
if (upstreamAgent) {
|
|
607
|
+
await sendUpstreamAgentApiText({
|
|
608
|
+
upstreamAgent,
|
|
609
|
+
primaryAgent: primaryAgentForUpstream,
|
|
610
|
+
...(upstreamReplyTarget ?? replyTarget),
|
|
611
|
+
text: prompt,
|
|
612
|
+
});
|
|
613
|
+
}
|
|
614
|
+
else {
|
|
615
|
+
await sendAgentApiText({ agent, ...replyTarget, text: prompt });
|
|
616
|
+
}
|
|
617
|
+
touchTransportSession?.({ lastOutboundAt: Date.now(), running: true });
|
|
618
|
+
log?.(`[wecom-agent] unauthorized command: replied to ${isGroup ? `chat:${peerId}` : fromUser}`);
|
|
619
|
+
}
|
|
620
|
+
catch (err) {
|
|
621
|
+
error?.(`[wecom-agent] unauthorized command reply failed: ${String(err)}`);
|
|
622
|
+
auditSink?.({
|
|
623
|
+
transport: "agent-callback",
|
|
624
|
+
category: "fallback-delivery-failed",
|
|
625
|
+
summary: `unauthorized prompt failed user=${fromUser}`,
|
|
626
|
+
raw: {
|
|
627
|
+
transport: "agent-callback",
|
|
628
|
+
envelopeType: "xml",
|
|
629
|
+
body: msg,
|
|
630
|
+
},
|
|
631
|
+
error: err instanceof Error ? err.message : String(err),
|
|
632
|
+
});
|
|
633
|
+
}
|
|
634
|
+
return;
|
|
635
|
+
}
|
|
636
|
+
const ctxPayload = core.channel.reply.finalizeInboundContext({
|
|
637
|
+
Body: body,
|
|
638
|
+
RawBody: finalContent,
|
|
639
|
+
CommandBody: finalContent,
|
|
640
|
+
Attachments: attachments.length > 0 ? attachments : undefined,
|
|
641
|
+
From: isGroup ? `wecom:group:${peerId}` : `wecom:user:${fromUser}`,
|
|
642
|
+
To: isGroup ? `wecom:group:${peerId}` : `wecom:user:${fromUser}`,
|
|
643
|
+
SessionKey: route.sessionKey,
|
|
644
|
+
AccountId: route.accountId,
|
|
645
|
+
ChatType: isGroup ? "group" : "direct",
|
|
646
|
+
ConversationLabel: fromLabel,
|
|
647
|
+
SenderName: fromUser,
|
|
648
|
+
SenderId: fromUser,
|
|
649
|
+
Provider: "wecom",
|
|
650
|
+
Surface: "webchat",
|
|
651
|
+
OriginatingChannel: "wecom",
|
|
652
|
+
// 标记为 Agent 会话的回复路由目标,避免与 Bot 会话混淆:
|
|
653
|
+
// - 用于让 /new /reset 这类命令回执不被 Bot 侧策略拦截
|
|
654
|
+
// - 群聊场景也统一路由为私信触发者(与 deliver 策略一致)
|
|
655
|
+
OriginatingTo: buildAgentSessionTarget(fromUser, agent.accountId),
|
|
656
|
+
CommandAuthorized: authz.commandAuthorized ?? true,
|
|
657
|
+
MediaPath: mediaPath,
|
|
658
|
+
MediaType: mediaType,
|
|
659
|
+
MediaUrl: mediaPath,
|
|
660
|
+
});
|
|
661
|
+
const sessionId = readContextSessionId(ctxPayload);
|
|
662
|
+
log?.(`[wecom-agent] session bound: sessionKey=${ctxPayload.SessionKey ?? route.sessionKey} sessionId=${sessionId ?? "N/A"} peer=${peerId} upstream=${String(Boolean(upstreamAgent))}`);
|
|
663
|
+
registerWecomSourceSnapshot({
|
|
664
|
+
accountId: agent.accountId,
|
|
665
|
+
source: "agent-callback",
|
|
666
|
+
messageId: extractMsgId(msg) ?? undefined,
|
|
667
|
+
sessionKey: ctxPayload.SessionKey ?? route.sessionKey,
|
|
668
|
+
sessionId,
|
|
669
|
+
peerKind: isGroup ? "group" : "direct",
|
|
670
|
+
peerId,
|
|
671
|
+
upstreamCorpId: upstreamAgent?.corpId,
|
|
672
|
+
});
|
|
673
|
+
setPeerContext(agent.accountId, peerId, {
|
|
674
|
+
peerKind: isGroup ? "group" : "direct",
|
|
675
|
+
lastSeen: Date.now(),
|
|
676
|
+
upstreamCorpId: upstreamAgent?.corpId,
|
|
677
|
+
});
|
|
678
|
+
// 记录会话
|
|
679
|
+
await core.channel.session.recordInboundSession({
|
|
680
|
+
storePath,
|
|
681
|
+
sessionKey: ctxPayload.SessionKey ?? route.sessionKey,
|
|
682
|
+
ctx: ctxPayload,
|
|
683
|
+
onRecordError: (err) => {
|
|
684
|
+
error?.(`[wecom-agent] session record failed: ${String(err)}`);
|
|
685
|
+
},
|
|
686
|
+
});
|
|
687
|
+
// 5秒无响应自动回复进度提示
|
|
688
|
+
let hasResponseSent = false;
|
|
689
|
+
const effectiveAgent = upstreamAgent ?? agent;
|
|
690
|
+
const effectiveReplyTarget = upstreamReplyTarget ?? replyTarget;
|
|
691
|
+
const processingTimer = setTimeout(async () => {
|
|
692
|
+
if (hasResponseSent)
|
|
693
|
+
return;
|
|
694
|
+
try {
|
|
695
|
+
if (upstreamAgent && primaryAgentForUpstream) {
|
|
696
|
+
await sendUpstreamAgentApiText({
|
|
697
|
+
upstreamAgent,
|
|
698
|
+
primaryAgent: primaryAgentForUpstream,
|
|
699
|
+
...effectiveReplyTarget,
|
|
700
|
+
text: "正在处理中,请稍候...",
|
|
701
|
+
});
|
|
702
|
+
}
|
|
703
|
+
else {
|
|
704
|
+
await sendAgentApiText({
|
|
705
|
+
agent: effectiveAgent,
|
|
706
|
+
...effectiveReplyTarget,
|
|
707
|
+
text: "正在处理中,请稍候...",
|
|
708
|
+
});
|
|
709
|
+
}
|
|
710
|
+
log?.(`[wecom-agent] sent processing notification to ${isGroup ? `chat:${peerId}` : fromUser}`);
|
|
711
|
+
}
|
|
712
|
+
catch (err) {
|
|
713
|
+
error?.(`[wecom-agent] failed to send processing notification: ${String(err)}`);
|
|
714
|
+
}
|
|
715
|
+
}, 5000);
|
|
716
|
+
// 发送队列锁:确保所有 deliver 调用(以及内部的分片发送)严格串行执行
|
|
717
|
+
let messageSendQueue = Promise.resolve();
|
|
718
|
+
let deferredMediaUrls = [];
|
|
719
|
+
const mergeDeferredMediaUrls = (mediaUrls) => {
|
|
720
|
+
if (mediaUrls.length === 0) {
|
|
721
|
+
return deferredMediaUrls;
|
|
722
|
+
}
|
|
723
|
+
const merged = [...deferredMediaUrls];
|
|
724
|
+
for (const mediaUrl of mediaUrls) {
|
|
725
|
+
if (!merged.includes(mediaUrl)) {
|
|
726
|
+
merged.push(mediaUrl);
|
|
727
|
+
}
|
|
728
|
+
}
|
|
729
|
+
deferredMediaUrls = merged;
|
|
730
|
+
return deferredMediaUrls;
|
|
731
|
+
};
|
|
732
|
+
const replyWecomTarget = effectiveReplyTarget.chatId
|
|
733
|
+
? { chatid: effectiveReplyTarget.chatId }
|
|
734
|
+
: { touser: effectiveReplyTarget.toUser };
|
|
735
|
+
try {
|
|
736
|
+
// 调度回复
|
|
737
|
+
await core.channel.reply.dispatchReplyWithBufferedBlockDispatcher({
|
|
738
|
+
ctx: ctxPayload,
|
|
739
|
+
cfg: config,
|
|
740
|
+
replyOptions: {
|
|
741
|
+
disableBlockStreaming: false,
|
|
742
|
+
},
|
|
743
|
+
dispatcherOptions: {
|
|
744
|
+
deliver: async (payload, info) => {
|
|
745
|
+
const text = payload.text ?? "";
|
|
746
|
+
const incomingMediaUrls = payload.mediaUrls || (payload.mediaUrl ? [payload.mediaUrl] : []);
|
|
747
|
+
if (info.kind !== "final" && incomingMediaUrls.length > 0) {
|
|
748
|
+
mergeDeferredMediaUrls(incomingMediaUrls);
|
|
749
|
+
}
|
|
750
|
+
const mediaUrls = info.kind === "final"
|
|
751
|
+
? mergeDeferredMediaUrls(incomingMediaUrls)
|
|
752
|
+
: incomingMediaUrls;
|
|
753
|
+
const outboundText = text;
|
|
754
|
+
if ((!outboundText || !outboundText.trim()) && mediaUrls.length === 0) {
|
|
755
|
+
return;
|
|
756
|
+
}
|
|
757
|
+
// 标记已有回复,清除/失效定时器
|
|
758
|
+
hasResponseSent = true;
|
|
759
|
+
clearTimeout(processingTimer);
|
|
760
|
+
// 将本次发送任务加入队列
|
|
761
|
+
// 即使 deliver 被并发调用,队列中的任务也会按入队顺序串行执行
|
|
762
|
+
const currentTask = async () => {
|
|
763
|
+
const MAX_CHUNK_SIZE = 600;
|
|
764
|
+
// 确保分片顺序发送
|
|
765
|
+
for (let i = 0; i < outboundText.length; i += MAX_CHUNK_SIZE) {
|
|
766
|
+
const chunk = outboundText.slice(i, i + MAX_CHUNK_SIZE);
|
|
767
|
+
try {
|
|
768
|
+
if (upstreamAgent) {
|
|
769
|
+
await sendUpstreamAgentApiText({
|
|
770
|
+
upstreamAgent,
|
|
771
|
+
primaryAgent: primaryAgentForUpstream,
|
|
772
|
+
...effectiveReplyTarget,
|
|
773
|
+
text: chunk,
|
|
774
|
+
});
|
|
775
|
+
}
|
|
776
|
+
else {
|
|
777
|
+
await sendAgentApiText({ agent: effectiveAgent, ...effectiveReplyTarget, text: chunk });
|
|
778
|
+
}
|
|
779
|
+
touchTransportSession?.({ lastOutboundAt: Date.now(), running: true });
|
|
780
|
+
log?.(`[wecom-agent] reply chunk delivered (${info.kind}) to ${isGroup ? `chat:${peerId}` : fromUser}, len=${chunk.length}, sessionKey=${ctxPayload.SessionKey ?? route.sessionKey}, sessionId=${sessionId ?? "N/A"}`);
|
|
781
|
+
// 强制延时:确保企业微信有足够时间处理顺序(优化:200ms → 50ms)
|
|
782
|
+
if (i + MAX_CHUNK_SIZE < outboundText.length) {
|
|
783
|
+
await new Promise((resolve) => setTimeout(resolve, 50));
|
|
784
|
+
}
|
|
785
|
+
}
|
|
786
|
+
catch (err) {
|
|
787
|
+
const message = err instanceof Error
|
|
788
|
+
? `${err.message}${err.cause ? ` (cause: ${String(err.cause)})` : ""}`
|
|
789
|
+
: String(err);
|
|
790
|
+
error?.(`[wecom-agent] reply failed: ${message}`);
|
|
791
|
+
auditSink?.({
|
|
792
|
+
transport: "agent-callback",
|
|
793
|
+
category: "fallback-delivery-failed",
|
|
794
|
+
summary: `agent callback reply failed user=${fromUser} kind=${info.kind}`,
|
|
795
|
+
raw: {
|
|
796
|
+
transport: "agent-callback",
|
|
797
|
+
envelopeType: "xml",
|
|
798
|
+
body: msg,
|
|
799
|
+
},
|
|
800
|
+
error: message,
|
|
801
|
+
});
|
|
802
|
+
}
|
|
803
|
+
}
|
|
804
|
+
if (info.kind === "final") {
|
|
805
|
+
for (const mediaUrl of mediaUrls) {
|
|
806
|
+
try {
|
|
807
|
+
const media = await resolveOutboundMediaAsset({
|
|
808
|
+
mediaUrl,
|
|
809
|
+
network: effectiveAgent.network,
|
|
810
|
+
});
|
|
811
|
+
if (upstreamAgent) {
|
|
812
|
+
await deliverUpstreamAgentApiMedia({
|
|
813
|
+
upstreamAgent,
|
|
814
|
+
primaryAgent: primaryAgentForUpstream,
|
|
815
|
+
target: replyWecomTarget,
|
|
816
|
+
buffer: media.buffer,
|
|
817
|
+
filename: media.filename,
|
|
818
|
+
contentType: media.contentType,
|
|
819
|
+
});
|
|
820
|
+
}
|
|
821
|
+
else {
|
|
822
|
+
await deliverAgentApiMedia({
|
|
823
|
+
agent: effectiveAgent,
|
|
824
|
+
target: replyWecomTarget,
|
|
825
|
+
buffer: media.buffer,
|
|
826
|
+
filename: media.filename,
|
|
827
|
+
contentType: media.contentType,
|
|
828
|
+
});
|
|
829
|
+
}
|
|
830
|
+
touchTransportSession?.({ lastOutboundAt: Date.now(), running: true });
|
|
831
|
+
log?.(`[wecom-agent] reply media delivered (${info.kind}) to ${isGroup ? `chat:${peerId}` : fromUser}, media=${media.filename}, sessionKey=${ctxPayload.SessionKey ?? route.sessionKey}, sessionId=${sessionId ?? "N/A"}`);
|
|
832
|
+
}
|
|
833
|
+
catch (err) {
|
|
834
|
+
const message = err instanceof Error
|
|
835
|
+
? `${err.message}${err.cause ? ` (cause: ${String(err.cause)})` : ""}`
|
|
836
|
+
: String(err);
|
|
837
|
+
error?.(`[wecom-agent] media reply failed: ${message}`);
|
|
838
|
+
auditSink?.({
|
|
839
|
+
transport: "agent-callback",
|
|
840
|
+
category: "fallback-delivery-failed",
|
|
841
|
+
summary: `agent callback media reply failed user=${fromUser} kind=${info.kind}`,
|
|
842
|
+
raw: {
|
|
843
|
+
transport: "agent-callback",
|
|
844
|
+
envelopeType: "xml",
|
|
845
|
+
body: msg,
|
|
846
|
+
},
|
|
847
|
+
error: message,
|
|
848
|
+
});
|
|
849
|
+
}
|
|
850
|
+
}
|
|
851
|
+
deferredMediaUrls = [];
|
|
852
|
+
}
|
|
853
|
+
// 不同 Block 之间也增加一点间隔(优化:200ms → 50ms)
|
|
854
|
+
if (info.kind !== "final") {
|
|
855
|
+
await new Promise((resolve) => setTimeout(resolve, 50));
|
|
856
|
+
}
|
|
857
|
+
};
|
|
858
|
+
// 更新队列链
|
|
859
|
+
// 使用 then 链接,并捕获前一个任务可能的错误,确保当前任务总能执行
|
|
860
|
+
messageSendQueue = messageSendQueue
|
|
861
|
+
.then(() => currentTask())
|
|
862
|
+
.catch((err) => {
|
|
863
|
+
error?.(`[wecom-agent] previous send task failed: ${String(err)}`);
|
|
864
|
+
// 前一个失败不应阻止当前任务,继续尝试执行当前任务
|
|
865
|
+
return currentTask();
|
|
866
|
+
});
|
|
867
|
+
// 等待当前任务完成(保持背压,虽然对于 http callback 模式这可能只是延迟了整体结束时间)
|
|
868
|
+
await messageSendQueue;
|
|
869
|
+
},
|
|
870
|
+
onError: (err, info) => {
|
|
871
|
+
clearTimeout(processingTimer);
|
|
872
|
+
error?.(`[wecom-agent] ${info.kind} reply error: ${String(err)}`);
|
|
873
|
+
},
|
|
874
|
+
},
|
|
875
|
+
});
|
|
876
|
+
}
|
|
877
|
+
finally {
|
|
878
|
+
clearTimeout(processingTimer);
|
|
879
|
+
// 确保所有排队的消息都发完了才退出(虽然对于 HTTP 响应来说,res.end 早就调用了)
|
|
880
|
+
await messageSendQueue;
|
|
881
|
+
}
|
|
882
|
+
}
|
|
883
|
+
/**
|
|
884
|
+
* **handleAgentWebhook (Agent Webhook 入口)**
|
|
885
|
+
*
|
|
886
|
+
* 统一处理 Agent 模式的 POST 消息回调请求。
|
|
887
|
+
* URL 验证与验签/解密由 monitor 层统一处理后再调用本函数。
|
|
888
|
+
*/
|
|
889
|
+
export async function handleAgentWebhook(params) {
|
|
890
|
+
const { req } = params;
|
|
891
|
+
if (req.method === "POST") {
|
|
892
|
+
return handleMessageCallback(params);
|
|
893
|
+
}
|
|
894
|
+
return false;
|
|
895
|
+
}
|