@gakr-gakr/feishu 0.1.0

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 (133) hide show
  1. package/api.ts +32 -0
  2. package/autobot.plugin.json +180 -0
  3. package/channel-entry.ts +20 -0
  4. package/channel-plugin-api.ts +1 -0
  5. package/contract-api.ts +16 -0
  6. package/index.ts +82 -0
  7. package/package.json +62 -0
  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.ts +13 -0
  14. package/skills/feishu-doc/SKILL.md +211 -0
  15. package/skills/feishu-doc/references/block-types.md +103 -0
  16. package/skills/feishu-drive/SKILL.md +97 -0
  17. package/skills/feishu-perm/SKILL.md +119 -0
  18. package/skills/feishu-wiki/SKILL.md +113 -0
  19. package/src/accounts.ts +333 -0
  20. package/src/agent-config.ts +21 -0
  21. package/src/app-registration.ts +331 -0
  22. package/src/approval-auth.ts +25 -0
  23. package/src/async.ts +104 -0
  24. package/src/audio-preflight.runtime.ts +9 -0
  25. package/src/bitable.ts +762 -0
  26. package/src/bot-content.ts +485 -0
  27. package/src/bot-runtime-api.ts +12 -0
  28. package/src/bot-sender-name.ts +125 -0
  29. package/src/bot.ts +1703 -0
  30. package/src/card-action.ts +447 -0
  31. package/src/card-interaction.ts +159 -0
  32. package/src/card-test-helpers.ts +54 -0
  33. package/src/card-ux-approval.ts +65 -0
  34. package/src/card-ux-launcher.ts +121 -0
  35. package/src/card-ux-shared.ts +33 -0
  36. package/src/channel-runtime-api.ts +16 -0
  37. package/src/channel.runtime.ts +47 -0
  38. package/src/channel.ts +1423 -0
  39. package/src/chat-schema.ts +25 -0
  40. package/src/chat.ts +188 -0
  41. package/src/client-timeout.ts +42 -0
  42. package/src/client.ts +262 -0
  43. package/src/comment-dispatcher-runtime-api.ts +6 -0
  44. package/src/comment-dispatcher.ts +107 -0
  45. package/src/comment-handler-runtime-api.ts +3 -0
  46. package/src/comment-handler.ts +303 -0
  47. package/src/comment-reaction.ts +259 -0
  48. package/src/comment-shared.ts +406 -0
  49. package/src/comment-target.ts +44 -0
  50. package/src/config-schema.ts +335 -0
  51. package/src/conversation-id.ts +199 -0
  52. package/src/dedup-runtime-api.ts +1 -0
  53. package/src/dedup.ts +141 -0
  54. package/src/dedupe-key.ts +72 -0
  55. package/src/directory.static.ts +61 -0
  56. package/src/directory.ts +124 -0
  57. package/src/doc-schema.ts +182 -0
  58. package/src/docx-batch-insert.ts +223 -0
  59. package/src/docx-color-text.ts +154 -0
  60. package/src/docx-table-ops.ts +316 -0
  61. package/src/docx-types.ts +38 -0
  62. package/src/docx.ts +1596 -0
  63. package/src/drive-schema.ts +92 -0
  64. package/src/drive.ts +829 -0
  65. package/src/dynamic-agent.ts +143 -0
  66. package/src/event-types.ts +45 -0
  67. package/src/external-keys.ts +19 -0
  68. package/src/lifecycle.test-support.ts +220 -0
  69. package/src/media.ts +1105 -0
  70. package/src/mention-target.types.ts +5 -0
  71. package/src/mention.ts +114 -0
  72. package/src/message-action-contract.ts +13 -0
  73. package/src/monitor-state-runtime-api.ts +7 -0
  74. package/src/monitor-transport-runtime-api.ts +10 -0
  75. package/src/monitor.account.ts +492 -0
  76. package/src/monitor.acp-init-failure.lifecycle.test-support.ts +219 -0
  77. package/src/monitor.bot-identity.ts +86 -0
  78. package/src/monitor.bot-menu-handler.ts +165 -0
  79. package/src/monitor.bot-menu.lifecycle.test-support.ts +224 -0
  80. package/src/monitor.broadcast.reply-once.lifecycle.test-support.ts +264 -0
  81. package/src/monitor.card-action.lifecycle.test-support.ts +421 -0
  82. package/src/monitor.comment-notice-handler.ts +105 -0
  83. package/src/monitor.comment.ts +1386 -0
  84. package/src/monitor.message-handler.ts +350 -0
  85. package/src/monitor.reaction.lifecycle.test-support.ts +68 -0
  86. package/src/monitor.startup.ts +74 -0
  87. package/src/monitor.state.ts +170 -0
  88. package/src/monitor.synthetic-error.ts +18 -0
  89. package/src/monitor.test-mocks.ts +46 -0
  90. package/src/monitor.transport.ts +451 -0
  91. package/src/monitor.ts +100 -0
  92. package/src/outbound-runtime-api.ts +1 -0
  93. package/src/outbound.ts +785 -0
  94. package/src/perm-schema.ts +52 -0
  95. package/src/perm.ts +170 -0
  96. package/src/pins.ts +108 -0
  97. package/src/policy.ts +321 -0
  98. package/src/post.ts +275 -0
  99. package/src/probe.ts +166 -0
  100. package/src/processing-claims.ts +59 -0
  101. package/src/qr-terminal.ts +1 -0
  102. package/src/reactions.ts +123 -0
  103. package/src/reasoning-preview.ts +28 -0
  104. package/src/reply-dispatcher-runtime-api.ts +7 -0
  105. package/src/reply-dispatcher.ts +748 -0
  106. package/src/runtime.ts +9 -0
  107. package/src/secret-contract.ts +145 -0
  108. package/src/secret-input.ts +1 -0
  109. package/src/security-audit-shared.ts +69 -0
  110. package/src/security-audit.ts +1 -0
  111. package/src/send-result.ts +80 -0
  112. package/src/send-target.ts +35 -0
  113. package/src/send.ts +861 -0
  114. package/src/sequential-key.ts +28 -0
  115. package/src/sequential-queue.ts +86 -0
  116. package/src/session-conversation.ts +42 -0
  117. package/src/session-route.ts +48 -0
  118. package/src/setup-core.ts +51 -0
  119. package/src/setup-surface.ts +618 -0
  120. package/src/streaming-card.ts +571 -0
  121. package/src/subagent-hooks.ts +413 -0
  122. package/src/targets.ts +97 -0
  123. package/src/thread-bindings.ts +331 -0
  124. package/src/tool-account.ts +93 -0
  125. package/src/tool-factory-test-harness.ts +79 -0
  126. package/src/tool-result.ts +16 -0
  127. package/src/tools-config.ts +22 -0
  128. package/src/types.ts +106 -0
  129. package/src/typing.ts +214 -0
  130. package/src/wiki-schema.ts +69 -0
  131. package/src/wiki.ts +270 -0
  132. package/subagent-hooks-api.ts +31 -0
  133. package/tsconfig.json +16 -0
@@ -0,0 +1,143 @@
1
+ import fs from "node:fs";
2
+ import os from "node:os";
3
+ import path from "node:path";
4
+ import type { AutoBotConfig, PluginRuntime } from "../runtime-api.js";
5
+ import type { DynamicAgentCreationConfig } from "./types.js";
6
+
7
+ type MaybeCreateDynamicAgentResult = {
8
+ created: boolean;
9
+ updatedCfg: AutoBotConfig;
10
+ agentId?: string;
11
+ };
12
+
13
+ /**
14
+ * Check if a dynamic agent should be created for a DM user and create it if needed.
15
+ * This creates a unique agent instance with its own workspace for each DM user.
16
+ */
17
+ export async function maybeCreateDynamicAgent(params: {
18
+ cfg: AutoBotConfig;
19
+ runtime: PluginRuntime;
20
+ senderOpenId: string;
21
+ dynamicCfg: DynamicAgentCreationConfig;
22
+ configWritesAllowed: boolean;
23
+ log: (msg: string) => void;
24
+ }): Promise<MaybeCreateDynamicAgentResult> {
25
+ const { cfg, runtime, senderOpenId, dynamicCfg, configWritesAllowed, log } = params;
26
+
27
+ if (!configWritesAllowed) {
28
+ log(`feishu: config writes disabled, not creating agent for ${senderOpenId}`);
29
+ return { created: false, updatedCfg: cfg };
30
+ }
31
+
32
+ // Check if there's already a binding for this user
33
+ const existingBindings = cfg.bindings ?? [];
34
+ const hasBinding = existingBindings.some(
35
+ (b) =>
36
+ b.match?.channel === "feishu" &&
37
+ b.match?.peer?.kind === "direct" &&
38
+ b.match?.peer?.id === senderOpenId,
39
+ );
40
+
41
+ if (hasBinding) {
42
+ return { created: false, updatedCfg: cfg };
43
+ }
44
+
45
+ // Check maxAgents limit if configured
46
+ if (dynamicCfg.maxAgents !== undefined) {
47
+ const feishuAgentCount = (cfg.agents?.list ?? []).filter((a) =>
48
+ a.id.startsWith("feishu-"),
49
+ ).length;
50
+ if (feishuAgentCount >= dynamicCfg.maxAgents) {
51
+ log(
52
+ `feishu: maxAgents limit (${dynamicCfg.maxAgents}) reached, not creating agent for ${senderOpenId}`,
53
+ );
54
+ return { created: false, updatedCfg: cfg };
55
+ }
56
+ }
57
+
58
+ // Use full OpenID as agent ID suffix (OpenID format: ou_xxx is already filesystem-safe)
59
+ const agentId = `feishu-${senderOpenId}`;
60
+
61
+ // Check if agent already exists (but binding was missing)
62
+ const existingAgent = (cfg.agents?.list ?? []).find((a) => a.id === agentId);
63
+ if (existingAgent) {
64
+ // Agent exists but binding doesn't - just add the binding
65
+ log(`feishu: agent "${agentId}" exists, adding missing binding for ${senderOpenId}`);
66
+
67
+ const updatedCfg: AutoBotConfig = {
68
+ ...cfg,
69
+ bindings: [
70
+ ...existingBindings,
71
+ {
72
+ agentId,
73
+ match: {
74
+ channel: "feishu",
75
+ peer: { kind: "direct", id: senderOpenId },
76
+ },
77
+ },
78
+ ],
79
+ };
80
+
81
+ await runtime.config.replaceConfigFile({
82
+ nextConfig: updatedCfg,
83
+ afterWrite: { mode: "auto" },
84
+ });
85
+ return { created: true, updatedCfg, agentId };
86
+ }
87
+
88
+ // Resolve path templates with substitutions
89
+ const workspaceTemplate = dynamicCfg.workspaceTemplate ?? "~/.autobot/workspace-{agentId}";
90
+ const agentDirTemplate = dynamicCfg.agentDirTemplate ?? "~/.autobot/agents/{agentId}/agent";
91
+
92
+ const workspace = resolveUserPath(
93
+ workspaceTemplate.replace("{userId}", senderOpenId).replace("{agentId}", agentId),
94
+ );
95
+ const agentDir = resolveUserPath(
96
+ agentDirTemplate.replace("{userId}", senderOpenId).replace("{agentId}", agentId),
97
+ );
98
+
99
+ log(`feishu: creating dynamic agent "${agentId}" for user ${senderOpenId}`);
100
+ log(` workspace: ${workspace}`);
101
+ log(` agentDir: ${agentDir}`);
102
+
103
+ // Create directories
104
+ await fs.promises.mkdir(workspace, { recursive: true });
105
+ await fs.promises.mkdir(agentDir, { recursive: true });
106
+
107
+ // Update configuration with new agent and binding
108
+ const updatedCfg: AutoBotConfig = {
109
+ ...cfg,
110
+ agents: {
111
+ ...cfg.agents,
112
+ list: [...(cfg.agents?.list ?? []), { id: agentId, workspace, agentDir }],
113
+ },
114
+ bindings: [
115
+ ...existingBindings,
116
+ {
117
+ agentId,
118
+ match: {
119
+ channel: "feishu",
120
+ peer: { kind: "direct", id: senderOpenId },
121
+ },
122
+ },
123
+ ],
124
+ };
125
+
126
+ // Write updated config using PluginRuntime API
127
+ await runtime.config.replaceConfigFile({
128
+ nextConfig: updatedCfg,
129
+ afterWrite: { mode: "auto" },
130
+ });
131
+
132
+ return { created: true, updatedCfg, agentId };
133
+ }
134
+
135
+ /**
136
+ * Resolve a path that may start with ~ to the user's home directory.
137
+ */
138
+ function resolveUserPath(p: string): string {
139
+ if (p.startsWith("~/")) {
140
+ return path.join(os.homedir(), p.slice(2));
141
+ }
142
+ return p;
143
+ }
@@ -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
+ };
@@ -0,0 +1,19 @@
1
+ const CONTROL_CHARS_RE = /\p{Cc}/u;
2
+ const MAX_EXTERNAL_KEY_LENGTH = 512;
3
+
4
+ export function normalizeFeishuExternalKey(value: unknown): string | undefined {
5
+ if (typeof value !== "string") {
6
+ return undefined;
7
+ }
8
+ const normalized = value.trim();
9
+ if (!normalized || normalized.length > MAX_EXTERNAL_KEY_LENGTH) {
10
+ return undefined;
11
+ }
12
+ if (CONTROL_CHARS_RE.test(normalized)) {
13
+ return undefined;
14
+ }
15
+ if (normalized.includes("/") || normalized.includes("\\") || normalized.includes("..")) {
16
+ return undefined;
17
+ }
18
+ return normalized;
19
+ }
@@ -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: "AUTOBOT_FEISHU_HTTP_TIMEOUT_MS",
128
+ FEISHU_HTTP_TIMEOUT_MAX_MS: 300_000,
129
+ FEISHU_HTTP_TIMEOUT_MS: 30_000,
130
+ FEISHU_USER_AGENT: "autobot-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(() => "autobot-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("autobot/plugin-sdk/conversation-runtime", async () => {
168
+ const actual = await vi.importActual<typeof import("autobot/plugin-sdk/conversation-runtime")>(
169
+ "autobot/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
+ });