@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.
- package/api.ts +32 -0
- package/autobot.plugin.json +180 -0
- package/channel-entry.ts +20 -0
- package/channel-plugin-api.ts +1 -0
- package/contract-api.ts +16 -0
- package/index.ts +82 -0
- package/package.json +62 -0
- package/runtime-api.ts +55 -0
- package/secret-contract-api.ts +5 -0
- package/security-contract-api.ts +1 -0
- package/session-key-api.ts +1 -0
- package/setup-api.ts +3 -0
- package/setup-entry.ts +13 -0
- package/skills/feishu-doc/SKILL.md +211 -0
- package/skills/feishu-doc/references/block-types.md +103 -0
- package/skills/feishu-drive/SKILL.md +97 -0
- package/skills/feishu-perm/SKILL.md +119 -0
- package/skills/feishu-wiki/SKILL.md +113 -0
- package/src/accounts.ts +333 -0
- package/src/agent-config.ts +21 -0
- package/src/app-registration.ts +331 -0
- package/src/approval-auth.ts +25 -0
- package/src/async.ts +104 -0
- package/src/audio-preflight.runtime.ts +9 -0
- package/src/bitable.ts +762 -0
- package/src/bot-content.ts +485 -0
- package/src/bot-runtime-api.ts +12 -0
- package/src/bot-sender-name.ts +125 -0
- package/src/bot.ts +1703 -0
- package/src/card-action.ts +447 -0
- package/src/card-interaction.ts +159 -0
- package/src/card-test-helpers.ts +54 -0
- package/src/card-ux-approval.ts +65 -0
- package/src/card-ux-launcher.ts +121 -0
- package/src/card-ux-shared.ts +33 -0
- package/src/channel-runtime-api.ts +16 -0
- package/src/channel.runtime.ts +47 -0
- package/src/channel.ts +1423 -0
- package/src/chat-schema.ts +25 -0
- package/src/chat.ts +188 -0
- package/src/client-timeout.ts +42 -0
- package/src/client.ts +262 -0
- package/src/comment-dispatcher-runtime-api.ts +6 -0
- package/src/comment-dispatcher.ts +107 -0
- package/src/comment-handler-runtime-api.ts +3 -0
- package/src/comment-handler.ts +303 -0
- package/src/comment-reaction.ts +259 -0
- package/src/comment-shared.ts +406 -0
- package/src/comment-target.ts +44 -0
- package/src/config-schema.ts +335 -0
- package/src/conversation-id.ts +199 -0
- package/src/dedup-runtime-api.ts +1 -0
- package/src/dedup.ts +141 -0
- package/src/dedupe-key.ts +72 -0
- package/src/directory.static.ts +61 -0
- package/src/directory.ts +124 -0
- package/src/doc-schema.ts +182 -0
- package/src/docx-batch-insert.ts +223 -0
- package/src/docx-color-text.ts +154 -0
- package/src/docx-table-ops.ts +316 -0
- package/src/docx-types.ts +38 -0
- package/src/docx.ts +1596 -0
- package/src/drive-schema.ts +92 -0
- package/src/drive.ts +829 -0
- package/src/dynamic-agent.ts +143 -0
- package/src/event-types.ts +45 -0
- package/src/external-keys.ts +19 -0
- package/src/lifecycle.test-support.ts +220 -0
- package/src/media.ts +1105 -0
- package/src/mention-target.types.ts +5 -0
- package/src/mention.ts +114 -0
- package/src/message-action-contract.ts +13 -0
- package/src/monitor-state-runtime-api.ts +7 -0
- package/src/monitor-transport-runtime-api.ts +10 -0
- package/src/monitor.account.ts +492 -0
- package/src/monitor.acp-init-failure.lifecycle.test-support.ts +219 -0
- package/src/monitor.bot-identity.ts +86 -0
- package/src/monitor.bot-menu-handler.ts +165 -0
- package/src/monitor.bot-menu.lifecycle.test-support.ts +224 -0
- package/src/monitor.broadcast.reply-once.lifecycle.test-support.ts +264 -0
- package/src/monitor.card-action.lifecycle.test-support.ts +421 -0
- package/src/monitor.comment-notice-handler.ts +105 -0
- package/src/monitor.comment.ts +1386 -0
- package/src/monitor.message-handler.ts +350 -0
- package/src/monitor.reaction.lifecycle.test-support.ts +68 -0
- package/src/monitor.startup.ts +74 -0
- package/src/monitor.state.ts +170 -0
- package/src/monitor.synthetic-error.ts +18 -0
- package/src/monitor.test-mocks.ts +46 -0
- package/src/monitor.transport.ts +451 -0
- package/src/monitor.ts +100 -0
- package/src/outbound-runtime-api.ts +1 -0
- package/src/outbound.ts +785 -0
- package/src/perm-schema.ts +52 -0
- package/src/perm.ts +170 -0
- package/src/pins.ts +108 -0
- package/src/policy.ts +321 -0
- package/src/post.ts +275 -0
- package/src/probe.ts +166 -0
- package/src/processing-claims.ts +59 -0
- package/src/qr-terminal.ts +1 -0
- package/src/reactions.ts +123 -0
- package/src/reasoning-preview.ts +28 -0
- package/src/reply-dispatcher-runtime-api.ts +7 -0
- package/src/reply-dispatcher.ts +748 -0
- package/src/runtime.ts +9 -0
- package/src/secret-contract.ts +145 -0
- package/src/secret-input.ts +1 -0
- package/src/security-audit-shared.ts +69 -0
- package/src/security-audit.ts +1 -0
- package/src/send-result.ts +80 -0
- package/src/send-target.ts +35 -0
- package/src/send.ts +861 -0
- package/src/sequential-key.ts +28 -0
- package/src/sequential-queue.ts +86 -0
- package/src/session-conversation.ts +42 -0
- package/src/session-route.ts +48 -0
- package/src/setup-core.ts +51 -0
- package/src/setup-surface.ts +618 -0
- package/src/streaming-card.ts +571 -0
- package/src/subagent-hooks.ts +413 -0
- package/src/targets.ts +97 -0
- package/src/thread-bindings.ts +331 -0
- package/src/tool-account.ts +93 -0
- package/src/tool-factory-test-harness.ts +79 -0
- package/src/tool-result.ts +16 -0
- package/src/tools-config.ts +22 -0
- package/src/types.ts +106 -0
- package/src/typing.ts +214 -0
- package/src/wiki-schema.ts +69 -0
- package/src/wiki.ts +270 -0
- package/subagent-hooks-api.ts +31 -0
- 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
|
+
});
|