@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,454 @@
1
+ import { randomUUID } from "node:crypto";
2
+ import { createPluginRuntimeMock } from "klaw/plugin-sdk/channel-test-helpers";
3
+ import { expect, vi, type Mock } from "vitest";
4
+ import type { ClawdbotConfig, PluginRuntime, RuntimeEnv } from "../../runtime-api.js";
5
+ import { setFeishuRuntime } from "../runtime.js";
6
+ import type { ResolvedFeishuAccount } from "../types.js";
7
+
8
+ const FEISHU_LIFECYCLE_WAIT_TIMEOUT_MS = 10_000;
9
+
10
+ type InboundDebouncerParams<T> = {
11
+ onFlush?: (items: T[]) => Promise<void>;
12
+ onError?: (err: unknown, items: T[]) => void;
13
+ };
14
+ type UnknownMock = Mock<(...args: unknown[]) => unknown>;
15
+ type AsyncUnknownMock = Mock<(...args: unknown[]) => Promise<unknown>>;
16
+ type FeishuDispatchReplyCounts = {
17
+ final: number;
18
+ block?: number;
19
+ tool?: number;
20
+ };
21
+ type FeishuDispatchReplyContext = Record<string, unknown> & {
22
+ SessionKey?: string;
23
+ };
24
+ type FeishuDispatchReplyDispatcher = {
25
+ sendFinalReply: (payload: { text: string }) => unknown;
26
+ };
27
+ type FeishuDispatchReplyMock = Mock<
28
+ (args: {
29
+ ctx: FeishuDispatchReplyContext;
30
+ dispatcher: FeishuDispatchReplyDispatcher;
31
+ }) => Promise<{ queuedFinal: boolean; counts: FeishuDispatchReplyCounts }>
32
+ >;
33
+ type FeishuLifecycleReplyDispatcher = {
34
+ dispatcher: {
35
+ sendToolResult: UnknownMock;
36
+ sendBlockReply: UnknownMock;
37
+ sendFinalReply: AsyncUnknownMock;
38
+ waitForIdle: AsyncUnknownMock;
39
+ getQueuedCounts: UnknownMock;
40
+ markComplete: UnknownMock;
41
+ };
42
+ replyOptions: Record<string, never>;
43
+ markDispatchIdle: UnknownMock;
44
+ };
45
+
46
+ export function setFeishuLifecycleStateDir(prefix: string) {
47
+ process.env.KLAW_STATE_DIR = `/tmp/${prefix}-${randomUUID()}`;
48
+ }
49
+
50
+ export function restoreFeishuLifecycleStateDir(originalStateDir: string | undefined) {
51
+ if (originalStateDir === undefined) {
52
+ delete process.env.KLAW_STATE_DIR;
53
+ return;
54
+ }
55
+ process.env.KLAW_STATE_DIR = originalStateDir;
56
+ }
57
+
58
+ const FEISHU_PREFETCHED_BOT_OPEN_ID_SOURCE = {
59
+ kind: "prefetched",
60
+ botOpenId: "ou_bot_1",
61
+ botName: "Bot",
62
+ } as const;
63
+
64
+ export function createFeishuLifecycleReplyDispatcher(): FeishuLifecycleReplyDispatcher {
65
+ return {
66
+ dispatcher: {
67
+ sendToolResult: vi.fn(() => false),
68
+ sendBlockReply: vi.fn(() => false),
69
+ sendFinalReply: vi.fn(async () => true),
70
+ waitForIdle: vi.fn(async () => {}),
71
+ getQueuedCounts: vi.fn(() => ({ tool: 0, block: 0, final: 0 })),
72
+ markComplete: vi.fn(),
73
+ },
74
+ replyOptions: {},
75
+ markDispatchIdle: vi.fn(),
76
+ };
77
+ }
78
+
79
+ function createImmediateInboundDebounce() {
80
+ return {
81
+ resolveInboundDebounceMs: vi.fn(() => 0),
82
+ createInboundDebouncer: <T>(params: InboundDebouncerParams<T>) => ({
83
+ enqueue: async (item: T) => {
84
+ try {
85
+ await params.onFlush?.([item]);
86
+ } catch (err) {
87
+ params.onError?.(err, [item]);
88
+ }
89
+ },
90
+ flushKey: async () => {},
91
+ cancelKey: () => false,
92
+ }),
93
+ };
94
+ }
95
+
96
+ function installFeishuLifecycleRuntime(params: {
97
+ resolveAgentRoute: PluginRuntime["channel"]["routing"]["resolveAgentRoute"];
98
+ finalizeInboundContext: PluginRuntime["channel"]["reply"]["finalizeInboundContext"];
99
+ dispatchReplyFromConfig: PluginRuntime["channel"]["reply"]["dispatchReplyFromConfig"];
100
+ withReplyDispatcher: PluginRuntime["channel"]["reply"]["withReplyDispatcher"];
101
+ resolveStorePath: PluginRuntime["channel"]["session"]["resolveStorePath"];
102
+ hasControlCommand?: PluginRuntime["channel"]["text"]["hasControlCommand"];
103
+ shouldComputeCommandAuthorized?: PluginRuntime["channel"]["commands"]["shouldComputeCommandAuthorized"];
104
+ resolveCommandAuthorizedFromAuthorizers?: PluginRuntime["channel"]["commands"]["resolveCommandAuthorizedFromAuthorizers"];
105
+ readAllowFromStore?: PluginRuntime["channel"]["pairing"]["readAllowFromStore"];
106
+ upsertPairingRequest?: PluginRuntime["channel"]["pairing"]["upsertPairingRequest"];
107
+ buildPairingReply?: PluginRuntime["channel"]["pairing"]["buildPairingReply"];
108
+ detectMime?: PluginRuntime["media"]["detectMime"];
109
+ }): PluginRuntime {
110
+ const runtime = createPluginRuntimeMock({
111
+ channel: {
112
+ debounce: createImmediateInboundDebounce(),
113
+ text: {
114
+ hasControlCommand: params.hasControlCommand ?? vi.fn(() => false),
115
+ },
116
+ routing: {
117
+ resolveAgentRoute: params.resolveAgentRoute,
118
+ },
119
+ reply: {
120
+ resolveEnvelopeFormatOptions: vi.fn(() => ({})),
121
+ formatAgentEnvelope: vi.fn((value: { body: string }) => value.body),
122
+ finalizeInboundContext: params.finalizeInboundContext,
123
+ dispatchReplyFromConfig: params.dispatchReplyFromConfig,
124
+ withReplyDispatcher: params.withReplyDispatcher,
125
+ },
126
+ commands: {
127
+ shouldComputeCommandAuthorized: params.shouldComputeCommandAuthorized ?? vi.fn(() => false),
128
+ resolveCommandAuthorizedFromAuthorizers:
129
+ params.resolveCommandAuthorizedFromAuthorizers ?? vi.fn(() => false),
130
+ },
131
+ session: {
132
+ readSessionUpdatedAt: vi.fn(),
133
+ resolveStorePath: params.resolveStorePath,
134
+ },
135
+ pairing: {
136
+ readAllowFromStore: params.readAllowFromStore ?? vi.fn().mockResolvedValue([]),
137
+ upsertPairingRequest: params.upsertPairingRequest ?? vi.fn(),
138
+ buildPairingReply: params.buildPairingReply ?? vi.fn(),
139
+ },
140
+ },
141
+ media: {
142
+ detectMime: params.detectMime ?? vi.fn(async () => "text/plain"),
143
+ },
144
+ }) as unknown as PluginRuntime;
145
+ setFeishuRuntime(runtime);
146
+ return runtime;
147
+ }
148
+
149
+ export function installFeishuLifecycleReplyRuntime(params: {
150
+ resolveAgentRouteMock: unknown;
151
+ finalizeInboundContextMock: unknown;
152
+ dispatchReplyFromConfigMock: unknown;
153
+ withReplyDispatcherMock: unknown;
154
+ storePath: string;
155
+ }): PluginRuntime {
156
+ return installFeishuLifecycleRuntime({
157
+ resolveAgentRoute:
158
+ params.resolveAgentRouteMock as PluginRuntime["channel"]["routing"]["resolveAgentRoute"],
159
+ finalizeInboundContext:
160
+ params.finalizeInboundContextMock as PluginRuntime["channel"]["reply"]["finalizeInboundContext"],
161
+ dispatchReplyFromConfig:
162
+ params.dispatchReplyFromConfigMock as PluginRuntime["channel"]["reply"]["dispatchReplyFromConfig"],
163
+ withReplyDispatcher:
164
+ params.withReplyDispatcherMock as PluginRuntime["channel"]["reply"]["withReplyDispatcher"],
165
+ resolveStorePath: vi.fn(() => params.storePath),
166
+ });
167
+ }
168
+
169
+ export function mockFeishuReplyOnceDispatch(params: {
170
+ dispatchReplyFromConfigMock: FeishuDispatchReplyMock;
171
+ replyText: string;
172
+ shouldSendFinalReply?: (ctx: unknown) => boolean;
173
+ }) {
174
+ params.dispatchReplyFromConfigMock.mockImplementation(async ({ ctx, dispatcher }) => {
175
+ const shouldSendFinalReply = params.shouldSendFinalReply?.(ctx) ?? true;
176
+ if (shouldSendFinalReply && typeof dispatcher?.sendFinalReply === "function") {
177
+ await dispatcher.sendFinalReply({ text: params.replyText });
178
+ }
179
+ return {
180
+ queuedFinal: false,
181
+ counts: { final: shouldSendFinalReply ? 1 : 0 },
182
+ };
183
+ });
184
+ }
185
+
186
+ export function createFeishuLifecycleConfig(params: {
187
+ accountId: string;
188
+ appId: string;
189
+ appSecret: string;
190
+ channelConfig?: Record<string, unknown>;
191
+ accountConfig?: Record<string, unknown>;
192
+ extraConfig?: Record<string, unknown>;
193
+ }): ClawdbotConfig {
194
+ const extraConfig = params.extraConfig ?? {};
195
+ return {
196
+ ...extraConfig,
197
+ channels: {
198
+ ...(extraConfig.channels as Record<string, unknown> | undefined),
199
+ feishu: {
200
+ enabled: true,
201
+ requireMention: false,
202
+ resolveSenderNames: false,
203
+ ...params.channelConfig,
204
+ accounts: {
205
+ [params.accountId]: {
206
+ enabled: true,
207
+ appId: params.appId,
208
+ appSecret: params.appSecret, // pragma: allowlist secret
209
+ connectionMode: "websocket",
210
+ requireMention: false,
211
+ resolveSenderNames: false,
212
+ ...params.accountConfig,
213
+ },
214
+ },
215
+ },
216
+ },
217
+ messages: {
218
+ inbound: {
219
+ debounceMs: 0,
220
+ byChannel: {
221
+ feishu: 0,
222
+ },
223
+ },
224
+ },
225
+ } as ClawdbotConfig;
226
+ }
227
+
228
+ export function createFeishuLifecycleFixture(params: {
229
+ accountId: string;
230
+ appId: string;
231
+ appSecret: string;
232
+ channelConfig?: Record<string, unknown>;
233
+ accountConfig?: Record<string, unknown>;
234
+ extraConfig?: Record<string, unknown>;
235
+ }) {
236
+ return {
237
+ cfg: createFeishuLifecycleConfig(params),
238
+ account: createResolvedFeishuLifecycleAccount({
239
+ accountId: params.accountId,
240
+ appId: params.appId,
241
+ appSecret: params.appSecret,
242
+ config: {
243
+ ...params.channelConfig,
244
+ ...params.accountConfig,
245
+ },
246
+ }),
247
+ };
248
+ }
249
+
250
+ export function createResolvedFeishuLifecycleAccount(params: {
251
+ accountId: string;
252
+ appId: string;
253
+ appSecret: string;
254
+ config: Record<string, unknown>;
255
+ }): ResolvedFeishuAccount {
256
+ return {
257
+ accountId: params.accountId,
258
+ selectionSource: "config",
259
+ enabled: true,
260
+ configured: true,
261
+ appId: params.appId,
262
+ appSecret: params.appSecret, // pragma: allowlist secret
263
+ domain: "feishu",
264
+ config: {
265
+ enabled: true,
266
+ connectionMode: "websocket",
267
+ requireMention: false,
268
+ resolveSenderNames: false,
269
+ ...params.config,
270
+ },
271
+ } as unknown as ResolvedFeishuAccount;
272
+ }
273
+
274
+ export function createFeishuTextMessageEvent(params: {
275
+ messageId: string;
276
+ chatId: string;
277
+ text: string;
278
+ chatType?: "group" | "p2p";
279
+ senderOpenId?: string;
280
+ rootId?: string;
281
+ threadId?: string;
282
+ }) {
283
+ return {
284
+ sender: {
285
+ sender_id: { open_id: params.senderOpenId ?? "ou_sender_1" },
286
+ sender_type: "user",
287
+ },
288
+ message: {
289
+ message_id: params.messageId,
290
+ ...(params.rootId ? { root_id: params.rootId } : {}),
291
+ ...(params.threadId ? { thread_id: params.threadId } : {}),
292
+ chat_id: params.chatId,
293
+ chat_type: params.chatType ?? "group",
294
+ message_type: "text",
295
+ content: JSON.stringify({ text: params.text }),
296
+ create_time: "1710000000000",
297
+ },
298
+ };
299
+ }
300
+
301
+ async function expectFeishuLifecycleEventually(
302
+ assertion: () => void | Promise<void>,
303
+ timeoutMs: number,
304
+ ) {
305
+ try {
306
+ await assertion();
307
+ } catch {
308
+ await vi.waitFor(assertion, { timeout: timeoutMs });
309
+ }
310
+ }
311
+
312
+ async function replayFeishuLifecycleEvent(params: {
313
+ handler: (data: unknown) => Promise<void>;
314
+ event: unknown;
315
+ waitForFirst: () => void | Promise<void>;
316
+ waitForSecond?: () => void | Promise<void>;
317
+ waitTimeoutMs?: number;
318
+ }) {
319
+ const waitTimeoutMs = params.waitTimeoutMs ?? FEISHU_LIFECYCLE_WAIT_TIMEOUT_MS;
320
+ await params.handler(params.event);
321
+ await expectFeishuLifecycleEventually(params.waitForFirst, waitTimeoutMs);
322
+ await params.handler(params.event);
323
+ await expectFeishuLifecycleEventually(params.waitForSecond ?? params.waitForFirst, waitTimeoutMs);
324
+ }
325
+
326
+ export async function runFeishuLifecycleSequence(
327
+ deliveries: Array<() => Promise<void>>,
328
+ waits: Array<() => void | Promise<void>>,
329
+ ) {
330
+ for (const [index, deliver] of deliveries.entries()) {
331
+ await deliver();
332
+ await expectFeishuLifecycleEventually(
333
+ waits[index] ?? waits.at(-1) ?? (() => {}),
334
+ FEISHU_LIFECYCLE_WAIT_TIMEOUT_MS,
335
+ );
336
+ }
337
+ }
338
+
339
+ export async function expectFeishuSingleEffectAcrossReplay(params: {
340
+ handler: (data: unknown) => Promise<void>;
341
+ event: unknown;
342
+ effectMock: ReturnType<typeof vi.fn>;
343
+ effectCount?: number;
344
+ }) {
345
+ const effectCount = params.effectCount ?? 1;
346
+ await replayFeishuLifecycleEvent({
347
+ handler: params.handler,
348
+ event: params.event,
349
+ waitForFirst: () => {
350
+ expect(params.effectMock).toHaveBeenCalledTimes(effectCount);
351
+ },
352
+ });
353
+ }
354
+
355
+ export async function expectFeishuReplyPipelineDedupedAcrossReplay(params: {
356
+ handler: (data: unknown) => Promise<void>;
357
+ event: unknown;
358
+ dispatchReplyFromConfigMock: ReturnType<typeof vi.fn>;
359
+ createFeishuReplyDispatcherMock: ReturnType<typeof vi.fn>;
360
+ waitTimeoutMs?: number;
361
+ }) {
362
+ const waitTimeoutMs = params.waitTimeoutMs ?? FEISHU_LIFECYCLE_WAIT_TIMEOUT_MS;
363
+ await replayFeishuLifecycleEvent({
364
+ handler: params.handler,
365
+ event: params.event,
366
+ waitTimeoutMs,
367
+ waitForFirst: () => {
368
+ expect(params.dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
369
+ },
370
+ waitForSecond: () => {
371
+ expect(params.dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
372
+ expect(params.createFeishuReplyDispatcherMock).toHaveBeenCalledTimes(1);
373
+ },
374
+ });
375
+ }
376
+
377
+ export async function expectFeishuReplyPipelineDedupedAfterPostSendFailure(params: {
378
+ handler: (data: unknown) => Promise<void>;
379
+ event: unknown;
380
+ dispatchReplyFromConfigMock: ReturnType<typeof vi.fn>;
381
+ runtimeErrorMock: ReturnType<typeof vi.fn>;
382
+ waitTimeoutMs?: number;
383
+ }) {
384
+ const waitTimeoutMs = params.waitTimeoutMs ?? FEISHU_LIFECYCLE_WAIT_TIMEOUT_MS;
385
+ await replayFeishuLifecycleEvent({
386
+ handler: params.handler,
387
+ event: params.event,
388
+ waitTimeoutMs,
389
+ waitForFirst: () => {
390
+ expect(params.dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
391
+ expect(params.runtimeErrorMock).toHaveBeenCalledTimes(1);
392
+ },
393
+ waitForSecond: () => {
394
+ expect(params.dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
395
+ expect(params.runtimeErrorMock).toHaveBeenCalledTimes(1);
396
+ },
397
+ });
398
+ }
399
+
400
+ export function expectFeishuReplyDispatcherSentFinalReplyOnce(params: {
401
+ createFeishuReplyDispatcherMock: ReturnType<typeof vi.fn>;
402
+ }) {
403
+ const dispatcher = params.createFeishuReplyDispatcherMock.mock.results[0]?.value.dispatcher as {
404
+ sendFinalReply: ReturnType<typeof vi.fn>;
405
+ };
406
+ expect(dispatcher.sendFinalReply).toHaveBeenCalledTimes(1);
407
+ }
408
+
409
+ async function loadMonitorSingleAccount() {
410
+ const module = await import("../monitor.account.js");
411
+ return module.monitorSingleAccount;
412
+ }
413
+
414
+ export async function setupFeishuLifecycleHandler(params: {
415
+ createEventDispatcherMock: {
416
+ mockReturnValue: (value: unknown) => unknown;
417
+ mockReturnValueOnce: (value: unknown) => unknown;
418
+ };
419
+ onRegister: (registered: Record<string, (data: unknown) => Promise<void>>) => void;
420
+ runtime: RuntimeEnv;
421
+ cfg: ClawdbotConfig;
422
+ account: ResolvedFeishuAccount;
423
+ handlerKey: string;
424
+ missingHandlerMessage: string;
425
+ once?: boolean;
426
+ }): Promise<(data: unknown) => Promise<void>> {
427
+ const register = vi.fn((registered: Record<string, (data: unknown) => Promise<void>>) => {
428
+ params.onRegister(registered);
429
+ });
430
+ if (params.once) {
431
+ params.createEventDispatcherMock.mockReturnValueOnce({ register });
432
+ } else {
433
+ params.createEventDispatcherMock.mockReturnValue({ register });
434
+ }
435
+
436
+ const monitorSingleAccount = await loadMonitorSingleAccount();
437
+ await monitorSingleAccount({
438
+ cfg: params.cfg,
439
+ account: params.account,
440
+ runtime: params.runtime,
441
+ botOpenIdSource: FEISHU_PREFETCHED_BOT_OPEN_ID_SOURCE,
442
+ fireAndForget: false,
443
+ });
444
+
445
+ const handlers: Record<string, (data: unknown) => Promise<void>> = {};
446
+ for (const [key, value] of Object.entries(register.mock.calls.at(0)?.[0] ?? {})) {
447
+ handlers[key] = value as (data: unknown) => Promise<void>;
448
+ }
449
+ const handler = handlers[params.handlerKey];
450
+ if (!handler) {
451
+ throw new Error(params.missingHandlerMessage);
452
+ }
453
+ return handler;
454
+ }
@@ -0,0 +1,180 @@
1
+ import type { KlawConfig } from "klaw/plugin-sdk/config-contracts";
2
+ import { getSessionBindingService } from "klaw/plugin-sdk/conversation-runtime";
3
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
4
+ import { testing, createFeishuThreadBindingManager } from "./thread-bindings.js";
5
+
6
+ const baseCfg = {
7
+ session: { mainKey: "main", scope: "per-sender" },
8
+ } satisfies KlawConfig;
9
+
10
+ describe("Feishu thread bindings", () => {
11
+ beforeEach(() => {
12
+ testing.resetFeishuThreadBindingsForTests();
13
+ });
14
+
15
+ afterEach(() => {
16
+ vi.restoreAllMocks();
17
+ });
18
+
19
+ it("registers current-placement adapter capabilities for Feishu", () => {
20
+ createFeishuThreadBindingManager({ cfg: baseCfg, accountId: "default" });
21
+
22
+ expect(
23
+ getSessionBindingService().getCapabilities({
24
+ channel: "feishu",
25
+ accountId: "default",
26
+ }),
27
+ ).toEqual({
28
+ adapterAvailable: true,
29
+ bindSupported: true,
30
+ unbindSupported: true,
31
+ placements: ["current"],
32
+ });
33
+ });
34
+
35
+ it("binds and resolves a Feishu topic conversation", async () => {
36
+ vi.spyOn(Date, "now").mockReturnValue(1_700_000_000_000);
37
+ createFeishuThreadBindingManager({ cfg: baseCfg, accountId: "default" });
38
+
39
+ const binding = await getSessionBindingService().bind({
40
+ targetSessionKey: "agent:codex:acp:binding:feishu:default:abc123",
41
+ targetKind: "session",
42
+ conversation: {
43
+ channel: "feishu",
44
+ accountId: "default",
45
+ conversationId: "oc_group_chat:topic:om_topic_root",
46
+ parentConversationId: "oc_group_chat",
47
+ },
48
+ placement: "current",
49
+ metadata: {
50
+ agentId: "codex",
51
+ label: "codex-main",
52
+ },
53
+ });
54
+
55
+ expect(binding.conversation.conversationId).toBe("oc_group_chat:topic:om_topic_root");
56
+ const resolved = getSessionBindingService().resolveByConversation({
57
+ channel: "feishu",
58
+ accountId: "default",
59
+ conversationId: "oc_group_chat:topic:om_topic_root",
60
+ });
61
+ expect(resolved).toEqual({
62
+ bindingId: "default:oc_group_chat:topic:om_topic_root",
63
+ targetSessionKey: "agent:codex:acp:binding:feishu:default:abc123",
64
+ targetKind: "session",
65
+ conversation: {
66
+ channel: "feishu",
67
+ accountId: "default",
68
+ conversationId: "oc_group_chat:topic:om_topic_root",
69
+ parentConversationId: "oc_group_chat",
70
+ },
71
+ status: "active",
72
+ boundAt: 1_700_000_000_000,
73
+ expiresAt: 1_700_086_400_000,
74
+ metadata: {
75
+ agentId: "codex",
76
+ label: "codex-main",
77
+ boundBy: undefined,
78
+ deliveryTo: undefined,
79
+ deliveryThreadId: undefined,
80
+ lastActivityAt: 1_700_000_000_000,
81
+ idleTimeoutMs: 86_400_000,
82
+ maxAgeMs: 0,
83
+ },
84
+ });
85
+ });
86
+
87
+ it("clears account-scoped bindings when the manager stops", async () => {
88
+ const manager = createFeishuThreadBindingManager({ cfg: baseCfg, accountId: "default" });
89
+
90
+ await getSessionBindingService().bind({
91
+ targetSessionKey: "agent:codex:acp:binding:feishu:default:abc123",
92
+ targetKind: "session",
93
+ conversation: {
94
+ channel: "feishu",
95
+ accountId: "default",
96
+ conversationId: "oc_group_chat:topic:om_topic_root",
97
+ parentConversationId: "oc_group_chat",
98
+ },
99
+ placement: "current",
100
+ metadata: {
101
+ agentId: "codex",
102
+ },
103
+ });
104
+
105
+ manager.stop();
106
+
107
+ expect(
108
+ getSessionBindingService().resolveByConversation({
109
+ channel: "feishu",
110
+ accountId: "default",
111
+ conversationId: "oc_group_chat:topic:om_topic_root",
112
+ }),
113
+ ).toBeNull();
114
+ });
115
+
116
+ it("preserves delivery routing metadata when rebinding the same conversation", async () => {
117
+ vi.spyOn(Date, "now").mockReturnValue(1_700_000_100_000);
118
+ const manager = createFeishuThreadBindingManager({ cfg: baseCfg, accountId: "default" });
119
+
120
+ manager.bindConversation({
121
+ conversationId: "oc_group_chat:topic:om_topic_root:sender:ou_sender_1",
122
+ parentConversationId: "oc_group_chat",
123
+ targetKind: "subagent",
124
+ targetSessionKey: "agent:main:subagent:child",
125
+ metadata: {
126
+ agentId: "codex",
127
+ label: "child",
128
+ boundBy: "system",
129
+ deliveryTo: "user:ou_sender_1",
130
+ deliveryThreadId: "om_topic_root",
131
+ },
132
+ });
133
+
134
+ await getSessionBindingService().bind({
135
+ targetSessionKey: "agent:main:subagent:child",
136
+ targetKind: "subagent",
137
+ conversation: {
138
+ channel: "feishu",
139
+ accountId: "default",
140
+ conversationId: "oc_group_chat:topic:om_topic_root:sender:ou_sender_1",
141
+ parentConversationId: "oc_group_chat",
142
+ },
143
+ placement: "current",
144
+ metadata: {
145
+ label: "child",
146
+ },
147
+ });
148
+
149
+ expect(
150
+ getSessionBindingService().resolveByConversation({
151
+ channel: "feishu",
152
+ accountId: "default",
153
+ conversationId: "oc_group_chat:topic:om_topic_root:sender:ou_sender_1",
154
+ }),
155
+ ).toEqual({
156
+ bindingId: "default:oc_group_chat:topic:om_topic_root:sender:ou_sender_1",
157
+ targetSessionKey: "agent:main:subagent:child",
158
+ targetKind: "subagent",
159
+ conversation: {
160
+ channel: "feishu",
161
+ accountId: "default",
162
+ conversationId: "oc_group_chat:topic:om_topic_root:sender:ou_sender_1",
163
+ parentConversationId: "oc_group_chat",
164
+ },
165
+ status: "active",
166
+ boundAt: 1_700_000_100_000,
167
+ expiresAt: 1_700_086_500_000,
168
+ metadata: {
169
+ agentId: "codex",
170
+ label: "child",
171
+ boundBy: "system",
172
+ deliveryTo: "user:ou_sender_1",
173
+ deliveryThreadId: "om_topic_root",
174
+ lastActivityAt: 1_700_000_100_000,
175
+ idleTimeoutMs: 86_400_000,
176
+ maxAgeMs: 0,
177
+ },
178
+ });
179
+ });
180
+ });