@kodelyth/feishu 2026.5.39 → 2026.5.42

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 (238) hide show
  1. package/api.ts +32 -0
  2. package/channel-entry.ts +20 -0
  3. package/channel-plugin-api.ts +1 -0
  4. package/contract-api.ts +16 -0
  5. package/dist/accounts-D0ow-lRb.js +429 -0
  6. package/dist/api.js +2308 -0
  7. package/dist/app-registration-DBSnysKJ.js +184 -0
  8. package/dist/audio-preflight.runtime-Dpjbn-7r.js +7 -0
  9. package/dist/channel-13WQvQ0u.js +2115 -0
  10. package/dist/channel-entry.js +22 -0
  11. package/dist/channel-plugin-api.js +2 -0
  12. package/dist/channel.runtime-JMJonrJ4.js +729 -0
  13. package/dist/client-D1pzbBGo.js +157 -0
  14. package/dist/contract-api.js +9 -0
  15. package/dist/conversation-id-_58ecqlx.js +139 -0
  16. package/dist/drive-CgHOluXx.js +883 -0
  17. package/dist/index.js +68 -0
  18. package/dist/monitor-oWptK0zL.js +60 -0
  19. package/dist/monitor.account-DHaWlslg.js +5207 -0
  20. package/dist/monitor.state-C211a4tX.js +100 -0
  21. package/dist/probe-CF4duEpK.js +149 -0
  22. package/dist/rolldown-runtime-DUslC3ob.js +14 -0
  23. package/dist/runtime-DSh5rL_d.js +8 -0
  24. package/dist/runtime-api.js +14 -0
  25. package/dist/secret-contract-NSee-WzN.js +119 -0
  26. package/dist/secret-contract-api.js +2 -0
  27. package/dist/security-audit-DWVC0vSK.js +11 -0
  28. package/dist/security-audit-shared-Dpcwxeft.js +38 -0
  29. package/dist/security-contract-api.js +2 -0
  30. package/dist/send-DfZuV4Fi.js +1212 -0
  31. package/dist/session-conversation-Duaukbnl.js +27 -0
  32. package/dist/session-key-api.js +2 -0
  33. package/dist/setup-api.js +2 -0
  34. package/dist/setup-entry.js +15 -0
  35. package/dist/subagent-hooks-Dtegs0kh.js +235 -0
  36. package/dist/subagent-hooks-api.js +23 -0
  37. package/dist/targets-DFskxX4p.js +48 -0
  38. package/dist/thread-bindings-DI7lVSOE.js +222 -0
  39. package/index.ts +82 -0
  40. package/klaw.plugin.json +47 -1712
  41. package/package.json +4 -4
  42. package/runtime-api.ts +52 -0
  43. package/secret-contract-api.ts +5 -0
  44. package/security-contract-api.ts +1 -0
  45. package/session-key-api.ts +1 -0
  46. package/setup-api.ts +3 -0
  47. package/setup-entry.test.ts +19 -0
  48. package/setup-entry.ts +13 -0
  49. package/src/accounts.test.ts +480 -0
  50. package/src/accounts.ts +333 -0
  51. package/src/agent-config.ts +21 -0
  52. package/src/app-registration.ts +331 -0
  53. package/src/approval-auth.test.ts +24 -0
  54. package/src/approval-auth.ts +25 -0
  55. package/src/async.test.ts +35 -0
  56. package/src/async.ts +104 -0
  57. package/src/audio-preflight.runtime.ts +9 -0
  58. package/src/bitable.test.ts +136 -0
  59. package/src/bitable.ts +762 -0
  60. package/src/bot-content.ts +485 -0
  61. package/src/bot-group-name.test.ts +116 -0
  62. package/src/bot-runtime-api.ts +12 -0
  63. package/src/bot-sender-name.ts +125 -0
  64. package/src/bot.broadcast.test.ts +523 -0
  65. package/src/bot.card-action.test.ts +552 -0
  66. package/src/bot.checkBotMentioned.test.ts +265 -0
  67. package/src/bot.helpers.test.ts +135 -0
  68. package/src/bot.stripBotMention.test.ts +126 -0
  69. package/src/bot.test.ts +3671 -0
  70. package/src/bot.ts +1703 -0
  71. package/src/card-action.ts +447 -0
  72. package/src/card-interaction.test.ts +131 -0
  73. package/src/card-interaction.ts +159 -0
  74. package/src/card-test-helpers.ts +54 -0
  75. package/src/card-ux-approval.ts +65 -0
  76. package/src/card-ux-launcher.test.ts +106 -0
  77. package/src/card-ux-launcher.ts +121 -0
  78. package/src/card-ux-shared.ts +33 -0
  79. package/src/channel-runtime-api.ts +16 -0
  80. package/src/channel.runtime.ts +47 -0
  81. package/src/channel.test.ts +1151 -0
  82. package/src/channel.ts +1423 -0
  83. package/src/chat-schema.ts +25 -0
  84. package/src/chat.test.ts +240 -0
  85. package/src/chat.ts +188 -0
  86. package/src/client-timeout.ts +42 -0
  87. package/src/client.test.ts +447 -0
  88. package/src/client.ts +262 -0
  89. package/src/comment-dispatcher-runtime-api.ts +6 -0
  90. package/src/comment-dispatcher.test.ts +185 -0
  91. package/src/comment-dispatcher.ts +107 -0
  92. package/src/comment-handler-runtime-api.ts +3 -0
  93. package/src/comment-handler.test.ts +592 -0
  94. package/src/comment-handler.ts +303 -0
  95. package/src/comment-reaction.test.ts +138 -0
  96. package/src/comment-reaction.ts +259 -0
  97. package/src/comment-shared.test.ts +183 -0
  98. package/src/comment-shared.ts +406 -0
  99. package/src/comment-target.ts +44 -0
  100. package/src/config-schema.test.ts +326 -0
  101. package/src/config-schema.ts +335 -0
  102. package/src/conversation-id.test.ts +18 -0
  103. package/src/conversation-id.ts +199 -0
  104. package/src/dedup-runtime-api.ts +1 -0
  105. package/src/dedup.ts +141 -0
  106. package/src/dedupe-key.ts +72 -0
  107. package/src/directory.static.ts +61 -0
  108. package/src/directory.test.ts +141 -0
  109. package/src/directory.ts +124 -0
  110. package/src/doc-schema.ts +182 -0
  111. package/src/docx-batch-insert.test.ts +116 -0
  112. package/src/docx-batch-insert.ts +223 -0
  113. package/src/docx-color-text.ts +154 -0
  114. package/src/docx-table-ops.test.ts +53 -0
  115. package/src/docx-table-ops.ts +316 -0
  116. package/src/docx-types.ts +38 -0
  117. package/src/docx.account-selection.test.ts +95 -0
  118. package/src/docx.test.ts +701 -0
  119. package/src/docx.ts +1596 -0
  120. package/src/drive-schema.ts +92 -0
  121. package/src/drive.test.ts +1237 -0
  122. package/src/drive.ts +829 -0
  123. package/src/dynamic-agent.test.ts +155 -0
  124. package/src/dynamic-agent.ts +143 -0
  125. package/src/event-types.ts +45 -0
  126. package/src/external-keys.test.ts +20 -0
  127. package/src/external-keys.ts +19 -0
  128. package/src/lifecycle.test-support.ts +220 -0
  129. package/src/media.test.ts +955 -0
  130. package/src/media.ts +1105 -0
  131. package/src/mention-target.types.ts +5 -0
  132. package/src/mention.ts +114 -0
  133. package/src/message-action-contract.ts +13 -0
  134. package/src/monitor-state-runtime-api.ts +7 -0
  135. package/src/monitor-transport-runtime-api.ts +10 -0
  136. package/src/monitor.account.ts +492 -0
  137. package/src/monitor.acp-init-failure.lifecycle.test-support.ts +219 -0
  138. package/src/monitor.bot-identity.ts +86 -0
  139. package/src/monitor.bot-menu-handler.ts +165 -0
  140. package/src/monitor.bot-menu.lifecycle.test-support.ts +224 -0
  141. package/src/monitor.bot-menu.test.ts +188 -0
  142. package/src/monitor.broadcast.reply-once.lifecycle.test-support.ts +264 -0
  143. package/src/monitor.card-action.lifecycle.test-support.ts +421 -0
  144. package/src/monitor.cleanup.test.ts +383 -0
  145. package/src/monitor.comment-notice-handler.ts +105 -0
  146. package/src/monitor.comment.test.ts +967 -0
  147. package/src/monitor.comment.ts +1386 -0
  148. package/src/monitor.lifecycle.test.ts +4 -0
  149. package/src/monitor.message-handler.ts +350 -0
  150. package/src/monitor.reaction.lifecycle.test-support.ts +68 -0
  151. package/src/monitor.reaction.test.ts +739 -0
  152. package/src/monitor.startup.test.ts +213 -0
  153. package/src/monitor.startup.ts +74 -0
  154. package/src/monitor.state.defaults.test.ts +46 -0
  155. package/src/monitor.state.ts +170 -0
  156. package/src/monitor.synthetic-error.ts +18 -0
  157. package/src/monitor.test-mocks.ts +46 -0
  158. package/src/monitor.transport.ts +451 -0
  159. package/src/monitor.ts +100 -0
  160. package/src/monitor.webhook-e2e.test.ts +279 -0
  161. package/src/monitor.webhook-security.test.ts +389 -0
  162. package/src/monitor.webhook.test-helpers.ts +116 -0
  163. package/src/outbound-runtime-api.ts +1 -0
  164. package/src/outbound.test.ts +1118 -0
  165. package/src/outbound.ts +785 -0
  166. package/src/perm-schema.ts +52 -0
  167. package/src/perm.ts +170 -0
  168. package/src/pins.ts +108 -0
  169. package/src/policy.test.ts +223 -0
  170. package/src/policy.ts +318 -0
  171. package/src/post.test.ts +105 -0
  172. package/src/post.ts +275 -0
  173. package/src/probe.test.ts +283 -0
  174. package/src/probe.ts +166 -0
  175. package/src/processing-claims.ts +59 -0
  176. package/src/qr-terminal.ts +1 -0
  177. package/src/reactions.ts +123 -0
  178. package/src/reasoning-preview.test.ts +113 -0
  179. package/src/reasoning-preview.ts +28 -0
  180. package/src/reply-dispatcher-runtime-api.ts +7 -0
  181. package/src/reply-dispatcher.test.ts +1513 -0
  182. package/src/reply-dispatcher.ts +748 -0
  183. package/src/runtime.ts +9 -0
  184. package/src/secret-contract.ts +145 -0
  185. package/src/secret-input.ts +1 -0
  186. package/src/security-audit-shared.ts +69 -0
  187. package/src/security-audit.test.ts +59 -0
  188. package/src/security-audit.ts +1 -0
  189. package/src/send-result.ts +80 -0
  190. package/src/send-target.test.ts +86 -0
  191. package/src/send-target.ts +35 -0
  192. package/src/send.reply-fallback.test.ts +417 -0
  193. package/src/send.test.ts +621 -0
  194. package/src/send.ts +861 -0
  195. package/src/sequential-key.test.ts +72 -0
  196. package/src/sequential-key.ts +25 -0
  197. package/src/sequential-queue.test.ts +165 -0
  198. package/src/sequential-queue.ts +86 -0
  199. package/src/session-conversation.ts +42 -0
  200. package/src/session-route.ts +48 -0
  201. package/src/setup-core.ts +51 -0
  202. package/src/setup-surface.test.ts +484 -0
  203. package/src/setup-surface.ts +618 -0
  204. package/src/streaming-card.test.ts +397 -0
  205. package/src/streaming-card.ts +571 -0
  206. package/src/subagent-hooks.test.ts +627 -0
  207. package/src/subagent-hooks.ts +413 -0
  208. package/src/targets.ts +97 -0
  209. package/src/test-support/lifecycle-test-support.ts +454 -0
  210. package/src/thread-bindings.test.ts +180 -0
  211. package/src/thread-bindings.ts +331 -0
  212. package/src/tool-account-routing.test.ts +250 -0
  213. package/src/tool-account.test.ts +44 -0
  214. package/src/tool-account.ts +93 -0
  215. package/src/tool-factory-test-harness.ts +79 -0
  216. package/src/tool-result.test.ts +32 -0
  217. package/src/tool-result.ts +16 -0
  218. package/src/tools-config.test.ts +21 -0
  219. package/src/tools-config.ts +22 -0
  220. package/src/types.ts +106 -0
  221. package/src/typing.test.ts +144 -0
  222. package/src/typing.ts +214 -0
  223. package/src/wiki-schema.ts +69 -0
  224. package/src/wiki.ts +270 -0
  225. package/subagent-hooks-api.ts +31 -0
  226. package/tsconfig.json +16 -0
  227. package/api.js +0 -7
  228. package/channel-entry.js +0 -7
  229. package/channel-plugin-api.js +0 -7
  230. package/contract-api.js +0 -7
  231. package/index.js +0 -7
  232. package/runtime-api.js +0 -7
  233. package/secret-contract-api.js +0 -7
  234. package/security-contract-api.js +0 -7
  235. package/session-key-api.js +0 -7
  236. package/setup-api.js +0 -7
  237. package/setup-entry.js +0 -7
  238. package/subagent-hooks-api.js +0 -7
@@ -0,0 +1,421 @@
1
+ import { createRuntimeEnv } from "klaw/plugin-sdk/plugin-test-runtime";
2
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
3
+ import "./lifecycle.test-support.js";
4
+ import { resetProcessedFeishuCardActionTokensForTests } from "./card-action.js";
5
+ import { createFeishuCardInteractionEnvelope } from "./card-interaction.js";
6
+ import {
7
+ getFeishuLifecycleTestMocks,
8
+ resetFeishuLifecycleTestMocks,
9
+ } from "./lifecycle.test-support.js";
10
+ import {
11
+ createFeishuLifecycleConfig,
12
+ createFeishuLifecycleReplyDispatcher,
13
+ createResolvedFeishuLifecycleAccount,
14
+ expectFeishuReplyDispatcherSentFinalReplyOnce,
15
+ expectFeishuReplyPipelineDedupedAcrossReplay,
16
+ expectFeishuReplyPipelineDedupedAfterPostSendFailure,
17
+ installFeishuLifecycleReplyRuntime,
18
+ mockFeishuReplyOnceDispatch,
19
+ restoreFeishuLifecycleStateDir,
20
+ setFeishuLifecycleStateDir,
21
+ setupFeishuLifecycleHandler,
22
+ } from "./test-support/lifecycle-test-support.js";
23
+
24
+ const {
25
+ createEventDispatcherMock,
26
+ createFeishuReplyDispatcherMock,
27
+ dispatchReplyFromConfigMock,
28
+ finalizeInboundContextMock,
29
+ resolveAgentRouteMock,
30
+ resolveBoundConversationMock,
31
+ sendCardFeishuMock,
32
+ sendMessageFeishuMock,
33
+ touchBindingMock,
34
+ withReplyDispatcherMock,
35
+ } = getFeishuLifecycleTestMocks();
36
+
37
+ let handlers: Record<string, (data: unknown) => Promise<void>> = {};
38
+ let lastRuntime = createRuntimeEnv();
39
+ const originalStateDir = process.env.KLAW_STATE_DIR;
40
+ const lifecycleConfig = createFeishuLifecycleConfig({
41
+ accountId: "acct-card",
42
+ appId: "cli_test",
43
+ appSecret: "secret_test",
44
+ channelConfig: {
45
+ dmPolicy: "open",
46
+ allowFrom: ["ou_user1"],
47
+ },
48
+ accountConfig: {
49
+ dmPolicy: "open",
50
+ allowFrom: ["ou_user1"],
51
+ },
52
+ });
53
+
54
+ const lifecycleAccount = createResolvedFeishuLifecycleAccount({
55
+ accountId: "acct-card",
56
+ appId: "cli_test",
57
+ appSecret: "secret_test",
58
+ config: {
59
+ dmPolicy: "open",
60
+ allowFrom: ["ou_user1"],
61
+ },
62
+ });
63
+
64
+ function createCardActionEvent(params: {
65
+ token: string;
66
+ action: string;
67
+ command: string;
68
+ chatId?: string;
69
+ chatType?: "group" | "p2p";
70
+ }) {
71
+ const openId = "ou_user1";
72
+ const chatId = params.chatId ?? "p2p:ou_user1";
73
+ const chatType = params.chatType ?? "p2p";
74
+ return {
75
+ operator: {
76
+ open_id: openId,
77
+ user_id: "user_1",
78
+ union_id: "union_1",
79
+ },
80
+ token: params.token,
81
+ action: {
82
+ tag: "button",
83
+ value: createFeishuCardInteractionEnvelope({
84
+ k: "quick",
85
+ a: params.action,
86
+ q: params.command,
87
+ c: {
88
+ u: openId,
89
+ h: chatId,
90
+ t: chatType,
91
+ e: Date.now() + 60_000,
92
+ },
93
+ }),
94
+ },
95
+ context: {
96
+ open_id: openId,
97
+ user_id: "user_1",
98
+ chat_id: chatId,
99
+ },
100
+ };
101
+ }
102
+
103
+ async function setupLifecycleMonitor() {
104
+ lastRuntime = createRuntimeEnv();
105
+ return setupFeishuLifecycleHandler({
106
+ createEventDispatcherMock,
107
+ onRegister: (registered) => {
108
+ handlers = registered;
109
+ },
110
+ runtime: lastRuntime,
111
+ cfg: lifecycleConfig,
112
+ account: lifecycleAccount,
113
+ handlerKey: "card.action.trigger",
114
+ missingHandlerMessage: "missing card.action.trigger handler",
115
+ });
116
+ }
117
+
118
+ function latestReplyDispatcherParams() {
119
+ const call = createFeishuReplyDispatcherMock.mock.calls.at(-1);
120
+ if (!call) {
121
+ throw new Error("expected Feishu reply dispatcher call");
122
+ }
123
+ return call[0] as {
124
+ accountId?: string;
125
+ chatId?: string;
126
+ replyToMessageId?: string;
127
+ };
128
+ }
129
+
130
+ function latestFinalizedContext() {
131
+ const call = finalizeInboundContextMock.mock.calls.at(-1);
132
+ if (!call) {
133
+ throw new Error("expected finalized inbound context call");
134
+ }
135
+ return call[0] as {
136
+ AccountId?: string;
137
+ SessionKey?: string;
138
+ MessageSid?: string;
139
+ };
140
+ }
141
+
142
+ describe("Feishu card-action lifecycle", () => {
143
+ beforeEach(() => {
144
+ vi.useRealTimers();
145
+ resetFeishuLifecycleTestMocks();
146
+ handlers = {};
147
+ lastRuntime = createRuntimeEnv();
148
+ resetProcessedFeishuCardActionTokensForTests();
149
+ setFeishuLifecycleStateDir("klaw-feishu-card-action");
150
+
151
+ createFeishuReplyDispatcherMock.mockReturnValue(createFeishuLifecycleReplyDispatcher());
152
+
153
+ resolveBoundConversationMock.mockImplementation(() => ({
154
+ bindingId: "binding-card",
155
+ targetSessionKey: "agent:bound-agent:feishu:direct:ou_user1",
156
+ }));
157
+
158
+ resolveAgentRouteMock.mockReturnValue({
159
+ agentId: "main",
160
+ channel: "feishu",
161
+ accountId: "acct-card",
162
+ sessionKey: "agent:main:feishu:direct:ou_user1",
163
+ mainSessionKey: "agent:main:main",
164
+ matchedBy: "default",
165
+ });
166
+
167
+ mockFeishuReplyOnceDispatch({
168
+ dispatchReplyFromConfigMock,
169
+ replyText: "card action reply once",
170
+ });
171
+
172
+ withReplyDispatcherMock.mockImplementation(async ({ run }) => await run());
173
+
174
+ installFeishuLifecycleReplyRuntime({
175
+ resolveAgentRouteMock,
176
+ finalizeInboundContextMock,
177
+ dispatchReplyFromConfigMock,
178
+ withReplyDispatcherMock,
179
+ storePath: "/tmp/feishu-card-action-sessions.json",
180
+ });
181
+ });
182
+
183
+ afterEach(() => {
184
+ vi.useRealTimers();
185
+ resetProcessedFeishuCardActionTokensForTests();
186
+ restoreFeishuLifecycleStateDir(originalStateDir);
187
+ });
188
+
189
+ it("routes one reply across duplicate callback delivery", async () => {
190
+ const onCardAction = await setupLifecycleMonitor();
191
+ const event = createCardActionEvent({
192
+ token: "tok-card-once",
193
+ action: "feishu.quick_actions.help",
194
+ command: "/help",
195
+ });
196
+
197
+ await expectFeishuReplyPipelineDedupedAcrossReplay({
198
+ handler: onCardAction,
199
+ event,
200
+ dispatchReplyFromConfigMock,
201
+ createFeishuReplyDispatcherMock,
202
+ });
203
+
204
+ expect(lastRuntime?.error).not.toHaveBeenCalled();
205
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
206
+ expect(createFeishuReplyDispatcherMock).toHaveBeenCalledTimes(1);
207
+ const dispatcherParams = latestReplyDispatcherParams();
208
+ expect(dispatcherParams.accountId).toBe("acct-card");
209
+ expect(dispatcherParams.chatId).toBe("p2p:ou_user1");
210
+ expect(dispatcherParams.replyToMessageId).toBeUndefined();
211
+ const finalized = latestFinalizedContext();
212
+ expect(finalized.AccountId).toBe("acct-card");
213
+ expect(finalized.SessionKey).toBe("agent:bound-agent:feishu:direct:ou_user1");
214
+ expect(finalized.MessageSid).toBe("card-action-tok-card-once");
215
+ expect(touchBindingMock).toHaveBeenCalledWith("binding-card");
216
+
217
+ expectFeishuReplyDispatcherSentFinalReplyOnce({ createFeishuReplyDispatcherMock });
218
+ expect(sendMessageFeishuMock).not.toHaveBeenCalled();
219
+ expect(sendCardFeishuMock).not.toHaveBeenCalled();
220
+ });
221
+
222
+ it("routes v2 callbacks that report open_chat_id instead of chat_id", async () => {
223
+ const onCardAction = await setupLifecycleMonitor();
224
+ const chatId = "oc_group_v2";
225
+
226
+ await onCardAction({
227
+ operator: {
228
+ open_id: "ou_user1",
229
+ },
230
+ token: "tok-card-v2-context",
231
+ action: {
232
+ tag: "button",
233
+ value: createFeishuCardInteractionEnvelope({
234
+ k: "quick",
235
+ a: "feishu.quick_actions.help",
236
+ q: "/help",
237
+ c: {
238
+ u: "ou_user1",
239
+ h: chatId,
240
+ t: "group",
241
+ e: Date.now() + 60_000,
242
+ },
243
+ }),
244
+ },
245
+ context: {
246
+ open_message_id: "om_card_v2",
247
+ open_chat_id: chatId,
248
+ },
249
+ });
250
+
251
+ expect(lastRuntime?.error).not.toHaveBeenCalled();
252
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
253
+ const dispatcherParams = latestReplyDispatcherParams();
254
+ expect(dispatcherParams.accountId).toBe("acct-card");
255
+ expect(dispatcherParams.chatId).toBe(chatId);
256
+ expect(dispatcherParams.replyToMessageId).toBe("om_card_v2");
257
+ expect(latestFinalizedContext().MessageSid).toBe("card-action-tok-card-v2-context");
258
+ });
259
+
260
+ it("routes v2 callbacks with nested operator identity", async () => {
261
+ const onCardAction = await setupLifecycleMonitor();
262
+ const chatId = "p2p:ou_user1";
263
+
264
+ await onCardAction({
265
+ operator: {
266
+ user_id: {
267
+ open_id: "ou_user1",
268
+ user_id: "user_1",
269
+ union_id: "union_1",
270
+ },
271
+ },
272
+ token: "tok-card-v2-nested-operator",
273
+ action: {
274
+ tag: "button",
275
+ value: createFeishuCardInteractionEnvelope({
276
+ k: "quick",
277
+ a: "feishu.quick_actions.help",
278
+ q: "/help",
279
+ c: {
280
+ u: "ou_user1",
281
+ h: chatId,
282
+ t: "p2p",
283
+ e: Date.now() + 60_000,
284
+ },
285
+ }),
286
+ },
287
+ context: {
288
+ open_message_id: "om_card_v2_nested",
289
+ open_chat_id: chatId,
290
+ },
291
+ });
292
+
293
+ expect(lastRuntime?.error).not.toHaveBeenCalled();
294
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
295
+ expect(createFeishuReplyDispatcherMock).toHaveBeenCalledWith(
296
+ expect.objectContaining({
297
+ accountId: "acct-card",
298
+ chatId,
299
+ replyToMessageId: "om_card_v2_nested",
300
+ }),
301
+ );
302
+ expect(finalizeInboundContextMock).toHaveBeenCalledWith(
303
+ expect.objectContaining({
304
+ AccountId: "acct-card",
305
+ SessionKey: "agent:bound-agent:feishu:direct:ou_user1",
306
+ MessageSid: "card-action-tok-card-v2-nested-operator",
307
+ }),
308
+ );
309
+ });
310
+
311
+ it("routes SDK-style card callbacks without context as direct callbacks", async () => {
312
+ const onCardAction = await setupLifecycleMonitor();
313
+
314
+ await onCardAction({
315
+ open_id: "ou_user1",
316
+ user_id: "user_1",
317
+ tenant_key: "tenant_1",
318
+ open_message_id: "om_sdk_card",
319
+ token: "tok-card-sdk-flat",
320
+ action: {
321
+ tag: "button",
322
+ value: {
323
+ command: "/help",
324
+ },
325
+ },
326
+ });
327
+
328
+ expect(lastRuntime?.error).not.toHaveBeenCalled();
329
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
330
+ const dispatcherParams = latestReplyDispatcherParams();
331
+ expect(dispatcherParams.accountId).toBe("acct-card");
332
+ expect(dispatcherParams.chatId).toBe("ou_user1");
333
+ expect(dispatcherParams.replyToMessageId).toBe("om_sdk_card");
334
+ expect(latestFinalizedContext().MessageSid).toBe("card-action-tok-card-sdk-flat");
335
+ });
336
+
337
+ it("plain-sends card action replies when Feishu provides no real message id", async () => {
338
+ const onCardAction = await setupLifecycleMonitor();
339
+
340
+ await onCardAction({
341
+ open_id: "ou_user1",
342
+ token: "tok-card-no-reply-target",
343
+ action: {
344
+ tag: "button",
345
+ value: {
346
+ command: "/help",
347
+ },
348
+ },
349
+ });
350
+
351
+ expect(lastRuntime?.error).not.toHaveBeenCalled();
352
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
353
+ const dispatcherParams = latestReplyDispatcherParams();
354
+ expect(dispatcherParams.accountId).toBe("acct-card");
355
+ expect(dispatcherParams.chatId).toBe("ou_user1");
356
+ expect(dispatcherParams.replyToMessageId).toBeUndefined();
357
+ expect(latestFinalizedContext().MessageSid).toBe("card-action-tok-card-no-reply-target");
358
+ });
359
+
360
+ it("does not duplicate delivery when retrying after a post-send failure", async () => {
361
+ const onCardAction = await setupLifecycleMonitor();
362
+ const event = createCardActionEvent({
363
+ token: "tok-card-retry",
364
+ action: "feishu.quick_actions.help",
365
+ command: "/help",
366
+ });
367
+
368
+ dispatchReplyFromConfigMock.mockImplementationOnce(async ({ dispatcher }) => {
369
+ await dispatcher.sendFinalReply({ text: "card action reply once" });
370
+ throw new Error("post-send failure");
371
+ });
372
+
373
+ await expectFeishuReplyPipelineDedupedAfterPostSendFailure({
374
+ handler: onCardAction,
375
+ event,
376
+ dispatchReplyFromConfigMock,
377
+ runtimeErrorMock: lastRuntime?.error as ReturnType<typeof vi.fn>,
378
+ });
379
+
380
+ expect(lastRuntime?.error).toHaveBeenCalledTimes(1);
381
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
382
+ expectFeishuReplyDispatcherSentFinalReplyOnce({ createFeishuReplyDispatcherMock });
383
+ });
384
+
385
+ it("drops malformed card-action events with empty tokens before handler dispatch", async () => {
386
+ const onCardAction = await setupLifecycleMonitor();
387
+
388
+ await onCardAction({
389
+ operator: {
390
+ open_id: "ou_user1",
391
+ user_id: "user_1",
392
+ union_id: "union_1",
393
+ },
394
+ token: "",
395
+ action: {
396
+ tag: "button",
397
+ value: createFeishuCardInteractionEnvelope({
398
+ k: "quick",
399
+ a: "feishu.quick_actions.help",
400
+ q: "/help",
401
+ c: {
402
+ u: "ou_user1",
403
+ h: "p2p:ou_user1",
404
+ t: "p2p",
405
+ e: Date.now() + 60_000,
406
+ },
407
+ }),
408
+ },
409
+ context: {
410
+ open_id: "ou_user1",
411
+ user_id: "user_1",
412
+ chat_id: "p2p:ou_user1",
413
+ },
414
+ });
415
+
416
+ expect(lastRuntime?.error).toHaveBeenCalledWith(
417
+ "feishu[acct-card]: ignoring malformed card action payload",
418
+ );
419
+ expect(dispatchReplyFromConfigMock).not.toHaveBeenCalled();
420
+ });
421
+ });