@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,373 @@
1
+ import { createRuntimeEnv } from "openclaw/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.OPENCLAW_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
+ describe("Feishu card-action lifecycle", () => {
119
+ beforeEach(() => {
120
+ vi.useRealTimers();
121
+ resetFeishuLifecycleTestMocks();
122
+ _handlers = {};
123
+ lastRuntime = createRuntimeEnv();
124
+ resetProcessedFeishuCardActionTokensForTests();
125
+ setFeishuLifecycleStateDir("openclaw-feishu-card-action");
126
+
127
+ createFeishuReplyDispatcherMock.mockReturnValue(createFeishuLifecycleReplyDispatcher());
128
+
129
+ resolveBoundConversationMock.mockImplementation(() => ({
130
+ bindingId: "binding-card",
131
+ targetSessionKey: "agent:bound-agent:feishu:direct:ou_user1",
132
+ }));
133
+
134
+ resolveAgentRouteMock.mockReturnValue({
135
+ agentId: "main",
136
+ channel: "feishu",
137
+ accountId: "acct-card",
138
+ sessionKey: "agent:main:feishu:direct:ou_user1",
139
+ mainSessionKey: "agent:main:main",
140
+ matchedBy: "default",
141
+ });
142
+
143
+ mockFeishuReplyOnceDispatch({
144
+ dispatchReplyFromConfigMock,
145
+ replyText: "card action reply once",
146
+ });
147
+
148
+ withReplyDispatcherMock.mockImplementation(async ({ run }) => await run());
149
+
150
+ installFeishuLifecycleReplyRuntime({
151
+ resolveAgentRouteMock,
152
+ finalizeInboundContextMock,
153
+ dispatchReplyFromConfigMock,
154
+ withReplyDispatcherMock,
155
+ storePath: "/tmp/feishu-card-action-sessions.json",
156
+ });
157
+ });
158
+
159
+ afterEach(() => {
160
+ vi.useRealTimers();
161
+ resetProcessedFeishuCardActionTokensForTests();
162
+ restoreFeishuLifecycleStateDir(originalStateDir);
163
+ });
164
+
165
+ it("routes one reply across duplicate callback delivery", async () => {
166
+ const onCardAction = await setupLifecycleMonitor();
167
+ const event = createCardActionEvent({
168
+ token: "tok-card-once",
169
+ action: "feishu.quick_actions.help",
170
+ command: "/help",
171
+ });
172
+
173
+ await expectFeishuReplyPipelineDedupedAcrossReplay({
174
+ handler: onCardAction,
175
+ event,
176
+ dispatchReplyFromConfigMock,
177
+ createFeishuReplyDispatcherMock,
178
+ });
179
+
180
+ expect(lastRuntime?.error).not.toHaveBeenCalled();
181
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
182
+ expect(createFeishuReplyDispatcherMock).toHaveBeenCalledTimes(1);
183
+ expect(createFeishuReplyDispatcherMock).toHaveBeenCalledWith(
184
+ expect.objectContaining({
185
+ accountId: "acct-card",
186
+ chatId: "p2p:ou_user1",
187
+ replyToMessageId: undefined,
188
+ }),
189
+ );
190
+ expect(finalizeInboundContextMock).toHaveBeenCalledWith(
191
+ expect.objectContaining({
192
+ AccountId: "acct-card",
193
+ SessionKey: "agent:bound-agent:feishu:direct:ou_user1",
194
+ MessageSid: "card-action-tok-card-once",
195
+ }),
196
+ );
197
+ expect(touchBindingMock).toHaveBeenCalledWith("binding-card");
198
+
199
+ expectFeishuReplyDispatcherSentFinalReplyOnce({ createFeishuReplyDispatcherMock });
200
+ expect(sendMessageFeishuMock).not.toHaveBeenCalled();
201
+ expect(sendCardFeishuMock).not.toHaveBeenCalled();
202
+ });
203
+
204
+ it("routes v2 callbacks that report open_chat_id instead of chat_id", async () => {
205
+ const onCardAction = await setupLifecycleMonitor();
206
+ const chatId = "oc_group_v2";
207
+
208
+ await onCardAction({
209
+ operator: {
210
+ open_id: "ou_user1",
211
+ },
212
+ token: "tok-card-v2-context",
213
+ action: {
214
+ tag: "button",
215
+ value: createFeishuCardInteractionEnvelope({
216
+ k: "quick",
217
+ a: "feishu.quick_actions.help",
218
+ q: "/help",
219
+ c: {
220
+ u: "ou_user1",
221
+ h: chatId,
222
+ t: "group",
223
+ e: Date.now() + 60_000,
224
+ },
225
+ }),
226
+ },
227
+ context: {
228
+ open_message_id: "om_card_v2",
229
+ open_chat_id: chatId,
230
+ },
231
+ });
232
+
233
+ expect(lastRuntime?.error).not.toHaveBeenCalled();
234
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
235
+ expect(createFeishuReplyDispatcherMock).toHaveBeenCalledWith(
236
+ expect.objectContaining({
237
+ accountId: "acct-card",
238
+ chatId,
239
+ replyToMessageId: "om_card_v2",
240
+ }),
241
+ );
242
+ expect(finalizeInboundContextMock).toHaveBeenCalledWith(
243
+ expect.objectContaining({
244
+ MessageSid: "card-action-tok-card-v2-context",
245
+ }),
246
+ );
247
+ });
248
+
249
+ it("routes SDK-style card callbacks without context as direct callbacks", async () => {
250
+ const onCardAction = await setupLifecycleMonitor();
251
+
252
+ await onCardAction({
253
+ open_id: "ou_user1",
254
+ user_id: "user_1",
255
+ tenant_key: "tenant_1",
256
+ open_message_id: "om_sdk_card",
257
+ token: "tok-card-sdk-flat",
258
+ action: {
259
+ tag: "button",
260
+ value: {
261
+ command: "/help",
262
+ },
263
+ },
264
+ });
265
+
266
+ expect(lastRuntime?.error).not.toHaveBeenCalled();
267
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
268
+ expect(createFeishuReplyDispatcherMock).toHaveBeenCalledWith(
269
+ expect.objectContaining({
270
+ accountId: "acct-card",
271
+ chatId: "ou_user1",
272
+ replyToMessageId: "om_sdk_card",
273
+ }),
274
+ );
275
+ expect(finalizeInboundContextMock).toHaveBeenCalledWith(
276
+ expect.objectContaining({
277
+ MessageSid: "card-action-tok-card-sdk-flat",
278
+ }),
279
+ );
280
+ });
281
+
282
+ it("plain-sends card action replies when Feishu provides no real message id", async () => {
283
+ const onCardAction = await setupLifecycleMonitor();
284
+
285
+ await onCardAction({
286
+ open_id: "ou_user1",
287
+ token: "tok-card-no-reply-target",
288
+ action: {
289
+ tag: "button",
290
+ value: {
291
+ command: "/help",
292
+ },
293
+ },
294
+ });
295
+
296
+ expect(lastRuntime?.error).not.toHaveBeenCalled();
297
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
298
+ expect(createFeishuReplyDispatcherMock).toHaveBeenCalledWith(
299
+ expect.objectContaining({
300
+ accountId: "acct-card",
301
+ chatId: "ou_user1",
302
+ replyToMessageId: undefined,
303
+ }),
304
+ );
305
+ expect(finalizeInboundContextMock).toHaveBeenCalledWith(
306
+ expect.objectContaining({
307
+ MessageSid: "card-action-tok-card-no-reply-target",
308
+ }),
309
+ );
310
+ });
311
+
312
+ it("does not duplicate delivery when retrying after a post-send failure", async () => {
313
+ const onCardAction = await setupLifecycleMonitor();
314
+ const event = createCardActionEvent({
315
+ token: "tok-card-retry",
316
+ action: "feishu.quick_actions.help",
317
+ command: "/help",
318
+ });
319
+
320
+ dispatchReplyFromConfigMock.mockImplementationOnce(async ({ dispatcher }) => {
321
+ await dispatcher.sendFinalReply({ text: "card action reply once" });
322
+ throw new Error("post-send failure");
323
+ });
324
+
325
+ await expectFeishuReplyPipelineDedupedAfterPostSendFailure({
326
+ handler: onCardAction,
327
+ event,
328
+ dispatchReplyFromConfigMock,
329
+ runtimeErrorMock: lastRuntime?.error as ReturnType<typeof vi.fn>,
330
+ });
331
+
332
+ expect(lastRuntime?.error).toHaveBeenCalledTimes(1);
333
+ expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
334
+ expectFeishuReplyDispatcherSentFinalReplyOnce({ createFeishuReplyDispatcherMock });
335
+ });
336
+
337
+ it("drops malformed card-action events with empty tokens before handler dispatch", async () => {
338
+ const onCardAction = await setupLifecycleMonitor();
339
+
340
+ await onCardAction({
341
+ operator: {
342
+ open_id: "ou_user1",
343
+ user_id: "user_1",
344
+ union_id: "union_1",
345
+ },
346
+ token: "",
347
+ action: {
348
+ tag: "button",
349
+ value: createFeishuCardInteractionEnvelope({
350
+ k: "quick",
351
+ a: "feishu.quick_actions.help",
352
+ q: "/help",
353
+ c: {
354
+ u: "ou_user1",
355
+ h: "p2p:ou_user1",
356
+ t: "p2p",
357
+ e: Date.now() + 60_000,
358
+ },
359
+ }),
360
+ },
361
+ context: {
362
+ open_id: "ou_user1",
363
+ user_id: "user_1",
364
+ chat_id: "p2p:ou_user1",
365
+ },
366
+ });
367
+
368
+ expect(lastRuntime?.error).toHaveBeenCalledWith(
369
+ "feishu[acct-card]: ignoring malformed card action payload",
370
+ );
371
+ expect(dispatchReplyFromConfigMock).not.toHaveBeenCalled();
372
+ });
373
+ });