import "./test-helpers.js"; import crypto from "node:crypto"; import fs from "node:fs/promises"; import os from "node:os"; import path from "node:path"; import { afterEach, beforeEach, describe, expect, it, vi } from "vitest"; vi.mock("../agents/pi-embedded.js", () => ({ abortEmbeddedPiRun: vi.fn().mockReturnValue(false), isEmbeddedPiRunActive: vi.fn().mockReturnValue(false), isEmbeddedPiRunStreaming: vi.fn().mockReturnValue(false), runEmbeddedPiAgent: vi.fn(), queueEmbeddedPiMessage: vi.fn().mockReturnValue(false), resolveEmbeddedSessionLane: (key: string) => `session:${key.trim() || "main"}`, })); import { resetInboundDedupe } from "../auto-reply/reply/inbound-dedupe.js"; import { expectInboundContextContract } from "../../test/helpers/inbound-contract.js"; import { resetLogger, setLoggerOverride } from "../logging.js"; import { monitorWebChannel, SILENT_REPLY_TOKEN } from "./auto-reply.js"; import { resetBaileysMocks, resetLoadConfigMock, setLoadConfigMock } from "./test-helpers.js"; let previousHome: string | undefined; let tempHome: string | undefined; const rmDirWithRetries = async (dir: string): Promise => { // Some tests can leave async session-store writes in-flight; recursive deletion can race and throw ENOTEMPTY. for (let attempt = 0; attempt < 10; attempt += 1) { try { await fs.rm(dir, { recursive: true, force: true }); return; } catch (err) { const code = err && typeof err === "object" && "code" in err ? String((err as { code?: unknown }).code) : null; if (code === "ENOTEMPTY" || code === "EBUSY" || code === "EPERM") { await new Promise((resolve) => setTimeout(resolve, 25)); continue; } throw err; } } await fs.rm(dir, { recursive: true, force: true }); }; beforeEach(async () => { resetInboundDedupe(); previousHome = process.env.HOME; tempHome = await fs.mkdtemp(path.join(os.tmpdir(), "clawdbot-web-home-")); process.env.HOME = tempHome; }); afterEach(async () => { process.env.HOME = previousHome; if (tempHome) { await rmDirWithRetries(tempHome); tempHome = undefined; } }); const makeSessionStore = async ( entries: Record = {}, ): Promise<{ storePath: string; cleanup: () => Promise }> => { const dir = await fs.mkdtemp(path.join(os.tmpdir(), "clawdbot-session-")); const storePath = path.join(dir, "sessions.json"); await fs.writeFile(storePath, JSON.stringify(entries)); const cleanup = async () => { // Session store writes can be in-flight when the test finishes (e.g. updateLastRoute // after a message flush). `fs.rm({ recursive })` can race and throw ENOTEMPTY. for (let attempt = 0; attempt < 10; attempt += 1) { try { await fs.rm(dir, { recursive: true, force: true }); return; } catch (err) { const code = err && typeof err === "object" && "code" in err ? String((err as { code?: unknown }).code) : null; if (code === "ENOTEMPTY" || code === "EBUSY" || code === "EPERM") { await new Promise((resolve) => setTimeout(resolve, 25)); continue; } throw err; } } await fs.rm(dir, { recursive: true, force: true }); }; return { storePath, cleanup, }; }; describe("web auto-reply", () => { beforeEach(() => { vi.clearAllMocks(); resetBaileysMocks(); resetLoadConfigMock(); }); afterEach(() => { resetLogger(); setLoggerOverride(null); vi.useRealTimers(); }); it("supports always-on group activation with silent token and clears pending history", async () => { const sendMedia = vi.fn(); const reply = vi.fn().mockResolvedValue(undefined); const sendComposing = vi.fn(); const resolver = vi .fn() .mockResolvedValueOnce({ text: SILENT_REPLY_TOKEN }) .mockResolvedValueOnce({ text: "ok" }); const { storePath, cleanup } = await makeSessionStore({ "agent:main:whatsapp:group:123@g.us": { sessionId: "g-1", updatedAt: Date.now(), groupActivation: "always", }, }); setLoadConfigMock(() => ({ messages: { groupChat: { mentionPatterns: ["@clawd"] }, }, session: { store: storePath }, })); let capturedOnMessage: | ((msg: import("./inbound.js").WebInboundMessage) => Promise) | undefined; const listenerFactory = async (opts: { onMessage: (msg: import("./inbound.js").WebInboundMessage) => Promise; }) => { capturedOnMessage = opts.onMessage; return { close: vi.fn() }; }; await monitorWebChannel(false, listenerFactory, false, resolver); expect(capturedOnMessage).toBeDefined(); await capturedOnMessage?.({ body: "first", from: "123@g.us", conversationId: "123@g.us", chatId: "123@g.us", chatType: "group", to: "+2", id: "g-always-1", senderE164: "+111", senderName: "Alice", selfE164: "+999", sendComposing, reply, sendMedia, }); expect(resolver).toHaveBeenCalledTimes(1); expect(reply).not.toHaveBeenCalled(); await capturedOnMessage?.({ body: "second", from: "123@g.us", conversationId: "123@g.us", chatId: "123@g.us", chatType: "group", to: "+2", id: "g-always-2", senderE164: "+222", senderName: "Bob", selfE164: "+999", sendComposing, reply, sendMedia, }); expect(resolver).toHaveBeenCalledTimes(2); const payload = resolver.mock.calls[1][0]; expect(payload.Body).not.toContain("Chat messages since your last reply"); expect(payload.Body).not.toContain("Alice (+111): first"); expect(payload.Body).not.toContain("[message_id: g-always-1]"); expect(payload.Body).toContain("second"); expectInboundContextContract(payload); expect(payload.SenderName).toBe("Bob"); expect(payload.SenderE164).toBe("+222"); expect(reply).toHaveBeenCalledTimes(1); await cleanup(); resetLoadConfigMock(); }); it("ignores JID mentions in self-chat mode (group chats)", async () => { const sendMedia = vi.fn(); const reply = vi.fn().mockResolvedValue(undefined); const sendComposing = vi.fn(); const resolver = vi.fn().mockResolvedValue({ text: "ok" }); setLoadConfigMock(() => ({ channels: { whatsapp: { // Self-chat heuristic: allowFrom includes selfE164. allowFrom: ["+999"], groups: { "*": { requireMention: true } }, }, }, messages: { groupChat: { mentionPatterns: ["\\bclawd\\b"], }, }, })); let capturedOnMessage: | ((msg: import("./inbound.js").WebInboundMessage) => Promise) | undefined; const listenerFactory = async (opts: { onMessage: (msg: import("./inbound.js").WebInboundMessage) => Promise; }) => { capturedOnMessage = opts.onMessage; return { close: vi.fn() }; }; await monitorWebChannel(false, listenerFactory, false, resolver); expect(capturedOnMessage).toBeDefined(); // WhatsApp @mention of the owner should NOT trigger the bot in self-chat mode. await capturedOnMessage?.({ body: "@owner ping", from: "123@g.us", conversationId: "123@g.us", chatId: "123@g.us", chatType: "group", to: "+2", id: "g-self-1", senderE164: "+111", senderName: "Alice", mentionedJids: ["999@s.whatsapp.net"], selfE164: "+999", selfJid: "999@s.whatsapp.net", sendComposing, reply, sendMedia, }); expect(resolver).not.toHaveBeenCalled(); // Text-based mentionPatterns still work (user can type "clawd" explicitly). await capturedOnMessage?.({ body: "clawd ping", from: "123@g.us", conversationId: "123@g.us", chatId: "123@g.us", chatType: "group", to: "+2", id: "g-self-2", senderE164: "+222", senderName: "Bob", selfE164: "+999", selfJid: "999@s.whatsapp.net", sendComposing, reply, sendMedia, }); expect(resolver).toHaveBeenCalledTimes(1); resetLoadConfigMock(); }); it("emits heartbeat logs with connection metadata", async () => { vi.useFakeTimers(); const logPath = `/tmp/clawdbot-heartbeat-${crypto.randomUUID()}.log`; setLoggerOverride({ level: "trace", file: logPath }); const runtime = { log: vi.fn(), error: vi.fn(), exit: vi.fn(), }; const controller = new AbortController(); const listenerFactory = vi.fn(async () => { const onClose = new Promise(() => { // never resolves; abort will short-circuit }); return { close: vi.fn(), onClose }; }); const run = monitorWebChannel( false, listenerFactory, true, async () => ({ text: "ok" }), runtime as never, controller.signal, { heartbeatSeconds: 1, reconnect: { initialMs: 5, maxMs: 5, maxAttempts: 1, factor: 1.1 }, }, ); await vi.advanceTimersByTimeAsync(1_000); controller.abort(); await vi.runAllTimersAsync(); await run.catch(() => {}); const content = await fs.readFile(logPath, "utf-8"); expect(content).toMatch(/web-heartbeat/); expect(content).toMatch(/connectionId/); expect(content).toMatch(/messagesHandled/); }); it("logs outbound replies to file", async () => { const logPath = `/tmp/clawdbot-log-test-${crypto.randomUUID()}.log`; setLoggerOverride({ level: "trace", file: logPath }); let capturedOnMessage: | ((msg: import("./inbound.js").WebInboundMessage) => Promise) | undefined; const listenerFactory = async (opts: { onMessage: (msg: import("./inbound.js").WebInboundMessage) => Promise; }) => { capturedOnMessage = opts.onMessage; return { close: vi.fn() }; }; const resolver = vi.fn().mockResolvedValue({ text: "auto" }); await monitorWebChannel(false, listenerFactory, false, resolver); expect(capturedOnMessage).toBeDefined(); await capturedOnMessage?.({ body: "hello", from: "+1", to: "+2", id: "msg1", sendComposing: vi.fn(), reply: vi.fn(), sendMedia: vi.fn(), }); const content = await fs.readFile(logPath, "utf-8"); expect(content).toMatch(/web-auto-reply/); expect(content).toMatch(/auto/); }); });