@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,627 @@
1
+ import {
2
+ getRequiredHookHandler,
3
+ registerHookHandlersForTest,
4
+ } from "klaw/plugin-sdk/channel-test-helpers";
5
+ import { beforeEach, describe, expect, it } from "vitest";
6
+ import type { ClawdbotConfig, KlawPluginApi } from "../runtime-api.js";
7
+ import { registerFeishuSubagentHooks } from "../subagent-hooks-api.js";
8
+ import {
9
+ createFeishuThreadBindingManager,
10
+ testing as threadBindingTesting,
11
+ } from "./thread-bindings.js";
12
+
13
+ const baseConfig: ClawdbotConfig = {
14
+ session: { mainKey: "main", scope: "per-sender" },
15
+ channels: { feishu: {} },
16
+ };
17
+
18
+ function registerHandlersForTest(config: Record<string, unknown> = baseConfig) {
19
+ return registerHookHandlersForTest<KlawPluginApi>({
20
+ config,
21
+ register: registerFeishuSubagentHooks,
22
+ });
23
+ }
24
+
25
+ async function expectHookError(value: unknown, expectedErrorFragment: string): Promise<void> {
26
+ const result = (await value) as { status?: unknown; error?: unknown };
27
+ expect(result.status).toBe("error");
28
+ expect(result.error).toContain(expectedErrorFragment);
29
+ }
30
+
31
+ describe("feishu subagent hook handlers", () => {
32
+ beforeEach(() => {
33
+ threadBindingTesting.resetFeishuThreadBindingsForTests();
34
+ });
35
+
36
+ it("binds a Feishu DM conversation on subagent_spawning", async () => {
37
+ const handlers = registerHandlersForTest();
38
+ const handler = getRequiredHookHandler(handlers, "subagent_spawning");
39
+ createFeishuThreadBindingManager({ cfg: baseConfig, accountId: "work" });
40
+
41
+ const result = await handler(
42
+ {
43
+ childSessionKey: "agent:main:subagent:child",
44
+ agentId: "codex",
45
+ label: "banana",
46
+ mode: "session",
47
+ requester: {
48
+ channel: "feishu",
49
+ accountId: "work",
50
+ to: "user:ou_sender_1",
51
+ },
52
+ threadRequested: true,
53
+ },
54
+ {},
55
+ );
56
+
57
+ expect(result).toEqual({
58
+ status: "ok",
59
+ threadBindingReady: true,
60
+ deliveryOrigin: {
61
+ channel: "feishu",
62
+ accountId: "work",
63
+ to: "user:ou_sender_1",
64
+ },
65
+ });
66
+
67
+ const deliveryTargetHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
68
+ await expect(
69
+ deliveryTargetHandler(
70
+ {
71
+ childSessionKey: "agent:main:subagent:child",
72
+ requesterSessionKey: "agent:main:main",
73
+ requesterOrigin: {
74
+ channel: "feishu",
75
+ accountId: "work",
76
+ to: "user:ou_sender_1",
77
+ },
78
+ expectsCompletionMessage: true,
79
+ },
80
+ {},
81
+ ),
82
+ ).resolves.toEqual({
83
+ origin: {
84
+ channel: "feishu",
85
+ accountId: "work",
86
+ to: "user:ou_sender_1",
87
+ },
88
+ });
89
+ });
90
+
91
+ it("preserves the original Feishu DM delivery target", async () => {
92
+ const handlers = registerHandlersForTest();
93
+ const deliveryHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
94
+ const manager = createFeishuThreadBindingManager({ cfg: baseConfig, accountId: "work" });
95
+
96
+ manager.bindConversation({
97
+ conversationId: "ou_sender_1",
98
+ targetKind: "subagent",
99
+ targetSessionKey: "agent:main:subagent:chat-dm-child",
100
+ metadata: {
101
+ deliveryTo: "chat:oc_dm_chat_1",
102
+ boundBy: "system",
103
+ },
104
+ });
105
+
106
+ await expect(
107
+ deliveryHandler(
108
+ {
109
+ childSessionKey: "agent:main:subagent:chat-dm-child",
110
+ requesterSessionKey: "agent:main:main",
111
+ requesterOrigin: {
112
+ channel: "feishu",
113
+ accountId: "work",
114
+ to: "chat:oc_dm_chat_1",
115
+ },
116
+ expectsCompletionMessage: true,
117
+ },
118
+ {},
119
+ ),
120
+ ).resolves.toEqual({
121
+ origin: {
122
+ channel: "feishu",
123
+ accountId: "work",
124
+ to: "chat:oc_dm_chat_1",
125
+ },
126
+ });
127
+ });
128
+
129
+ it("binds a Feishu topic conversation and preserves parent context", async () => {
130
+ const handlers = registerHandlersForTest();
131
+ const spawnHandler = getRequiredHookHandler(handlers, "subagent_spawning");
132
+ const deliveryHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
133
+ createFeishuThreadBindingManager({ cfg: baseConfig, accountId: "work" });
134
+
135
+ const result = await spawnHandler(
136
+ {
137
+ childSessionKey: "agent:main:subagent:topic-child",
138
+ agentId: "codex",
139
+ label: "topic-child",
140
+ mode: "session",
141
+ requester: {
142
+ channel: "feishu",
143
+ accountId: "work",
144
+ to: "chat:oc_group_chat",
145
+ threadId: "om_topic_root",
146
+ },
147
+ threadRequested: true,
148
+ },
149
+ {},
150
+ );
151
+
152
+ expect(result).toEqual({
153
+ status: "ok",
154
+ threadBindingReady: true,
155
+ deliveryOrigin: {
156
+ channel: "feishu",
157
+ accountId: "work",
158
+ to: "chat:oc_group_chat",
159
+ threadId: "om_topic_root",
160
+ },
161
+ });
162
+ await expect(
163
+ deliveryHandler(
164
+ {
165
+ childSessionKey: "agent:main:subagent:topic-child",
166
+ requesterSessionKey: "agent:main:main",
167
+ requesterOrigin: {
168
+ channel: "feishu",
169
+ accountId: "work",
170
+ to: "chat:oc_group_chat",
171
+ threadId: "om_topic_root",
172
+ },
173
+ expectsCompletionMessage: true,
174
+ },
175
+ {},
176
+ ),
177
+ ).resolves.toEqual({
178
+ origin: {
179
+ channel: "feishu",
180
+ accountId: "work",
181
+ to: "chat:oc_group_chat",
182
+ threadId: "om_topic_root",
183
+ },
184
+ });
185
+ });
186
+
187
+ it("uses the requester session binding to preserve sender-scoped topic conversations", async () => {
188
+ const handlers = registerHandlersForTest();
189
+ const spawnHandler = getRequiredHookHandler(handlers, "subagent_spawning");
190
+ const deliveryHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
191
+ const manager = createFeishuThreadBindingManager({ cfg: baseConfig, accountId: "work" });
192
+
193
+ manager.bindConversation({
194
+ conversationId: "oc_group_chat:topic:om_topic_root:sender:ou_sender_1",
195
+ parentConversationId: "oc_group_chat",
196
+ targetKind: "subagent",
197
+ targetSessionKey: "agent:main:parent",
198
+ metadata: {
199
+ agentId: "codex",
200
+ label: "parent",
201
+ boundBy: "system",
202
+ },
203
+ });
204
+
205
+ const reboundResult = await spawnHandler(
206
+ {
207
+ childSessionKey: "agent:main:subagent:sender-child",
208
+ agentId: "codex",
209
+ label: "sender-child",
210
+ mode: "session",
211
+ requester: {
212
+ channel: "feishu",
213
+ accountId: "work",
214
+ to: "chat:oc_group_chat",
215
+ threadId: "om_topic_root",
216
+ },
217
+ threadRequested: true,
218
+ },
219
+ {
220
+ requesterSessionKey: "agent:main:parent",
221
+ },
222
+ );
223
+
224
+ expect(reboundResult).toEqual({
225
+ status: "ok",
226
+ threadBindingReady: true,
227
+ deliveryOrigin: {
228
+ channel: "feishu",
229
+ accountId: "work",
230
+ to: "chat:oc_group_chat",
231
+ threadId: "om_topic_root",
232
+ },
233
+ });
234
+ const childBindings = manager.listBySessionKey("agent:main:subagent:sender-child");
235
+ expect(childBindings).toHaveLength(1);
236
+ expect(childBindings[0]?.conversationId).toBe(
237
+ "oc_group_chat:topic:om_topic_root:sender:ou_sender_1",
238
+ );
239
+ expect(childBindings[0]?.parentConversationId).toBe("oc_group_chat");
240
+ await expect(
241
+ deliveryHandler(
242
+ {
243
+ childSessionKey: "agent:main:subagent:sender-child",
244
+ requesterSessionKey: "agent:main:parent",
245
+ requesterOrigin: {
246
+ channel: "feishu",
247
+ accountId: "work",
248
+ to: "chat:oc_group_chat",
249
+ threadId: "om_topic_root",
250
+ },
251
+ expectsCompletionMessage: true,
252
+ },
253
+ {},
254
+ ),
255
+ ).resolves.toEqual({
256
+ origin: {
257
+ channel: "feishu",
258
+ accountId: "work",
259
+ to: "chat:oc_group_chat",
260
+ threadId: "om_topic_root",
261
+ },
262
+ });
263
+ });
264
+
265
+ it("prefers requester-matching bindings when multiple child bindings exist", async () => {
266
+ const handlers = registerHandlersForTest();
267
+ const spawnHandler = getRequiredHookHandler(handlers, "subagent_spawning");
268
+ const deliveryHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
269
+ createFeishuThreadBindingManager({ cfg: baseConfig, accountId: "work" });
270
+
271
+ await spawnHandler(
272
+ {
273
+ childSessionKey: "agent:main:subagent:shared",
274
+ agentId: "codex",
275
+ label: "shared",
276
+ mode: "session",
277
+ requester: {
278
+ channel: "feishu",
279
+ accountId: "work",
280
+ to: "user:ou_sender_1",
281
+ },
282
+ threadRequested: true,
283
+ },
284
+ {},
285
+ );
286
+ await spawnHandler(
287
+ {
288
+ childSessionKey: "agent:main:subagent:shared",
289
+ agentId: "codex",
290
+ label: "shared",
291
+ mode: "session",
292
+ requester: {
293
+ channel: "feishu",
294
+ accountId: "work",
295
+ to: "user:ou_sender_2",
296
+ },
297
+ threadRequested: true,
298
+ },
299
+ {},
300
+ );
301
+
302
+ await expect(
303
+ deliveryHandler(
304
+ {
305
+ childSessionKey: "agent:main:subagent:shared",
306
+ requesterSessionKey: "agent:main:main",
307
+ requesterOrigin: {
308
+ channel: "feishu",
309
+ accountId: "work",
310
+ to: "user:ou_sender_2",
311
+ },
312
+ expectsCompletionMessage: true,
313
+ },
314
+ {},
315
+ ),
316
+ ).resolves.toEqual({
317
+ origin: {
318
+ channel: "feishu",
319
+ accountId: "work",
320
+ to: "user:ou_sender_2",
321
+ },
322
+ });
323
+ });
324
+
325
+ it("fails closed when requester-session bindings remain ambiguous for the same topic", async () => {
326
+ const handlers = registerHandlersForTest();
327
+ const spawnHandler = getRequiredHookHandler(handlers, "subagent_spawning");
328
+ const deliveryHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
329
+ const manager = createFeishuThreadBindingManager({ cfg: baseConfig, accountId: "work" });
330
+
331
+ manager.bindConversation({
332
+ conversationId: "oc_group_chat:topic:om_topic_root:sender:ou_sender_1",
333
+ parentConversationId: "oc_group_chat",
334
+ targetKind: "subagent",
335
+ targetSessionKey: "agent:main:parent",
336
+ metadata: { boundBy: "system" },
337
+ });
338
+ manager.bindConversation({
339
+ conversationId: "oc_group_chat:topic:om_topic_root:sender:ou_sender_2",
340
+ parentConversationId: "oc_group_chat",
341
+ targetKind: "subagent",
342
+ targetSessionKey: "agent:main:parent",
343
+ metadata: { boundBy: "system" },
344
+ });
345
+
346
+ await expectHookError(
347
+ spawnHandler(
348
+ {
349
+ childSessionKey: "agent:main:subagent:ambiguous-child",
350
+ agentId: "codex",
351
+ label: "ambiguous-child",
352
+ mode: "session",
353
+ requester: {
354
+ channel: "feishu",
355
+ accountId: "work",
356
+ to: "chat:oc_group_chat",
357
+ threadId: "om_topic_root",
358
+ },
359
+ threadRequested: true,
360
+ },
361
+ {
362
+ requesterSessionKey: "agent:main:parent",
363
+ },
364
+ ),
365
+ "direct messages or topic conversations",
366
+ );
367
+
368
+ await expect(
369
+ deliveryHandler(
370
+ {
371
+ childSessionKey: "agent:main:subagent:ambiguous-child",
372
+ requesterSessionKey: "agent:main:parent",
373
+ requesterOrigin: {
374
+ channel: "feishu",
375
+ accountId: "work",
376
+ to: "chat:oc_group_chat",
377
+ threadId: "om_topic_root",
378
+ },
379
+ expectsCompletionMessage: true,
380
+ },
381
+ {},
382
+ ),
383
+ ).resolves.toBeUndefined();
384
+ });
385
+
386
+ it("fails closed when both topic-level and sender-scoped requester bindings exist", async () => {
387
+ const handlers = registerHandlersForTest();
388
+ const spawnHandler = getRequiredHookHandler(handlers, "subagent_spawning");
389
+ const deliveryHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
390
+ const manager = createFeishuThreadBindingManager({ cfg: baseConfig, accountId: "work" });
391
+
392
+ manager.bindConversation({
393
+ conversationId: "oc_group_chat:topic:om_topic_root",
394
+ parentConversationId: "oc_group_chat",
395
+ targetKind: "subagent",
396
+ targetSessionKey: "agent:main:parent",
397
+ metadata: { boundBy: "system" },
398
+ });
399
+ manager.bindConversation({
400
+ conversationId: "oc_group_chat:topic:om_topic_root:sender:ou_sender_1",
401
+ parentConversationId: "oc_group_chat",
402
+ targetKind: "subagent",
403
+ targetSessionKey: "agent:main:parent",
404
+ metadata: { boundBy: "system" },
405
+ });
406
+
407
+ await expectHookError(
408
+ spawnHandler(
409
+ {
410
+ childSessionKey: "agent:main:subagent:mixed-topic-child",
411
+ agentId: "codex",
412
+ label: "mixed-topic-child",
413
+ mode: "session",
414
+ requester: {
415
+ channel: "feishu",
416
+ accountId: "work",
417
+ to: "chat:oc_group_chat",
418
+ threadId: "om_topic_root",
419
+ },
420
+ threadRequested: true,
421
+ },
422
+ {
423
+ requesterSessionKey: "agent:main:parent",
424
+ },
425
+ ),
426
+ "direct messages or topic conversations",
427
+ );
428
+
429
+ await expect(
430
+ deliveryHandler(
431
+ {
432
+ childSessionKey: "agent:main:subagent:mixed-topic-child",
433
+ requesterSessionKey: "agent:main:parent",
434
+ requesterOrigin: {
435
+ channel: "feishu",
436
+ accountId: "work",
437
+ to: "chat:oc_group_chat",
438
+ threadId: "om_topic_root",
439
+ },
440
+ expectsCompletionMessage: true,
441
+ },
442
+ {},
443
+ ),
444
+ ).resolves.toBeUndefined();
445
+ });
446
+
447
+ it("no-ops for non-Feishu channels and non-threaded spawns", async () => {
448
+ const handlers = registerHandlersForTest();
449
+ const spawnHandler = getRequiredHookHandler(handlers, "subagent_spawning");
450
+ const deliveryHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
451
+ const endedHandler = getRequiredHookHandler(handlers, "subagent_ended");
452
+
453
+ await expect(
454
+ spawnHandler(
455
+ {
456
+ childSessionKey: "agent:main:subagent:child",
457
+ agentId: "codex",
458
+ mode: "run",
459
+ requester: {
460
+ channel: "discord",
461
+ accountId: "work",
462
+ to: "channel:123",
463
+ },
464
+ threadRequested: true,
465
+ },
466
+ {},
467
+ ),
468
+ ).resolves.toBeUndefined();
469
+
470
+ await expect(
471
+ spawnHandler(
472
+ {
473
+ childSessionKey: "agent:main:subagent:child",
474
+ agentId: "codex",
475
+ mode: "run",
476
+ requester: {
477
+ channel: "feishu",
478
+ accountId: "work",
479
+ to: "user:ou_sender_1",
480
+ },
481
+ threadRequested: false,
482
+ },
483
+ {},
484
+ ),
485
+ ).resolves.toBeUndefined();
486
+
487
+ await expect(
488
+ deliveryHandler(
489
+ {
490
+ childSessionKey: "agent:main:subagent:child",
491
+ requesterSessionKey: "agent:main:main",
492
+ requesterOrigin: {
493
+ channel: "discord",
494
+ accountId: "work",
495
+ to: "channel:123",
496
+ },
497
+ expectsCompletionMessage: true,
498
+ },
499
+ {},
500
+ ),
501
+ ).resolves.toBeUndefined();
502
+
503
+ await expect(
504
+ endedHandler(
505
+ {
506
+ targetSessionKey: "agent:main:subagent:child",
507
+ targetKind: "subagent",
508
+ reason: "done",
509
+ accountId: "work",
510
+ },
511
+ {},
512
+ ),
513
+ ).resolves.toBeUndefined();
514
+ });
515
+
516
+ it("returns an error for unsupported non-topic Feishu group conversations", async () => {
517
+ const handler = getRequiredHookHandler(registerHandlersForTest(), "subagent_spawning");
518
+ createFeishuThreadBindingManager({ cfg: baseConfig, accountId: "work" });
519
+
520
+ await expectHookError(
521
+ handler(
522
+ {
523
+ childSessionKey: "agent:main:subagent:child",
524
+ agentId: "codex",
525
+ mode: "session",
526
+ requester: {
527
+ channel: "feishu",
528
+ accountId: "work",
529
+ to: "chat:oc_group_chat",
530
+ },
531
+ threadRequested: true,
532
+ },
533
+ {},
534
+ ),
535
+ "direct messages or topic conversations",
536
+ );
537
+ });
538
+
539
+ it("unbinds Feishu bindings on subagent_ended", async () => {
540
+ const handlers = registerHandlersForTest();
541
+ const spawnHandler = getRequiredHookHandler(handlers, "subagent_spawning");
542
+ const deliveryHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
543
+ const endedHandler = getRequiredHookHandler(handlers, "subagent_ended");
544
+ createFeishuThreadBindingManager({ cfg: baseConfig, accountId: "work" });
545
+
546
+ await spawnHandler(
547
+ {
548
+ childSessionKey: "agent:main:subagent:child",
549
+ agentId: "codex",
550
+ mode: "session",
551
+ requester: {
552
+ channel: "feishu",
553
+ accountId: "work",
554
+ to: "user:ou_sender_1",
555
+ },
556
+ threadRequested: true,
557
+ },
558
+ {},
559
+ );
560
+
561
+ await endedHandler(
562
+ {
563
+ targetSessionKey: "agent:main:subagent:child",
564
+ targetKind: "subagent",
565
+ reason: "done",
566
+ accountId: "work",
567
+ },
568
+ {},
569
+ );
570
+
571
+ await expect(
572
+ deliveryHandler(
573
+ {
574
+ childSessionKey: "agent:main:subagent:child",
575
+ requesterSessionKey: "agent:main:main",
576
+ requesterOrigin: {
577
+ channel: "feishu",
578
+ accountId: "work",
579
+ to: "user:ou_sender_1",
580
+ },
581
+ expectsCompletionMessage: true,
582
+ },
583
+ {},
584
+ ),
585
+ ).resolves.toBeUndefined();
586
+ });
587
+
588
+ it("fails closed when the Feishu monitor-owned binding manager is unavailable", async () => {
589
+ const handlers = registerHandlersForTest();
590
+ const spawnHandler = getRequiredHookHandler(handlers, "subagent_spawning");
591
+ const deliveryHandler = getRequiredHookHandler(handlers, "subagent_delivery_target");
592
+
593
+ await expectHookError(
594
+ spawnHandler(
595
+ {
596
+ childSessionKey: "agent:main:subagent:no-manager",
597
+ agentId: "codex",
598
+ mode: "session",
599
+ requester: {
600
+ channel: "feishu",
601
+ accountId: "work",
602
+ to: "user:ou_sender_1",
603
+ },
604
+ threadRequested: true,
605
+ },
606
+ {},
607
+ ),
608
+ "monitor is not active",
609
+ );
610
+
611
+ await expect(
612
+ deliveryHandler(
613
+ {
614
+ childSessionKey: "agent:main:subagent:no-manager",
615
+ requesterSessionKey: "agent:main:main",
616
+ requesterOrigin: {
617
+ channel: "feishu",
618
+ accountId: "work",
619
+ to: "user:ou_sender_1",
620
+ },
621
+ expectsCompletionMessage: true,
622
+ },
623
+ {},
624
+ ),
625
+ ).resolves.toBeUndefined();
626
+ });
627
+ });