mirror of https://github.com/openclaw/openclaw.git
225 lines
6.8 KiB
TypeScript
225 lines
6.8 KiB
TypeScript
import "./lifecycle.test-support.js";
|
|
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
|
import { createRuntimeEnv } from "../../../test/helpers/plugins/runtime-env.js";
|
|
import type { ClawdbotConfig, RuntimeEnv } from "../runtime-api.js";
|
|
import { resetProcessedFeishuCardActionTokensForTests } from "./card-action.js";
|
|
import { createFeishuCardInteractionEnvelope } from "./card-interaction.js";
|
|
import { getFeishuLifecycleTestMocks } from "./lifecycle.test-support.js";
|
|
import {
|
|
createFeishuLifecycleConfig,
|
|
createFeishuLifecycleReplyDispatcher,
|
|
createResolvedFeishuLifecycleAccount,
|
|
expectFeishuReplyPipelineDedupedAcrossReplay,
|
|
expectFeishuReplyPipelineDedupedAfterPostSendFailure,
|
|
expectFeishuReplyDispatcherSentFinalReplyOnce,
|
|
installFeishuLifecycleReplyRuntime,
|
|
mockFeishuReplyOnceDispatch,
|
|
restoreFeishuLifecycleStateDir,
|
|
setFeishuLifecycleStateDir,
|
|
setupFeishuLifecycleHandler,
|
|
} from "./test-support/lifecycle-test-support.js";
|
|
import type { ResolvedFeishuAccount } from "./types.js";
|
|
|
|
const {
|
|
createEventDispatcherMock,
|
|
createFeishuReplyDispatcherMock,
|
|
dispatchReplyFromConfigMock,
|
|
finalizeInboundContextMock,
|
|
resolveAgentRouteMock,
|
|
resolveBoundConversationMock,
|
|
sendCardFeishuMock,
|
|
sendMessageFeishuMock,
|
|
touchBindingMock,
|
|
withReplyDispatcherMock,
|
|
} = getFeishuLifecycleTestMocks();
|
|
|
|
let handlers: Record<string, (data: unknown) => Promise<void>> = {};
|
|
let lastRuntime: RuntimeEnv | null = null;
|
|
const originalStateDir = process.env.OPENCLAW_STATE_DIR;
|
|
const lifecycleConfig = createFeishuLifecycleConfig({
|
|
accountId: "acct-card",
|
|
appId: "cli_test",
|
|
appSecret: "secret_test",
|
|
channelConfig: {
|
|
dmPolicy: "open",
|
|
},
|
|
accountConfig: {
|
|
dmPolicy: "open",
|
|
},
|
|
}) as ClawdbotConfig;
|
|
|
|
const lifecycleAccount = createResolvedFeishuLifecycleAccount({
|
|
accountId: "acct-card",
|
|
appId: "cli_test",
|
|
appSecret: "secret_test",
|
|
config: {
|
|
dmPolicy: "open",
|
|
},
|
|
}) as ResolvedFeishuAccount;
|
|
|
|
function createCardActionEvent(params: {
|
|
token: string;
|
|
action: string;
|
|
command: string;
|
|
chatId?: string;
|
|
chatType?: "group" | "p2p";
|
|
}) {
|
|
const openId = "ou_user1";
|
|
const chatId = params.chatId ?? "p2p:ou_user1";
|
|
const chatType = params.chatType ?? "p2p";
|
|
return {
|
|
operator: {
|
|
open_id: openId,
|
|
user_id: "user_1",
|
|
union_id: "union_1",
|
|
},
|
|
token: params.token,
|
|
action: {
|
|
tag: "button",
|
|
value: createFeishuCardInteractionEnvelope({
|
|
k: "quick",
|
|
a: params.action,
|
|
q: params.command,
|
|
c: {
|
|
u: openId,
|
|
h: chatId,
|
|
t: chatType,
|
|
e: Date.now() + 60_000,
|
|
},
|
|
}),
|
|
},
|
|
context: {
|
|
open_id: openId,
|
|
user_id: "user_1",
|
|
chat_id: chatId,
|
|
},
|
|
};
|
|
}
|
|
|
|
async function setupLifecycleMonitor() {
|
|
lastRuntime = createRuntimeEnv();
|
|
return setupFeishuLifecycleHandler({
|
|
createEventDispatcherMock,
|
|
onRegister: (registered) => {
|
|
handlers = registered;
|
|
},
|
|
runtime: lastRuntime,
|
|
cfg: lifecycleConfig,
|
|
account: lifecycleAccount,
|
|
handlerKey: "card.action.trigger",
|
|
missingHandlerMessage: "missing card.action.trigger handler",
|
|
});
|
|
}
|
|
|
|
describe("Feishu card-action lifecycle", () => {
|
|
beforeEach(() => {
|
|
vi.useRealTimers();
|
|
vi.clearAllMocks();
|
|
handlers = {};
|
|
lastRuntime = null;
|
|
resetProcessedFeishuCardActionTokensForTests();
|
|
setFeishuLifecycleStateDir("openclaw-feishu-card-action");
|
|
|
|
createFeishuReplyDispatcherMock.mockReturnValue(createFeishuLifecycleReplyDispatcher());
|
|
|
|
resolveBoundConversationMock.mockImplementation(() => ({
|
|
bindingId: "binding-card",
|
|
targetSessionKey: "agent:bound-agent:feishu:direct:ou_user1",
|
|
}));
|
|
|
|
resolveAgentRouteMock.mockReturnValue({
|
|
agentId: "main",
|
|
channel: "feishu",
|
|
accountId: "acct-card",
|
|
sessionKey: "agent:main:feishu:direct:ou_user1",
|
|
mainSessionKey: "agent:main:main",
|
|
matchedBy: "default",
|
|
});
|
|
|
|
mockFeishuReplyOnceDispatch({
|
|
dispatchReplyFromConfigMock,
|
|
replyText: "card action reply once",
|
|
});
|
|
|
|
withReplyDispatcherMock.mockImplementation(async ({ run }) => await run());
|
|
|
|
installFeishuLifecycleReplyRuntime({
|
|
resolveAgentRouteMock,
|
|
finalizeInboundContextMock,
|
|
dispatchReplyFromConfigMock,
|
|
withReplyDispatcherMock,
|
|
storePath: "/tmp/feishu-card-action-sessions.json",
|
|
});
|
|
});
|
|
|
|
afterEach(() => {
|
|
vi.useRealTimers();
|
|
resetProcessedFeishuCardActionTokensForTests();
|
|
restoreFeishuLifecycleStateDir(originalStateDir);
|
|
});
|
|
|
|
it("routes one reply across duplicate callback delivery", async () => {
|
|
const onCardAction = await setupLifecycleMonitor();
|
|
const event = createCardActionEvent({
|
|
token: "tok-card-once",
|
|
action: "feishu.quick_actions.help",
|
|
command: "/help",
|
|
});
|
|
|
|
await expectFeishuReplyPipelineDedupedAcrossReplay({
|
|
handler: onCardAction,
|
|
event,
|
|
dispatchReplyFromConfigMock,
|
|
createFeishuReplyDispatcherMock,
|
|
});
|
|
|
|
expect(lastRuntime?.error).not.toHaveBeenCalled();
|
|
expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
|
|
expect(createFeishuReplyDispatcherMock).toHaveBeenCalledTimes(1);
|
|
expect(createFeishuReplyDispatcherMock).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
accountId: "acct-card",
|
|
chatId: "p2p:ou_user1",
|
|
replyToMessageId: "card-action-tok-card-once",
|
|
}),
|
|
);
|
|
expect(finalizeInboundContextMock).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
AccountId: "acct-card",
|
|
SessionKey: "agent:bound-agent:feishu:direct:ou_user1",
|
|
MessageSid: "card-action-tok-card-once",
|
|
}),
|
|
);
|
|
expect(touchBindingMock).toHaveBeenCalledWith("binding-card");
|
|
|
|
expectFeishuReplyDispatcherSentFinalReplyOnce({ createFeishuReplyDispatcherMock });
|
|
expect(sendMessageFeishuMock).not.toHaveBeenCalled();
|
|
expect(sendCardFeishuMock).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("does not duplicate delivery when retrying after a post-send failure", async () => {
|
|
const onCardAction = await setupLifecycleMonitor();
|
|
const event = createCardActionEvent({
|
|
token: "tok-card-retry",
|
|
action: "feishu.quick_actions.help",
|
|
command: "/help",
|
|
});
|
|
|
|
dispatchReplyFromConfigMock.mockImplementationOnce(async ({ dispatcher }) => {
|
|
await dispatcher.sendFinalReply({ text: "card action reply once" });
|
|
throw new Error("post-send failure");
|
|
});
|
|
|
|
await expectFeishuReplyPipelineDedupedAfterPostSendFailure({
|
|
handler: onCardAction,
|
|
event,
|
|
dispatchReplyFromConfigMock,
|
|
runtimeErrorMock: lastRuntime?.error as ReturnType<typeof vi.fn>,
|
|
});
|
|
|
|
expect(lastRuntime?.error).toHaveBeenCalledTimes(1);
|
|
expect(dispatchReplyFromConfigMock).toHaveBeenCalledTimes(1);
|
|
expectFeishuReplyDispatcherSentFinalReplyOnce({ createFeishuReplyDispatcherMock });
|
|
});
|
|
});
|