import { Mock, vi } from "vitest"; type SlackHandler = (args: unknown) => Promise; type SlackProviderMonitor = (params: { botToken: string; appToken: string; abortSignal: AbortSignal; config?: Record; }) => Promise; type SlackTestState = { config: Record; sendMock: Mock<(...args: unknown[]) => Promise>; replyMock: Mock<(...args: unknown[]) => unknown>; updateLastRouteMock: Mock<(...args: unknown[]) => unknown>; reactMock: Mock<(...args: unknown[]) => unknown>; readAllowFromStoreMock: Mock<(...args: unknown[]) => Promise>; upsertPairingRequestMock: Mock<(...args: unknown[]) => Promise>; }; const slackTestState: SlackTestState = vi.hoisted(() => ({ config: {} as Record, sendMock: vi.fn(), replyMock: vi.fn(), updateLastRouteMock: vi.fn(), reactMock: vi.fn(), readAllowFromStoreMock: vi.fn(), upsertPairingRequestMock: vi.fn(), })); export const getSlackTestState = (): SlackTestState => slackTestState; type SlackClient = { auth: { test: Mock<(...args: unknown[]) => Promise>> }; conversations: { info: Mock<(...args: unknown[]) => Promise>>; replies: Mock<(...args: unknown[]) => Promise>>; history: Mock<(...args: unknown[]) => Promise>>; }; users: { info: Mock<(...args: unknown[]) => Promise<{ user: { profile: { display_name: string } } }>>; }; assistant: { threads: { setStatus: Mock<(...args: unknown[]) => Promise<{ ok: boolean }>>; }; }; reactions: { add: (...args: unknown[]) => unknown; }; }; export const getSlackHandlers = () => ensureSlackTestRuntime().handlers; export const getSlackClient = () => ensureSlackTestRuntime().client; function ensureSlackTestRuntime(): { handlers: Map; client: SlackClient; } { const globalState = globalThis as { __slackHandlers?: Map; __slackClient?: SlackClient; }; if (!globalState.__slackHandlers) { globalState.__slackHandlers = new Map(); } if (!globalState.__slackClient) { globalState.__slackClient = { auth: { test: vi.fn().mockResolvedValue({ user_id: "bot-user" }) }, conversations: { info: vi.fn().mockResolvedValue({ channel: { name: "dm", is_im: true }, }), replies: vi.fn().mockResolvedValue({ messages: [] }), history: vi.fn().mockResolvedValue({ messages: [] }), }, users: { info: vi.fn().mockResolvedValue({ user: { profile: { display_name: "Ada" } }, }), }, assistant: { threads: { setStatus: vi.fn().mockResolvedValue({ ok: true }), }, }, reactions: { add: (...args: unknown[]) => slackTestState.reactMock(...args), }, }; } return { handlers: globalState.__slackHandlers, client: globalState.__slackClient, }; } export const flush = () => new Promise((resolve) => setTimeout(resolve, 0)); export async function waitForSlackEvent(name: string) { for (let i = 0; i < 10; i += 1) { if (getSlackHandlers()?.has(name)) { return; } await flush(); } } export function startSlackMonitor( monitorSlackProvider: SlackProviderMonitor, opts?: { botToken?: string; appToken?: string }, ) { const controller = new AbortController(); const run = monitorSlackProvider({ botToken: opts?.botToken ?? "bot-token", appToken: opts?.appToken ?? "app-token", abortSignal: controller.signal, config: slackTestState.config, }); return { controller, run }; } export async function getSlackHandlerOrThrow(name: string) { await waitForSlackEvent(name); const handler = getSlackHandlers()?.get(name); if (!handler) { throw new Error(`Slack ${name} handler not registered`); } return handler; } export async function stopSlackMonitor(params: { controller: AbortController; run: Promise; }) { await flush(); params.controller.abort(); await params.run; } export async function runSlackEventOnce( monitorSlackProvider: SlackProviderMonitor, name: string, args: unknown, opts?: { botToken?: string; appToken?: string }, ) { const { controller, run } = startSlackMonitor(monitorSlackProvider, opts); const handler = await getSlackHandlerOrThrow(name); await handler(args); await stopSlackMonitor({ controller, run }); } export async function runSlackMessageOnce( monitorSlackProvider: SlackProviderMonitor, args: unknown, opts?: { botToken?: string; appToken?: string }, ) { await runSlackEventOnce(monitorSlackProvider, "message", args, opts); } export const defaultSlackTestConfig = () => ({ messages: { responsePrefix: "PFX", ackReaction: "👀", ackReactionScope: "group-mentions", }, channels: { slack: { dm: { enabled: true, policy: "open", allowFrom: ["*"] }, groupPolicy: "open", }, }, }); export function resetSlackTestState(config: Record = defaultSlackTestConfig()) { slackTestState.config = config; slackTestState.sendMock.mockReset().mockResolvedValue(undefined); slackTestState.replyMock.mockReset(); slackTestState.updateLastRouteMock.mockReset(); slackTestState.reactMock.mockReset(); slackTestState.readAllowFromStoreMock.mockReset().mockResolvedValue([]); slackTestState.upsertPairingRequestMock.mockReset().mockResolvedValue({ code: "PAIRCODE", created: true, }); getSlackHandlers()?.clear(); } vi.mock("../../../src/config/config.js", async (importOriginal) => { const actual = await importOriginal(); return { ...actual, loadConfig: () => slackTestState.config, }; }); vi.mock("../../../src/auto-reply/reply.js", () => ({ getReplyFromConfig: (...args: unknown[]) => slackTestState.replyMock(...args), })); vi.mock("./resolve-channels.js", () => ({ resolveSlackChannelAllowlist: async ({ entries }: { entries: string[] }) => entries.map((input) => ({ input, resolved: false })), })); vi.mock("./resolve-users.js", () => ({ resolveSlackUserAllowlist: async ({ entries }: { entries: string[] }) => entries.map((input) => ({ input, resolved: false })), })); vi.mock("./send.js", () => ({ sendMessageSlack: (...args: unknown[]) => slackTestState.sendMock(...args), })); vi.mock("../../../src/pairing/pairing-store.js", () => ({ readChannelAllowFromStore: (...args: unknown[]) => slackTestState.readAllowFromStoreMock(...args), upsertChannelPairingRequest: (...args: unknown[]) => slackTestState.upsertPairingRequestMock(...args), })); vi.mock("../../../src/config/sessions.js", async (importOriginal) => { const actual = await importOriginal(); return { ...actual, resolveStorePath: vi.fn(() => "/tmp/openclaw-sessions.json"), updateLastRoute: (...args: unknown[]) => slackTestState.updateLastRouteMock(...args), resolveSessionKey: vi.fn(), readSessionUpdatedAt: vi.fn(() => undefined), recordSessionMetaFromInbound: vi.fn().mockResolvedValue(undefined), }; }); vi.mock("@slack/bolt", () => { const { handlers, client: slackClient } = ensureSlackTestRuntime(); class App { client = slackClient; event(name: string, handler: SlackHandler) { handlers.set(name, handler); } command() { /* no-op */ } start = vi.fn().mockResolvedValue(undefined); stop = vi.fn().mockResolvedValue(undefined); } class HTTPReceiver { requestListener = vi.fn(); } return { App, HTTPReceiver, default: { App, HTTPReceiver } }; });