@openclaw/feishu 2026.3.13 → 2026.5.2-beta.1

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 (187) hide show
  1. package/api.ts +31 -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/index.ts +70 -53
  6. package/openclaw.plugin.json +1827 -4
  7. package/package.json +32 -7
  8. package/runtime-api.ts +55 -0
  9. package/secret-contract-api.ts +5 -0
  10. package/security-contract-api.ts +1 -0
  11. package/session-key-api.ts +1 -0
  12. package/setup-api.ts +3 -0
  13. package/setup-entry.test.ts +14 -0
  14. package/setup-entry.ts +13 -0
  15. package/src/accounts.test.ts +95 -7
  16. package/src/accounts.ts +199 -117
  17. package/src/app-registration.ts +331 -0
  18. package/src/approval-auth.test.ts +24 -0
  19. package/src/approval-auth.ts +25 -0
  20. package/src/async.test.ts +35 -0
  21. package/src/async.ts +43 -1
  22. package/src/audio-preflight.runtime.ts +9 -0
  23. package/src/bitable.test.ts +131 -0
  24. package/src/bitable.ts +59 -22
  25. package/src/bot-content.ts +474 -0
  26. package/src/bot-group-name.test.ts +108 -0
  27. package/src/bot-runtime-api.ts +12 -0
  28. package/src/bot-sender-name.ts +125 -0
  29. package/src/bot.broadcast.test.ts +463 -0
  30. package/src/bot.card-action.test.ts +519 -5
  31. package/src/bot.checkBotMentioned.test.ts +92 -20
  32. package/src/bot.helpers.test.ts +118 -0
  33. package/src/bot.stripBotMention.test.ts +13 -21
  34. package/src/bot.test.ts +1334 -401
  35. package/src/bot.ts +778 -775
  36. package/src/card-action.ts +408 -40
  37. package/src/card-interaction.test.ts +129 -0
  38. package/src/card-interaction.ts +159 -0
  39. package/src/card-test-helpers.ts +47 -0
  40. package/src/card-ux-approval.ts +65 -0
  41. package/src/card-ux-launcher.test.ts +99 -0
  42. package/src/card-ux-launcher.ts +121 -0
  43. package/src/card-ux-shared.ts +33 -0
  44. package/src/channel-runtime-api.ts +16 -0
  45. package/src/channel.runtime.ts +47 -0
  46. package/src/channel.test.ts +914 -3
  47. package/src/channel.ts +1253 -309
  48. package/src/chat-schema.ts +5 -4
  49. package/src/chat.test.ts +135 -28
  50. package/src/chat.ts +68 -10
  51. package/src/client.test.ts +212 -103
  52. package/src/client.ts +115 -21
  53. package/src/comment-dispatcher-runtime-api.ts +6 -0
  54. package/src/comment-dispatcher.test.ts +169 -0
  55. package/src/comment-dispatcher.ts +107 -0
  56. package/src/comment-handler-runtime-api.ts +3 -0
  57. package/src/comment-handler.test.ts +486 -0
  58. package/src/comment-handler.ts +309 -0
  59. package/src/comment-reaction.test.ts +166 -0
  60. package/src/comment-reaction.ts +259 -0
  61. package/src/comment-shared.test.ts +182 -0
  62. package/src/comment-shared.ts +406 -0
  63. package/src/comment-target.ts +44 -0
  64. package/src/config-schema.test.ts +63 -1
  65. package/src/config-schema.ts +31 -4
  66. package/src/conversation-id.test.ts +18 -0
  67. package/src/conversation-id.ts +199 -0
  68. package/src/dedup-runtime-api.ts +1 -0
  69. package/src/dedup.ts +33 -95
  70. package/src/directory.static.ts +61 -0
  71. package/src/directory.test.ts +116 -20
  72. package/src/directory.ts +60 -92
  73. package/src/doc-schema.ts +1 -1
  74. package/src/docx-batch-insert.test.ts +39 -38
  75. package/src/docx-batch-insert.ts +55 -19
  76. package/src/docx-color-text.ts +9 -4
  77. package/src/docx-table-ops.test.ts +53 -0
  78. package/src/docx-table-ops.ts +52 -34
  79. package/src/docx-types.ts +38 -0
  80. package/src/docx.account-selection.test.ts +12 -3
  81. package/src/docx.test.ts +314 -74
  82. package/src/docx.ts +278 -122
  83. package/src/drive-schema.ts +47 -1
  84. package/src/drive.test.ts +1219 -0
  85. package/src/drive.ts +614 -13
  86. package/src/dynamic-agent.ts +10 -4
  87. package/src/event-types.ts +45 -0
  88. package/src/external-keys.ts +1 -1
  89. package/src/lifecycle.test-support.ts +220 -0
  90. package/src/media.test.ts +403 -26
  91. package/src/media.ts +509 -132
  92. package/src/mention-target.types.ts +5 -0
  93. package/src/mention.ts +32 -51
  94. package/src/message-action-contract.ts +13 -0
  95. package/src/monitor-state-runtime-api.ts +7 -0
  96. package/src/monitor-transport-runtime-api.ts +7 -0
  97. package/src/monitor.account.ts +218 -312
  98. package/src/monitor.acp-init-failure.lifecycle.test-support.ts +219 -0
  99. package/src/monitor.bot-identity.ts +86 -0
  100. package/src/monitor.bot-menu-handler.ts +165 -0
  101. package/src/monitor.bot-menu.lifecycle.test-support.ts +224 -0
  102. package/src/monitor.bot-menu.test.ts +178 -0
  103. package/src/monitor.broadcast.reply-once.lifecycle.test-support.ts +264 -0
  104. package/src/monitor.card-action.lifecycle.test-support.ts +373 -0
  105. package/src/monitor.cleanup.test.ts +376 -0
  106. package/src/monitor.comment-notice-handler.ts +105 -0
  107. package/src/monitor.comment.test.ts +937 -0
  108. package/src/monitor.comment.ts +1386 -0
  109. package/src/monitor.lifecycle.test.ts +4 -0
  110. package/src/monitor.message-handler.ts +339 -0
  111. package/src/monitor.reaction.lifecycle.test-support.ts +68 -0
  112. package/src/monitor.reaction.test.ts +108 -48
  113. package/src/monitor.startup.test.ts +11 -9
  114. package/src/monitor.startup.ts +26 -16
  115. package/src/monitor.state.ts +20 -5
  116. package/src/monitor.synthetic-error.ts +18 -0
  117. package/src/monitor.test-mocks.ts +2 -2
  118. package/src/monitor.transport.ts +220 -60
  119. package/src/monitor.ts +15 -10
  120. package/src/monitor.webhook-e2e.test.ts +65 -7
  121. package/src/monitor.webhook-security.test.ts +122 -0
  122. package/src/monitor.webhook.test-helpers.ts +44 -26
  123. package/src/outbound-runtime-api.ts +1 -0
  124. package/src/outbound.test.ts +616 -37
  125. package/src/outbound.ts +623 -81
  126. package/src/perm-schema.ts +1 -1
  127. package/src/perm.ts +1 -7
  128. package/src/pins.ts +108 -0
  129. package/src/policy.test.ts +297 -117
  130. package/src/policy.ts +142 -29
  131. package/src/post.ts +7 -6
  132. package/src/probe.test.ts +14 -9
  133. package/src/probe.ts +26 -16
  134. package/src/processing-claims.ts +59 -0
  135. package/src/qr-terminal.ts +1 -0
  136. package/src/reactions.ts +4 -34
  137. package/src/reasoning-preview.test.ts +59 -0
  138. package/src/reasoning-preview.ts +20 -0
  139. package/src/reply-dispatcher-runtime-api.ts +7 -0
  140. package/src/reply-dispatcher.test.ts +660 -29
  141. package/src/reply-dispatcher.ts +407 -154
  142. package/src/runtime.ts +6 -3
  143. package/src/secret-contract.ts +145 -0
  144. package/src/secret-input.ts +1 -13
  145. package/src/security-audit-shared.ts +69 -0
  146. package/src/security-audit.test.ts +61 -0
  147. package/src/security-audit.ts +1 -0
  148. package/src/send-result.ts +1 -1
  149. package/src/send-target.test.ts +9 -3
  150. package/src/send-target.ts +10 -4
  151. package/src/send.reply-fallback.test.ts +105 -2
  152. package/src/send.test.ts +386 -4
  153. package/src/send.ts +414 -95
  154. package/src/sequential-key.test.ts +72 -0
  155. package/src/sequential-key.ts +28 -0
  156. package/src/sequential-queue.test.ts +92 -0
  157. package/src/sequential-queue.ts +16 -0
  158. package/src/session-conversation.ts +42 -0
  159. package/src/session-route.ts +48 -0
  160. package/src/setup-core.ts +51 -0
  161. package/src/{onboarding.test.ts → setup-surface.test.ts} +52 -21
  162. package/src/setup-surface.ts +581 -0
  163. package/src/streaming-card.test.ts +138 -2
  164. package/src/streaming-card.ts +134 -18
  165. package/src/subagent-hooks.test.ts +603 -0
  166. package/src/subagent-hooks.ts +397 -0
  167. package/src/targets.ts +3 -13
  168. package/src/test-support/lifecycle-test-support.ts +453 -0
  169. package/src/thread-bindings.test.ts +143 -0
  170. package/src/thread-bindings.ts +330 -0
  171. package/src/tool-account-routing.test.ts +66 -8
  172. package/src/tool-account.test.ts +44 -0
  173. package/src/tool-account.ts +40 -17
  174. package/src/tool-factory-test-harness.ts +11 -8
  175. package/src/tool-result.ts +3 -1
  176. package/src/tools-config.ts +1 -1
  177. package/src/types.ts +16 -15
  178. package/src/typing.ts +10 -6
  179. package/src/wiki-schema.ts +1 -1
  180. package/src/wiki.ts +1 -7
  181. package/subagent-hooks-api.ts +31 -0
  182. package/tsconfig.json +16 -0
  183. package/src/feishu-command-handler.ts +0 -59
  184. package/src/onboarding.status.test.ts +0 -25
  185. package/src/onboarding.ts +0 -489
  186. package/src/send-message.ts +0 -71
  187. package/src/targets.test.ts +0 -70
@@ -0,0 +1,45 @@
1
+ export type FeishuMessageEvent = {
2
+ sender: {
3
+ sender_id: {
4
+ open_id?: string;
5
+ user_id?: string;
6
+ union_id?: string;
7
+ };
8
+ sender_type?: string;
9
+ tenant_key?: string;
10
+ };
11
+ message: {
12
+ message_id: string;
13
+ reply_target_message_id?: string;
14
+ suppress_reply_target?: boolean;
15
+ root_id?: string;
16
+ parent_id?: string;
17
+ thread_id?: string;
18
+ chat_id: string;
19
+ chat_type: "p2p" | "group" | "topic_group" | "private";
20
+ message_type: string;
21
+ content: string;
22
+ create_time?: string;
23
+ mentions?: Array<{
24
+ key: string;
25
+ id: {
26
+ open_id?: string;
27
+ user_id?: string;
28
+ union_id?: string;
29
+ };
30
+ name: string;
31
+ tenant_key?: string;
32
+ }>;
33
+ };
34
+ };
35
+
36
+ export type FeishuBotAddedEvent = {
37
+ chat_id: string;
38
+ operator_id: {
39
+ open_id?: string;
40
+ user_id?: string;
41
+ union_id?: string;
42
+ };
43
+ external: boolean;
44
+ operator_tenant_key?: string;
45
+ };
@@ -1,4 +1,4 @@
1
- const CONTROL_CHARS_RE = /[\u0000-\u001f\u007f]/;
1
+ const CONTROL_CHARS_RE = /\p{Cc}/u;
2
2
  const MAX_EXTERNAL_KEY_LENGTH = 512;
3
3
 
4
4
  export function normalizeFeishuExternalKey(value: unknown): string | undefined {
@@ -0,0 +1,220 @@
1
+ import { vi, type Mock } from "vitest";
2
+
3
+ type BoundConversation = {
4
+ bindingId: string;
5
+ targetSessionKey: string;
6
+ };
7
+ type UnknownMock = Mock<(...args: unknown[]) => unknown>;
8
+ type AsyncUnknownMock = Mock<(...args: unknown[]) => Promise<unknown>>;
9
+ type FinalizeInboundContextMock = Mock<
10
+ (ctx: Record<string, unknown>, opts?: unknown) => Record<string, unknown>
11
+ >;
12
+ type DispatchReplyCounts = {
13
+ final: number;
14
+ block?: number;
15
+ tool?: number;
16
+ };
17
+ type DispatchReplyContext = Record<string, unknown> & {
18
+ SessionKey?: string;
19
+ };
20
+ type DispatchReplyDispatcher = {
21
+ sendFinalReply: (payload: { text: string }) => unknown;
22
+ };
23
+ type FeishuReplyDispatcherMockValue = {
24
+ dispatcher: DispatchReplyDispatcher;
25
+ replyOptions: Record<string, never>;
26
+ markDispatchIdle: () => unknown;
27
+ };
28
+ type CreateFeishuReplyDispatcherMock = Mock<(params?: unknown) => FeishuReplyDispatcherMockValue>;
29
+ type DispatchReplyFromConfigMock = Mock<
30
+ (params: {
31
+ ctx: DispatchReplyContext;
32
+ dispatcher: DispatchReplyDispatcher;
33
+ }) => Promise<{ queuedFinal: boolean; counts: DispatchReplyCounts }>
34
+ >;
35
+ type WithReplyDispatcherMock = Mock<
36
+ (params: {
37
+ dispatcher?: DispatchReplyDispatcher;
38
+ onSettled?: () => unknown;
39
+ run: () => unknown;
40
+ }) => Promise<unknown>
41
+ >;
42
+ type FeishuLifecycleTestMocks = {
43
+ createEventDispatcherMock: UnknownMock;
44
+ monitorWebSocketMock: AsyncUnknownMock;
45
+ monitorWebhookMock: AsyncUnknownMock;
46
+ createFeishuThreadBindingManagerMock: UnknownMock;
47
+ createFeishuReplyDispatcherMock: CreateFeishuReplyDispatcherMock;
48
+ resolveBoundConversationMock: Mock<(ref?: unknown) => BoundConversation | null>;
49
+ touchBindingMock: UnknownMock;
50
+ resolveAgentRouteMock: UnknownMock;
51
+ resolveConfiguredBindingRouteMock: UnknownMock;
52
+ ensureConfiguredBindingRouteReadyMock: UnknownMock;
53
+ dispatchReplyFromConfigMock: DispatchReplyFromConfigMock;
54
+ withReplyDispatcherMock: WithReplyDispatcherMock;
55
+ finalizeInboundContextMock: FinalizeInboundContextMock;
56
+ getMessageFeishuMock: AsyncUnknownMock;
57
+ listFeishuThreadMessagesMock: AsyncUnknownMock;
58
+ sendMessageFeishuMock: AsyncUnknownMock;
59
+ sendCardFeishuMock: AsyncUnknownMock;
60
+ };
61
+
62
+ const feishuLifecycleTestMocks = vi.hoisted(
63
+ (): FeishuLifecycleTestMocks => ({
64
+ createEventDispatcherMock: vi.fn(),
65
+ monitorWebSocketMock: vi.fn(async () => {}),
66
+ monitorWebhookMock: vi.fn(async () => {}),
67
+ createFeishuThreadBindingManagerMock: vi.fn(() => ({ stop: vi.fn() })),
68
+ createFeishuReplyDispatcherMock: vi.fn(),
69
+ resolveBoundConversationMock: vi.fn<(ref?: unknown) => BoundConversation | null>(() => null),
70
+ touchBindingMock: vi.fn(),
71
+ resolveAgentRouteMock: vi.fn(),
72
+ resolveConfiguredBindingRouteMock: vi.fn(),
73
+ ensureConfiguredBindingRouteReadyMock: vi.fn(),
74
+ dispatchReplyFromConfigMock: vi.fn(),
75
+ withReplyDispatcherMock: vi.fn(),
76
+ finalizeInboundContextMock: vi.fn((ctx) => ctx),
77
+ getMessageFeishuMock: vi.fn(async () => null),
78
+ listFeishuThreadMessagesMock: vi.fn(async () => []),
79
+ sendMessageFeishuMock: vi.fn(async () => ({ messageId: "om_sent", chatId: "chat_default" })),
80
+ sendCardFeishuMock: vi.fn(async () => ({ messageId: "om_card", chatId: "chat_default" })),
81
+ }),
82
+ );
83
+
84
+ export function getFeishuLifecycleTestMocks(): FeishuLifecycleTestMocks {
85
+ return feishuLifecycleTestMocks;
86
+ }
87
+
88
+ export function resetFeishuLifecycleTestMocks(): void {
89
+ for (const mock of Object.values(feishuLifecycleTestMocks)) {
90
+ mock.mockReset();
91
+ }
92
+ feishuLifecycleTestMocks.monitorWebSocketMock.mockResolvedValue(undefined);
93
+ feishuLifecycleTestMocks.monitorWebhookMock.mockResolvedValue(undefined);
94
+ feishuLifecycleTestMocks.createFeishuThreadBindingManagerMock.mockReturnValue({ stop: vi.fn() });
95
+ feishuLifecycleTestMocks.resolveBoundConversationMock.mockReturnValue(null);
96
+ feishuLifecycleTestMocks.finalizeInboundContextMock.mockImplementation((ctx) => ctx);
97
+ feishuLifecycleTestMocks.getMessageFeishuMock.mockResolvedValue(null);
98
+ feishuLifecycleTestMocks.listFeishuThreadMessagesMock.mockResolvedValue([]);
99
+ feishuLifecycleTestMocks.sendMessageFeishuMock.mockResolvedValue({
100
+ messageId: "om_sent",
101
+ chatId: "chat_default",
102
+ });
103
+ feishuLifecycleTestMocks.sendCardFeishuMock.mockResolvedValue({
104
+ messageId: "om_card",
105
+ chatId: "chat_default",
106
+ });
107
+ }
108
+
109
+ const {
110
+ createEventDispatcherMock,
111
+ monitorWebSocketMock,
112
+ monitorWebhookMock,
113
+ createFeishuThreadBindingManagerMock,
114
+ createFeishuReplyDispatcherMock,
115
+ resolveBoundConversationMock,
116
+ touchBindingMock,
117
+ resolveConfiguredBindingRouteMock,
118
+ ensureConfiguredBindingRouteReadyMock,
119
+ getMessageFeishuMock,
120
+ listFeishuThreadMessagesMock,
121
+ sendMessageFeishuMock,
122
+ sendCardFeishuMock,
123
+ } = feishuLifecycleTestMocks;
124
+
125
+ vi.mock("./client.js", () => {
126
+ return {
127
+ FEISHU_HTTP_TIMEOUT_ENV_VAR: "OPENCLAW_FEISHU_HTTP_TIMEOUT_MS",
128
+ FEISHU_HTTP_TIMEOUT_MAX_MS: 300_000,
129
+ FEISHU_HTTP_TIMEOUT_MS: 30_000,
130
+ FEISHU_USER_AGENT: "openclaw-feishu-test",
131
+ clearClientCache: vi.fn(),
132
+ createFeishuClient: vi.fn(() => {
133
+ throw new Error("unexpected Feishu client call in lifecycle test");
134
+ }),
135
+ createFeishuWSClient: vi.fn(async () => ({
136
+ close: vi.fn(),
137
+ start: vi.fn(),
138
+ })),
139
+ createEventDispatcher: createEventDispatcherMock,
140
+ getFeishuClient: vi.fn(() => null),
141
+ getFeishuUserAgent: vi.fn(() => "openclaw-feishu-test"),
142
+ pluginVersion: "test",
143
+ setFeishuClientRuntimeForTest: vi.fn(),
144
+ };
145
+ });
146
+
147
+ vi.mock("./monitor.transport.js", () => ({
148
+ monitorWebSocket: monitorWebSocketMock,
149
+ monitorWebhook: monitorWebhookMock,
150
+ }));
151
+
152
+ vi.mock("./thread-bindings.js", () => ({
153
+ createFeishuThreadBindingManager: createFeishuThreadBindingManagerMock,
154
+ }));
155
+
156
+ vi.mock("./reply-dispatcher.js", () => ({
157
+ createFeishuReplyDispatcher: createFeishuReplyDispatcherMock,
158
+ }));
159
+
160
+ vi.mock("./send.js", () => ({
161
+ sendCardFeishu: sendCardFeishuMock,
162
+ getMessageFeishu: getMessageFeishuMock,
163
+ listFeishuThreadMessages: listFeishuThreadMessagesMock,
164
+ sendMessageFeishu: sendMessageFeishuMock,
165
+ }));
166
+
167
+ vi.mock("openclaw/plugin-sdk/conversation-runtime", async () => {
168
+ const actual = await vi.importActual<typeof import("openclaw/plugin-sdk/conversation-runtime")>(
169
+ "openclaw/plugin-sdk/conversation-runtime",
170
+ );
171
+ return {
172
+ ...actual,
173
+ resolveConfiguredBindingRoute: (
174
+ params: Parameters<typeof actual.resolveConfiguredBindingRoute>[0],
175
+ ) =>
176
+ resolveConfiguredBindingRouteMock.getMockImplementation()
177
+ ? resolveConfiguredBindingRouteMock(params)
178
+ : actual.resolveConfiguredBindingRoute(params),
179
+ resolveRuntimeConversationBindingRoute: (
180
+ params: Parameters<typeof actual.resolveRuntimeConversationBindingRoute>[0],
181
+ ) => {
182
+ const conversation =
183
+ "conversation" in params
184
+ ? params.conversation
185
+ : {
186
+ channel: params.channel,
187
+ accountId: params.accountId,
188
+ conversationId: params.conversationId,
189
+ parentConversationId: params.parentConversationId,
190
+ };
191
+ const bindingRecord = resolveBoundConversationMock(conversation);
192
+ const boundSessionKey = bindingRecord?.targetSessionKey?.trim();
193
+ if (!bindingRecord || !boundSessionKey) {
194
+ return { bindingRecord: null, route: params.route };
195
+ }
196
+ touchBindingMock(bindingRecord.bindingId);
197
+ return {
198
+ bindingRecord,
199
+ boundSessionKey,
200
+ boundAgentId: params.route.agentId,
201
+ route: {
202
+ ...params.route,
203
+ sessionKey: boundSessionKey,
204
+ lastRoutePolicy: boundSessionKey === params.route.mainSessionKey ? "main" : "session",
205
+ matchedBy: "binding.channel",
206
+ },
207
+ };
208
+ },
209
+ ensureConfiguredBindingRouteReady: (
210
+ params: Parameters<typeof actual.ensureConfiguredBindingRouteReady>[0],
211
+ ) =>
212
+ ensureConfiguredBindingRouteReadyMock.getMockImplementation()
213
+ ? ensureConfiguredBindingRouteReadyMock(params)
214
+ : actual.ensureConfiguredBindingRouteReady(params),
215
+ getSessionBindingService: () => ({
216
+ resolveByConversation: resolveBoundConversationMock,
217
+ touch: touchBindingMock,
218
+ }),
219
+ };
220
+ });