@xuanyue202/wechat-mp 2026.3.21

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 (95) hide show
  1. package/README.md +74 -0
  2. package/dist/index.d.ts +20 -0
  3. package/dist/index.d.ts.map +1 -0
  4. package/dist/index.js +59 -0
  5. package/dist/index.js.map +1 -0
  6. package/dist/src/api.d.ts +101 -0
  7. package/dist/src/api.d.ts.map +1 -0
  8. package/dist/src/api.js +142 -0
  9. package/dist/src/api.js.map +1 -0
  10. package/dist/src/channel.d.ts +296 -0
  11. package/dist/src/channel.d.ts.map +1 -0
  12. package/dist/src/channel.js +341 -0
  13. package/dist/src/channel.js.map +1 -0
  14. package/dist/src/config.d.ts +69 -0
  15. package/dist/src/config.d.ts.map +1 -0
  16. package/dist/src/config.js +167 -0
  17. package/dist/src/config.js.map +1 -0
  18. package/dist/src/crypto.d.ts +117 -0
  19. package/dist/src/crypto.d.ts.map +1 -0
  20. package/dist/src/crypto.js +270 -0
  21. package/dist/src/crypto.js.map +1 -0
  22. package/dist/src/crypto.test.d.ts +2 -0
  23. package/dist/src/crypto.test.d.ts.map +1 -0
  24. package/dist/src/crypto.test.js +76 -0
  25. package/dist/src/crypto.test.js.map +1 -0
  26. package/dist/src/dispatch.d.ts +15 -0
  27. package/dist/src/dispatch.d.ts.map +1 -0
  28. package/dist/src/dispatch.js +193 -0
  29. package/dist/src/dispatch.js.map +1 -0
  30. package/dist/src/dispatch.test.d.ts +2 -0
  31. package/dist/src/dispatch.test.d.ts.map +1 -0
  32. package/dist/src/dispatch.test.js +231 -0
  33. package/dist/src/dispatch.test.js.map +1 -0
  34. package/dist/src/inbound.d.ts +7 -0
  35. package/dist/src/inbound.d.ts.map +1 -0
  36. package/dist/src/inbound.js +82 -0
  37. package/dist/src/inbound.js.map +1 -0
  38. package/dist/src/onboarding.d.ts +25 -0
  39. package/dist/src/onboarding.d.ts.map +1 -0
  40. package/dist/src/onboarding.js +49 -0
  41. package/dist/src/onboarding.js.map +1 -0
  42. package/dist/src/outbound.d.ts +17 -0
  43. package/dist/src/outbound.d.ts.map +1 -0
  44. package/dist/src/outbound.js +55 -0
  45. package/dist/src/outbound.js.map +1 -0
  46. package/dist/src/outbound.test.d.ts +2 -0
  47. package/dist/src/outbound.test.d.ts.map +1 -0
  48. package/dist/src/outbound.test.js +175 -0
  49. package/dist/src/outbound.test.js.map +1 -0
  50. package/dist/src/probe.d.ts +15 -0
  51. package/dist/src/probe.d.ts.map +1 -0
  52. package/dist/src/probe.js +55 -0
  53. package/dist/src/probe.js.map +1 -0
  54. package/dist/src/runtime.d.ts +22 -0
  55. package/dist/src/runtime.d.ts.map +1 -0
  56. package/dist/src/runtime.js +33 -0
  57. package/dist/src/runtime.js.map +1 -0
  58. package/dist/src/send.d.ts +27 -0
  59. package/dist/src/send.d.ts.map +1 -0
  60. package/dist/src/send.js +103 -0
  61. package/dist/src/send.js.map +1 -0
  62. package/dist/src/state.d.ts +7 -0
  63. package/dist/src/state.d.ts.map +1 -0
  64. package/dist/src/state.js +109 -0
  65. package/dist/src/state.js.map +1 -0
  66. package/dist/src/text.d.ts +46 -0
  67. package/dist/src/text.d.ts.map +1 -0
  68. package/dist/src/text.js +192 -0
  69. package/dist/src/text.js.map +1 -0
  70. package/dist/src/text.test.d.ts +2 -0
  71. package/dist/src/text.test.d.ts.map +1 -0
  72. package/dist/src/text.test.js +110 -0
  73. package/dist/src/text.test.js.map +1 -0
  74. package/dist/src/token.d.ts +40 -0
  75. package/dist/src/token.d.ts.map +1 -0
  76. package/dist/src/token.js +154 -0
  77. package/dist/src/token.js.map +1 -0
  78. package/dist/src/token.test.d.ts +2 -0
  79. package/dist/src/token.test.d.ts.map +1 -0
  80. package/dist/src/token.test.js +74 -0
  81. package/dist/src/token.test.js.map +1 -0
  82. package/dist/src/types.d.ts +320 -0
  83. package/dist/src/types.d.ts.map +1 -0
  84. package/dist/src/types.js +2 -0
  85. package/dist/src/types.js.map +1 -0
  86. package/dist/src/webhook.d.ts +6 -0
  87. package/dist/src/webhook.d.ts.map +1 -0
  88. package/dist/src/webhook.js +381 -0
  89. package/dist/src/webhook.js.map +1 -0
  90. package/dist/src/webhook.test.d.ts +2 -0
  91. package/dist/src/webhook.test.d.ts.map +1 -0
  92. package/dist/src/webhook.test.js +737 -0
  93. package/dist/src/webhook.test.js.map +1 -0
  94. package/openclaw.plugin.json +83 -0
  95. package/package.json +103 -0
@@ -0,0 +1,76 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { buildWechatMpXml, computeMsgSignature, computeSignature, decryptWechatMpMessage, encryptWechatMpMessage, parseWechatMpXml, verifyMsgSignature, verifySignature, } from "./crypto.js";
3
+ const encodingAESKey = "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG";
4
+ const appId = "wx-test-appid";
5
+ describe("wechat-mp crypto", () => {
6
+ it("verifies callback signatures", () => {
7
+ const signature = computeSignature({
8
+ token: "callback-token",
9
+ timestamp: "1710000000",
10
+ nonce: "nonce-1",
11
+ });
12
+ expect(verifySignature({
13
+ token: "callback-token",
14
+ timestamp: "1710000000",
15
+ nonce: "nonce-1",
16
+ signature,
17
+ })).toBe(true);
18
+ expect(verifySignature({
19
+ token: "callback-token",
20
+ timestamp: "1710000000",
21
+ nonce: "nonce-1",
22
+ signature: "bad-signature",
23
+ })).toBe(false);
24
+ });
25
+ it("encrypts and decrypts safe-mode payloads", () => {
26
+ const plaintext = buildWechatMpXml({
27
+ ToUserName: appId,
28
+ FromUserName: "openid-1",
29
+ CreateTime: "1710000000",
30
+ MsgType: "text",
31
+ Content: "hello",
32
+ MsgId: "msg-1",
33
+ });
34
+ const encrypted = encryptWechatMpMessage({
35
+ encodingAESKey,
36
+ appId,
37
+ plaintext,
38
+ }).encrypt;
39
+ const decrypted = decryptWechatMpMessage({
40
+ encodingAESKey,
41
+ encrypt: encrypted,
42
+ expectedAppId: appId,
43
+ });
44
+ expect(decrypted.plaintext).toBe(plaintext);
45
+ expect(decrypted.appId).toBe(appId);
46
+ });
47
+ it("verifies msg_signature with encrypted payload", () => {
48
+ const signature = computeMsgSignature({
49
+ token: "callback-token",
50
+ timestamp: "1710000000",
51
+ nonce: "nonce-1",
52
+ encrypt: "encrypted-body",
53
+ });
54
+ expect(verifyMsgSignature({
55
+ token: "callback-token",
56
+ timestamp: "1710000000",
57
+ nonce: "nonce-1",
58
+ encrypt: "encrypted-body",
59
+ msgSignature: signature,
60
+ })).toBe(true);
61
+ });
62
+ it("parses xml bodies", () => {
63
+ const xml = buildWechatMpXml({
64
+ ToUserName: appId,
65
+ FromUserName: "openid-1",
66
+ MsgType: "text",
67
+ Content: "hello",
68
+ MsgId: "msg-1",
69
+ });
70
+ const parsed = parseWechatMpXml(xml);
71
+ expect(parsed.ToUserName).toBe(appId);
72
+ expect(parsed.FromUserName).toBe("openid-1");
73
+ expect(parsed.Content).toBe("hello");
74
+ });
75
+ });
76
+ //# sourceMappingURL=crypto.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"crypto.test.js","sourceRoot":"","sources":["../../src/crypto.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAE9C,OAAO,EACL,gBAAgB,EAChB,mBAAmB,EACnB,gBAAgB,EAChB,sBAAsB,EACtB,sBAAsB,EACtB,gBAAgB,EAChB,kBAAkB,EAClB,eAAe,GAChB,MAAM,aAAa,CAAC;AAErB,MAAM,cAAc,GAAG,6CAA6C,CAAC;AACrE,MAAM,KAAK,GAAG,eAAe,CAAC;AAE9B,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;IAChC,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,MAAM,SAAS,GAAG,gBAAgB,CAAC;YACjC,KAAK,EAAE,gBAAgB;YACvB,SAAS,EAAE,YAAY;YACvB,KAAK,EAAE,SAAS;SACjB,CAAC,CAAC;QAEH,MAAM,CACJ,eAAe,CAAC;YACd,KAAK,EAAE,gBAAgB;YACvB,SAAS,EAAE,YAAY;YACvB,KAAK,EAAE,SAAS;YAChB,SAAS;SACV,CAAC,CACH,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACb,MAAM,CACJ,eAAe,CAAC;YACd,KAAK,EAAE,gBAAgB;YACvB,SAAS,EAAE,YAAY;YACvB,KAAK,EAAE,SAAS;YAChB,SAAS,EAAE,eAAe;SAC3B,CAAC,CACH,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAChB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,MAAM,SAAS,GAAG,gBAAgB,CAAC;YACjC,UAAU,EAAE,KAAK;YACjB,YAAY,EAAE,UAAU;YACxB,UAAU,EAAE,YAAY;YACxB,OAAO,EAAE,MAAM;YACf,OAAO,EAAE,OAAO;YAChB,KAAK,EAAE,OAAO;SACf,CAAC,CAAC;QACH,MAAM,SAAS,GAAG,sBAAsB,CAAC;YACvC,cAAc;YACd,KAAK;YACL,SAAS;SACV,CAAC,CAAC,OAAO,CAAC;QAEX,MAAM,SAAS,GAAG,sBAAsB,CAAC;YACvC,cAAc;YACd,OAAO,EAAE,SAAS;YAClB,aAAa,EAAE,KAAK;SACrB,CAAC,CAAC;QAEH,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC5C,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACtC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,+CAA+C,EAAE,GAAG,EAAE;QACvD,MAAM,SAAS,GAAG,mBAAmB,CAAC;YACpC,KAAK,EAAE,gBAAgB;YACvB,SAAS,EAAE,YAAY;YACvB,KAAK,EAAE,SAAS;YAChB,OAAO,EAAE,gBAAgB;SAC1B,CAAC,CAAC;QAEH,MAAM,CACJ,kBAAkB,CAAC;YACjB,KAAK,EAAE,gBAAgB;YACvB,SAAS,EAAE,YAAY;YACvB,KAAK,EAAE,SAAS;YAChB,OAAO,EAAE,gBAAgB;YACzB,YAAY,EAAE,SAAS;SACxB,CAAC,CACH,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACf,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mBAAmB,EAAE,GAAG,EAAE;QAC3B,MAAM,GAAG,GAAG,gBAAgB,CAAC;YAC3B,UAAU,EAAE,KAAK;YACjB,YAAY,EAAE,UAAU;YACxB,OAAO,EAAE,MAAM;YACf,OAAO,EAAE,OAAO;YAChB,KAAK,EAAE,OAAO;SACf,CAAC,CAAC;QACH,MAAM,MAAM,GAAG,gBAAgB,CAAC,GAAG,CAAC,CAAC;QACrC,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACtC,MAAM,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC7C,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACvC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,15 @@
1
+ import type { PluginConfig, PluginRuntime, ResolvedWechatMpAccount, WechatMpInboundCandidate } from "./types.js";
2
+ export declare function dispatchWechatMpCandidate(params: {
3
+ cfg: PluginConfig;
4
+ account: ResolvedWechatMpAccount;
5
+ candidate: WechatMpInboundCandidate;
6
+ runtime: PluginRuntime;
7
+ onChunk?: (text: string) => Promise<void>;
8
+ log?: (message: string) => void;
9
+ error?: (message: string) => void;
10
+ }): Promise<{
11
+ dispatched: boolean;
12
+ reason?: string;
13
+ combinedReply?: string;
14
+ }>;
15
+ //# sourceMappingURL=dispatch.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dispatch.d.ts","sourceRoot":"","sources":["../../src/dispatch.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EACV,YAAY,EACZ,aAAa,EACb,uBAAuB,EACvB,wBAAwB,EACzB,MAAM,YAAY,CAAC;AA4CpB,wBAAsB,yBAAyB,CAAC,MAAM,EAAE;IACtD,GAAG,EAAE,YAAY,CAAC;IAClB,OAAO,EAAE,uBAAuB,CAAC;IACjC,SAAS,EAAE,wBAAwB,CAAC;IACpC,OAAO,EAAE,aAAa,CAAC;IACvB,OAAO,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC1C,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAC;IAChC,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAC;CACnC,GAAG,OAAO,CAAC;IAAE,UAAU,EAAE,OAAO,CAAC;IAAC,MAAM,CAAC,EAAE,MAAM,CAAC;IAAC,aAAa,CAAC,EAAE,MAAM,CAAA;CAAE,CAAC,CAsK5E"}
@@ -0,0 +1,193 @@
1
+ import { resolveAllowFrom } from "./config.js";
2
+ import { updateAccountState } from "./state.js";
3
+ import { normalizeWechatMpText, resolveRenderMarkdown } from "./text.js";
4
+ function createLogger(opts) {
5
+ return {
6
+ info: (message) => (opts.log ?? console.log)(`[wechat-mp] ${message}`),
7
+ warn: (message) => (opts.log ?? console.log)(`[wechat-mp] [WARN] ${message}`),
8
+ error: (message) => (opts.error ?? console.error)(`[wechat-mp] [ERROR] ${message}`),
9
+ };
10
+ }
11
+ function isSenderAllowed(account, senderId) {
12
+ const policy = account.config.dmPolicy ?? "open";
13
+ if (policy === "disabled") {
14
+ return { allowed: false, reason: "dm disabled" };
15
+ }
16
+ if (policy === "allowlist") {
17
+ const allowFrom = resolveAllowFrom(account.config);
18
+ const allowed = allowFrom.includes(senderId.trim().toLowerCase());
19
+ return allowed ? { allowed: true } : { allowed: false, reason: "sender not in allowlist" };
20
+ }
21
+ if (policy === "pairing") {
22
+ const allowFrom = resolveAllowFrom(account.config);
23
+ if (allowFrom.length > 0 && !allowFrom.includes(senderId.trim().toLowerCase())) {
24
+ return { allowed: false, reason: "sender not paired" };
25
+ }
26
+ }
27
+ return { allowed: true };
28
+ }
29
+ function buildCandidateBody(candidate) {
30
+ if (candidate.msgType === "text") {
31
+ return String(candidate.content ?? "").trim();
32
+ }
33
+ const parts = [`[event:${candidate.event ?? "unknown"}]`];
34
+ if (candidate.eventKey) {
35
+ parts.push(`eventKey=${candidate.eventKey}`);
36
+ }
37
+ if (candidate.ticket) {
38
+ parts.push(`ticket=${candidate.ticket}`);
39
+ }
40
+ return parts.join("\n").trim();
41
+ }
42
+ export async function dispatchWechatMpCandidate(params) {
43
+ const logger = createLogger({ log: params.log, error: params.error });
44
+ const { candidate } = params;
45
+ if (!candidate.hasUserIntent) {
46
+ return { dispatched: false, reason: "non-intentful event" };
47
+ }
48
+ const bodyRaw = buildCandidateBody(candidate);
49
+ if (!bodyRaw) {
50
+ return { dispatched: false, reason: "empty inbound body" };
51
+ }
52
+ const policyResult = isSenderAllowed(params.account, candidate.openId);
53
+ if (!policyResult.allowed) {
54
+ logger.info(`skip sender=${candidate.openId} reason=${policyResult.reason ?? "policy rejected"}`);
55
+ return { dispatched: false, reason: policyResult.reason };
56
+ }
57
+ const channel = params.runtime.channel;
58
+ const resolveAgentRoute = channel?.routing?.resolveAgentRoute;
59
+ const dispatchReply = channel?.reply?.dispatchReplyWithBufferedBlockDispatcher;
60
+ if (!resolveAgentRoute || !dispatchReply) {
61
+ const message = "runtime routing or buffered reply dispatcher unavailable";
62
+ logger.warn(message);
63
+ await updateAccountState(params.account.accountId, { lastError: message });
64
+ return { dispatched: false, reason: message };
65
+ }
66
+ const route = resolveAgentRoute({
67
+ cfg: params.cfg,
68
+ channel: "wechat-mp",
69
+ accountId: params.account.accountId,
70
+ peer: { kind: "dm", id: candidate.openId },
71
+ });
72
+ const fromLabel = `user:${candidate.openId}`;
73
+ const from = `wechat-mp:${candidate.target}`;
74
+ const to = candidate.target;
75
+ const effectiveSessionKey = route.sessionKey;
76
+ const storePath = channel.session?.resolveStorePath?.(params.cfg.session?.store, {
77
+ agentId: route.agentId,
78
+ });
79
+ const previousTimestamp = storePath
80
+ ? channel.session?.readSessionUpdatedAt?.({
81
+ storePath,
82
+ sessionKey: effectiveSessionKey,
83
+ })
84
+ : null;
85
+ const envelopeOptions = channel.reply?.resolveEnvelopeFormatOptions?.(params.cfg);
86
+ const body = channel.reply?.formatAgentEnvelope
87
+ ? channel.reply.formatAgentEnvelope({
88
+ channel: "WeChat MP",
89
+ from: fromLabel,
90
+ previousTimestamp: previousTimestamp ?? undefined,
91
+ envelope: envelopeOptions,
92
+ body: bodyRaw,
93
+ })
94
+ : bodyRaw;
95
+ const ctxPayload = channel.reply?.finalizeInboundContext?.({
96
+ Body: body,
97
+ RawBody: bodyRaw,
98
+ CommandBody: bodyRaw,
99
+ From: from,
100
+ To: to,
101
+ SessionKey: effectiveSessionKey,
102
+ AccountId: route.accountId ?? params.account.accountId,
103
+ ChatType: "direct",
104
+ ConversationLabel: fromLabel,
105
+ SenderName: candidate.openId,
106
+ SenderId: candidate.openId,
107
+ Provider: "wechat-mp",
108
+ Surface: "wechat-mp",
109
+ MessageSid: candidate.msgId,
110
+ OriginatingChannel: "wechat-mp",
111
+ OriginatingTo: to,
112
+ EventName: candidate.event,
113
+ EventKey: candidate.eventKey,
114
+ }) ?? {
115
+ Body: body,
116
+ RawBody: bodyRaw,
117
+ CommandBody: bodyRaw,
118
+ From: from,
119
+ To: to,
120
+ SessionKey: effectiveSessionKey,
121
+ AccountId: route.accountId ?? params.account.accountId,
122
+ ChatType: "direct",
123
+ ConversationLabel: fromLabel,
124
+ SenderName: candidate.openId,
125
+ SenderId: candidate.openId,
126
+ Provider: "wechat-mp",
127
+ Surface: "wechat-mp",
128
+ MessageSid: candidate.msgId,
129
+ OriginatingChannel: "wechat-mp",
130
+ OriginatingTo: to,
131
+ EventName: candidate.event,
132
+ EventKey: candidate.eventKey,
133
+ };
134
+ ctxPayload.CommandAuthorized = true;
135
+ if (channel.session?.recordInboundSession && storePath) {
136
+ await channel.session.recordInboundSession({
137
+ storePath,
138
+ sessionKey: String(ctxPayload.SessionKey ?? effectiveSessionKey),
139
+ ctx: ctxPayload,
140
+ updateLastRoute: {
141
+ sessionKey: String((route.mainSessionKey ?? effectiveSessionKey) || effectiveSessionKey),
142
+ channel: "wechat-mp",
143
+ to,
144
+ accountId: route.accountId ?? params.account.accountId,
145
+ },
146
+ onRecordError: (error) => {
147
+ logger.error(`recordInboundSession failed: ${String(error)}`);
148
+ },
149
+ });
150
+ }
151
+ const convertTables = channel.text?.convertMarkdownTables && channel.text?.resolveMarkdownTableMode
152
+ ? (text) => channel.text.convertMarkdownTables(text, channel.text.resolveMarkdownTableMode({
153
+ cfg: params.cfg,
154
+ channel: "wechat-mp",
155
+ accountId: params.account.accountId,
156
+ }))
157
+ : (text) => text;
158
+ const renderMarkdown = resolveRenderMarkdown(params.account.config);
159
+ const responseChunks = [];
160
+ await dispatchReply({
161
+ ctx: ctxPayload,
162
+ cfg: params.cfg,
163
+ dispatcherOptions: {
164
+ deliver: async (payload) => {
165
+ const text = String(payload.text ?? "").trim();
166
+ if (!text)
167
+ return;
168
+ const convertedText = convertTables(text);
169
+ if (!convertedText)
170
+ return;
171
+ const normalizedText = normalizeWechatMpText(convertedText, renderMarkdown);
172
+ if (!normalizedText)
173
+ return;
174
+ if (params.onChunk) {
175
+ await params.onChunk(normalizedText);
176
+ return;
177
+ }
178
+ responseChunks.push(normalizedText);
179
+ },
180
+ onError: (error, info) => {
181
+ logger.error(`${info.kind} reply failed: ${String(error)}`);
182
+ },
183
+ },
184
+ });
185
+ const combinedReply = params.onChunk ? "" : responseChunks.join("\n\n").trim();
186
+ if (combinedReply) {
187
+ await updateAccountState(params.account.accountId, {
188
+ lastOutboundAt: Date.now(),
189
+ });
190
+ }
191
+ return { dispatched: true, combinedReply };
192
+ }
193
+ //# sourceMappingURL=dispatch.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dispatch.js","sourceRoot":"","sources":["../../src/dispatch.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=dispatch.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dispatch.test.d.ts","sourceRoot":"","sources":["../../src/dispatch.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,231 @@
1
+ import { describe, expect, it, vi } from "vitest";
2
+ import { dispatchWechatMpCandidate } from "./dispatch.js";
3
+ function createAccount(overrides) {
4
+ return {
5
+ accountId: "default",
6
+ name: "WeChat MP (default)",
7
+ enabled: true,
8
+ configured: true,
9
+ canSendActive: true,
10
+ config: {
11
+ appId: "wx-test-appid",
12
+ appSecret: "secret",
13
+ token: "token",
14
+ encodingAESKey: "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG",
15
+ webhookPath: "/wechat-mp",
16
+ messageMode: "safe",
17
+ replyMode: "passive",
18
+ dmPolicy: "open",
19
+ },
20
+ ...overrides,
21
+ };
22
+ }
23
+ function createRuntime(replyTexts = ["reply text"]) {
24
+ const resolveAgentRoute = vi.fn(() => ({
25
+ sessionKey: "session-1",
26
+ accountId: "default",
27
+ agentId: "agent-1",
28
+ }));
29
+ const dispatchReplyWithBufferedBlockDispatcher = vi.fn(async (params) => {
30
+ for (const text of replyTexts) {
31
+ await params.dispatcherOptions.deliver({ text });
32
+ }
33
+ });
34
+ const recordInboundSession = vi.fn(async () => undefined);
35
+ const readSessionUpdatedAt = vi.fn(() => null);
36
+ const resolveStorePath = vi.fn(() => "/tmp/session-store");
37
+ const runtime = {
38
+ channel: {
39
+ routing: { resolveAgentRoute },
40
+ reply: {
41
+ dispatchReplyWithBufferedBlockDispatcher,
42
+ },
43
+ session: {
44
+ resolveStorePath,
45
+ readSessionUpdatedAt,
46
+ recordInboundSession,
47
+ },
48
+ },
49
+ };
50
+ return {
51
+ runtime,
52
+ resolveAgentRoute,
53
+ dispatchReplyWithBufferedBlockDispatcher,
54
+ recordInboundSession,
55
+ };
56
+ }
57
+ function createTextCandidate() {
58
+ return {
59
+ accountId: "default",
60
+ openId: "openid-1",
61
+ appId: "wx-test-appid",
62
+ target: "user:openid-1@default",
63
+ sessionKey: "dm:wx-test-appid:openid-1",
64
+ createTime: 1710000000,
65
+ msgType: "text",
66
+ msgId: "msg-1",
67
+ dedupeKey: "msg-1",
68
+ encrypted: true,
69
+ hasUserIntent: true,
70
+ content: "hello",
71
+ toUserName: "gh_xxx",
72
+ raw: {
73
+ ToUserName: "gh_xxx",
74
+ FromUserName: "openid-1",
75
+ CreateTime: 1710000000,
76
+ MsgType: "text",
77
+ Content: "hello",
78
+ MsgId: "msg-1",
79
+ },
80
+ };
81
+ }
82
+ function createEventCandidate(intentful) {
83
+ return {
84
+ accountId: "default",
85
+ openId: "openid-1",
86
+ appId: "wx-test-appid",
87
+ target: "user:openid-1@default",
88
+ sessionKey: "dm:wx-test-appid:openid-1",
89
+ createTime: 1710000000,
90
+ msgType: "event",
91
+ dedupeKey: `event:${intentful ? "click" : "view"}`,
92
+ encrypted: true,
93
+ hasUserIntent: intentful,
94
+ event: intentful ? "click" : "view",
95
+ eventKey: intentful ? "MENU_KEY" : "https://example.com",
96
+ toUserName: "gh_xxx",
97
+ raw: {
98
+ ToUserName: "gh_xxx",
99
+ FromUserName: "openid-1",
100
+ CreateTime: 1710000000,
101
+ MsgType: "event",
102
+ Event: intentful ? "CLICK" : "VIEW",
103
+ EventKey: intentful ? "MENU_KEY" : "https://example.com",
104
+ },
105
+ };
106
+ }
107
+ describe("wechat-mp dispatch", () => {
108
+ it("dispatches text candidate into runtime mainline", async () => {
109
+ const account = createAccount();
110
+ const { runtime, resolveAgentRoute, dispatchReplyWithBufferedBlockDispatcher, recordInboundSession } = createRuntime();
111
+ const result = await dispatchWechatMpCandidate({
112
+ cfg: {},
113
+ account,
114
+ candidate: createTextCandidate(),
115
+ runtime,
116
+ });
117
+ expect(result.dispatched).toBe(true);
118
+ expect(result.combinedReply).toBe("reply text");
119
+ expect(resolveAgentRoute).toHaveBeenCalledTimes(1);
120
+ expect(dispatchReplyWithBufferedBlockDispatcher).toHaveBeenCalledTimes(1);
121
+ expect(recordInboundSession).toHaveBeenCalledTimes(1);
122
+ const calls = recordInboundSession.mock.calls;
123
+ const recordCall = calls[0]?.[0];
124
+ expect(recordCall?.sessionKey).toBe("session-1");
125
+ expect(recordCall?.ctx?.SessionKey).toBe("session-1");
126
+ expect(recordCall?.ctx?.CommandAuthorized).toBe(true);
127
+ });
128
+ it("skips non-intentful event candidate", async () => {
129
+ const account = createAccount();
130
+ const { runtime, resolveAgentRoute, dispatchReplyWithBufferedBlockDispatcher } = createRuntime();
131
+ const result = await dispatchWechatMpCandidate({
132
+ cfg: {},
133
+ account,
134
+ candidate: createEventCandidate(false),
135
+ runtime,
136
+ });
137
+ expect(result.dispatched).toBe(false);
138
+ expect(result.reason).toBe("non-intentful event");
139
+ expect(resolveAgentRoute).not.toHaveBeenCalled();
140
+ expect(dispatchReplyWithBufferedBlockDispatcher).not.toHaveBeenCalled();
141
+ });
142
+ it("dispatches intentful event candidate", async () => {
143
+ const account = createAccount();
144
+ const { runtime, resolveAgentRoute, dispatchReplyWithBufferedBlockDispatcher } = createRuntime();
145
+ const result = await dispatchWechatMpCandidate({
146
+ cfg: {},
147
+ account,
148
+ candidate: createEventCandidate(true),
149
+ runtime,
150
+ });
151
+ expect(result.dispatched).toBe(true);
152
+ expect(resolveAgentRoute).toHaveBeenCalledTimes(1);
153
+ expect(dispatchReplyWithBufferedBlockDispatcher).toHaveBeenCalledTimes(1);
154
+ });
155
+ it("merges delivered chunks by default", async () => {
156
+ const account = createAccount();
157
+ const { runtime } = createRuntime(["step 1", "step 2"]);
158
+ const result = await dispatchWechatMpCandidate({
159
+ cfg: {},
160
+ account,
161
+ candidate: createTextCandidate(),
162
+ runtime,
163
+ });
164
+ expect(result.combinedReply).toBe("step 1\n\nstep 2");
165
+ });
166
+ it("forwards chunks one by one in split mode", async () => {
167
+ const account = createAccount();
168
+ const { runtime } = createRuntime(["step 1", "", "step 2"]);
169
+ const onChunk = vi.fn(async () => undefined);
170
+ const result = await dispatchWechatMpCandidate({
171
+ cfg: {},
172
+ account,
173
+ candidate: createTextCandidate(),
174
+ runtime,
175
+ onChunk,
176
+ });
177
+ expect(onChunk).toHaveBeenCalledTimes(2);
178
+ expect(onChunk).toHaveBeenNthCalledWith(1, "step 1");
179
+ expect(onChunk).toHaveBeenNthCalledWith(2, "step 2");
180
+ expect(result.combinedReply).toBe("");
181
+ });
182
+ it("normalizes markdown text by default in reply path", async () => {
183
+ const account = createAccount();
184
+ const { runtime } = createRuntime(["**bold text** and `code`"]);
185
+ const result = await dispatchWechatMpCandidate({
186
+ cfg: {},
187
+ account,
188
+ candidate: createTextCandidate(),
189
+ runtime,
190
+ });
191
+ // Default renderMarkdown=true strips markdown formatting
192
+ expect(result.combinedReply).toBe("bold text and code");
193
+ });
194
+ it("skips markdown normalization when renderMarkdown is false", async () => {
195
+ const account = createAccount({
196
+ config: {
197
+ appId: "wx-test-appid",
198
+ appSecret: "secret",
199
+ token: "token",
200
+ encodingAESKey: "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG",
201
+ webhookPath: "/wechat-mp",
202
+ messageMode: "safe",
203
+ replyMode: "passive",
204
+ dmPolicy: "open",
205
+ renderMarkdown: false,
206
+ },
207
+ });
208
+ const { runtime } = createRuntime(["**bold text** and `code`"]);
209
+ const result = await dispatchWechatMpCandidate({
210
+ cfg: {},
211
+ account,
212
+ candidate: createTextCandidate(),
213
+ runtime,
214
+ });
215
+ // renderMarkdown=false preserves original markdown
216
+ expect(result.combinedReply).toBe("**bold text** and `code`");
217
+ });
218
+ it("applies markdown normalization to multiple chunks", async () => {
219
+ const account = createAccount();
220
+ const { runtime } = createRuntime(["# Heading", "**bold** text"]);
221
+ const result = await dispatchWechatMpCandidate({
222
+ cfg: {},
223
+ account,
224
+ candidate: createTextCandidate(),
225
+ runtime,
226
+ });
227
+ // Headings become bracketed, bold is stripped
228
+ expect(result.combinedReply).toBe("[Heading]\n\nbold text");
229
+ });
230
+ });
231
+ //# sourceMappingURL=dispatch.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dispatch.test.js","sourceRoot":"","sources":["../../src/dispatch.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ import type { ResolvedWechatMpAccount, WechatMpInboundCandidate, WechatMpInboundMessage } from "./types.js";
2
+ export declare function normalizeWechatMpInbound(params: {
3
+ account: ResolvedWechatMpAccount;
4
+ message: WechatMpInboundMessage;
5
+ encrypted: boolean;
6
+ }): WechatMpInboundCandidate | null;
7
+ //# sourceMappingURL=inbound.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"inbound.d.ts","sourceRoot":"","sources":["../../src/inbound.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EACV,uBAAuB,EACvB,wBAAwB,EAExB,sBAAsB,EACvB,MAAM,YAAY,CAAC;AAmBpB,wBAAgB,wBAAwB,CAAC,MAAM,EAAE;IAC/C,OAAO,EAAE,uBAAuB,CAAC;IACjC,OAAO,EAAE,sBAAsB,CAAC;IAChC,SAAS,EAAE,OAAO,CAAC;CACpB,GAAG,wBAAwB,GAAG,IAAI,CAoElC"}
@@ -0,0 +1,82 @@
1
+ import { buildWechatMpSessionKey, buildWechatMpTarget } from "./config.js";
2
+ const INTENTFUL_EVENTS = new Set(["subscribe", "scan", "click"]);
3
+ const TRACKED_EVENTS = new Set([
4
+ "subscribe",
5
+ "unsubscribe",
6
+ "scan",
7
+ "click",
8
+ "view",
9
+ ]);
10
+ function normalizeEventName(raw) {
11
+ const value = String(raw ?? "").trim().toLowerCase();
12
+ if (TRACKED_EVENTS.has(value)) {
13
+ return value;
14
+ }
15
+ return undefined;
16
+ }
17
+ export function normalizeWechatMpInbound(params) {
18
+ const { account, message, encrypted } = params;
19
+ const openId = String(message.FromUserName ?? "").trim();
20
+ if (!openId)
21
+ return null;
22
+ const createTime = Number(message.CreateTime ?? 0) || 0;
23
+ const target = buildWechatMpTarget(openId, account.accountId);
24
+ const sessionKey = account.config.appId
25
+ ? buildWechatMpSessionKey(account.config.appId, openId)
26
+ : undefined;
27
+ if (message.MsgType === "text") {
28
+ const content = String(message.Content ?? "").trim();
29
+ if (!content)
30
+ return null;
31
+ const msgId = String(message.MsgId ?? "").trim() || undefined;
32
+ return {
33
+ accountId: account.accountId,
34
+ openId,
35
+ appId: account.config.appId,
36
+ target,
37
+ sessionKey,
38
+ createTime,
39
+ msgType: "text",
40
+ msgId,
41
+ dedupeKey: msgId || `text:${account.accountId}:${openId}:${createTime}:${content}`,
42
+ encrypted,
43
+ hasUserIntent: true,
44
+ content,
45
+ toUserName: String(message.ToUserName ?? "").trim() || undefined,
46
+ raw: message,
47
+ };
48
+ }
49
+ if (message.MsgType === "event") {
50
+ const eventValue = "Event" in message && typeof message.Event === "string"
51
+ ? message.Event
52
+ : undefined;
53
+ const event = normalizeEventName(eventValue);
54
+ if (!event)
55
+ return null;
56
+ const eventKey = "EventKey" in message && typeof message.EventKey === "string"
57
+ ? message.EventKey.trim() || undefined
58
+ : undefined;
59
+ const ticket = "Ticket" in message && typeof message.Ticket === "string"
60
+ ? message.Ticket.trim() || undefined
61
+ : undefined;
62
+ return {
63
+ accountId: account.accountId,
64
+ openId,
65
+ appId: account.config.appId,
66
+ target,
67
+ sessionKey,
68
+ createTime,
69
+ msgType: "event",
70
+ dedupeKey: `event:${account.accountId}:${openId}:${event}:${eventKey ?? ""}:${createTime}`,
71
+ encrypted,
72
+ hasUserIntent: INTENTFUL_EVENTS.has(event),
73
+ event,
74
+ eventKey,
75
+ ticket,
76
+ toUserName: String(message.ToUserName ?? "").trim() || undefined,
77
+ raw: message,
78
+ };
79
+ }
80
+ return null;
81
+ }
82
+ //# sourceMappingURL=inbound.js.map