* refactor: move WhatsApp channel from src/web/ to extensions/whatsapp/ Move all WhatsApp implementation code (77 source/test files + 9 channel plugin files) from src/web/ and src/channels/plugins/*/whatsapp* to extensions/whatsapp/src/. - Leave thin re-export shims at all original locations so cross-cutting imports continue to resolve - Update plugin-sdk/whatsapp.ts to only re-export generic framework utilities; channel-specific functions imported locally by the extension - Update vi.mock paths in 15 cross-cutting test files - Rename outbound.ts -> send.ts to match extension naming conventions and avoid false positive in cfg-threading guard test - Widen tsconfig.plugin-sdk.dts.json rootDir to support shim->extension cross-directory references Part of the core-channels-to-extensions migration (PR 6/10). * style: format WhatsApp extension files * fix: correct stale import paths in WhatsApp extension tests Fix vi.importActual, test mock, and hardcoded source paths that weren't updated during the file move: - media.test.ts: vi.importActual path - onboarding.test.ts: vi.importActual path - test-helpers.ts: test/mocks/baileys.js path - monitor-inbox.test-harness.ts: incomplete media/store mock - login.test.ts: hardcoded source file path - message-action-runner.media.test.ts: vi.mock/importActual path
146 lines
5.2 KiB
TypeScript
146 lines
5.2 KiB
TypeScript
import { vi } from "vitest";
|
|
import type { MockBaileysSocket } from "../../../test/mocks/baileys.js";
|
|
import { createMockBaileys } from "../../../test/mocks/baileys.js";
|
|
|
|
// Use globalThis to store the mock config so it survives vi.mock hoisting
|
|
const CONFIG_KEY = Symbol.for("openclaw:testConfigMock");
|
|
const DEFAULT_CONFIG = {
|
|
channels: {
|
|
whatsapp: {
|
|
// Tests can override; default remains open to avoid surprising fixtures
|
|
allowFrom: ["*"],
|
|
},
|
|
},
|
|
messages: {
|
|
messagePrefix: undefined,
|
|
responsePrefix: undefined,
|
|
},
|
|
};
|
|
|
|
// Initialize default if not set
|
|
if (!(globalThis as Record<symbol, unknown>)[CONFIG_KEY]) {
|
|
(globalThis as Record<symbol, unknown>)[CONFIG_KEY] = () => DEFAULT_CONFIG;
|
|
}
|
|
|
|
export function setLoadConfigMock(fn: unknown) {
|
|
(globalThis as Record<symbol, unknown>)[CONFIG_KEY] = typeof fn === "function" ? fn : () => fn;
|
|
}
|
|
|
|
export function resetLoadConfigMock() {
|
|
(globalThis as Record<symbol, unknown>)[CONFIG_KEY] = () => DEFAULT_CONFIG;
|
|
}
|
|
|
|
vi.mock("../../../src/config/config.js", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("../../../src/config/config.js")>();
|
|
return {
|
|
...actual,
|
|
loadConfig: () => {
|
|
const getter = (globalThis as Record<symbol, unknown>)[CONFIG_KEY];
|
|
if (typeof getter === "function") {
|
|
return getter();
|
|
}
|
|
return DEFAULT_CONFIG;
|
|
},
|
|
};
|
|
});
|
|
|
|
// Some web modules live under `src/web/auto-reply/*` and import config via a different
|
|
// relative path (`../../config/config.js`). Mock both specifiers so tests stay stable
|
|
// across refactors that move files between folders.
|
|
vi.mock("../../config/config.js", async (importOriginal) => {
|
|
// `../../config/config.js` is correct for modules under `src/web/auto-reply/*`.
|
|
// For typing in this file (which lives in `src/web/*`), refer to the same module
|
|
// via the local relative path.
|
|
const actual = await importOriginal<typeof import("../../../src/config/config.js")>();
|
|
return {
|
|
...actual,
|
|
loadConfig: () => {
|
|
const getter = (globalThis as Record<symbol, unknown>)[CONFIG_KEY];
|
|
if (typeof getter === "function") {
|
|
return getter();
|
|
}
|
|
return DEFAULT_CONFIG;
|
|
},
|
|
};
|
|
});
|
|
|
|
vi.mock("../../../src/media/store.js", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("../../../src/media/store.js")>();
|
|
const mockModule = Object.create(null) as Record<string, unknown>;
|
|
Object.defineProperties(mockModule, Object.getOwnPropertyDescriptors(actual));
|
|
Object.defineProperty(mockModule, "saveMediaBuffer", {
|
|
configurable: true,
|
|
enumerable: true,
|
|
writable: true,
|
|
value: vi.fn().mockImplementation(async (_buf: Buffer, contentType?: string) => ({
|
|
id: "mid",
|
|
path: "/tmp/mid",
|
|
size: _buf.length,
|
|
contentType,
|
|
})),
|
|
});
|
|
return mockModule;
|
|
});
|
|
|
|
vi.mock("@whiskeysockets/baileys", () => {
|
|
const created = createMockBaileys();
|
|
(globalThis as Record<PropertyKey, unknown>)[Symbol.for("openclaw:lastSocket")] =
|
|
created.lastSocket;
|
|
return created.mod;
|
|
});
|
|
|
|
vi.mock("qrcode-terminal", () => ({
|
|
default: { generate: vi.fn() },
|
|
generate: vi.fn(),
|
|
}));
|
|
|
|
export const baileys = await import("@whiskeysockets/baileys");
|
|
|
|
export function resetBaileysMocks() {
|
|
const recreated = createMockBaileys();
|
|
(globalThis as Record<PropertyKey, unknown>)[Symbol.for("openclaw:lastSocket")] =
|
|
recreated.lastSocket;
|
|
|
|
const makeWASocket = vi.mocked(baileys.makeWASocket);
|
|
const makeWASocketImpl: typeof baileys.makeWASocket = (...args) =>
|
|
(recreated.mod.makeWASocket as unknown as typeof baileys.makeWASocket)(...args);
|
|
makeWASocket.mockReset();
|
|
makeWASocket.mockImplementation(makeWASocketImpl);
|
|
|
|
const useMultiFileAuthState = vi.mocked(baileys.useMultiFileAuthState);
|
|
const useMultiFileAuthStateImpl: typeof baileys.useMultiFileAuthState = (...args) =>
|
|
(recreated.mod.useMultiFileAuthState as unknown as typeof baileys.useMultiFileAuthState)(
|
|
...args,
|
|
);
|
|
useMultiFileAuthState.mockReset();
|
|
useMultiFileAuthState.mockImplementation(useMultiFileAuthStateImpl);
|
|
|
|
const fetchLatestBaileysVersion = vi.mocked(baileys.fetchLatestBaileysVersion);
|
|
const fetchLatestBaileysVersionImpl: typeof baileys.fetchLatestBaileysVersion = (...args) =>
|
|
(
|
|
recreated.mod.fetchLatestBaileysVersion as unknown as typeof baileys.fetchLatestBaileysVersion
|
|
)(...args);
|
|
fetchLatestBaileysVersion.mockReset();
|
|
fetchLatestBaileysVersion.mockImplementation(fetchLatestBaileysVersionImpl);
|
|
|
|
const makeCacheableSignalKeyStore = vi.mocked(baileys.makeCacheableSignalKeyStore);
|
|
const makeCacheableSignalKeyStoreImpl: typeof baileys.makeCacheableSignalKeyStore = (...args) =>
|
|
(
|
|
recreated.mod
|
|
.makeCacheableSignalKeyStore as unknown as typeof baileys.makeCacheableSignalKeyStore
|
|
)(...args);
|
|
makeCacheableSignalKeyStore.mockReset();
|
|
makeCacheableSignalKeyStore.mockImplementation(makeCacheableSignalKeyStoreImpl);
|
|
}
|
|
|
|
export function getLastSocket(): MockBaileysSocket {
|
|
const getter = (globalThis as Record<PropertyKey, unknown>)[Symbol.for("openclaw:lastSocket")];
|
|
if (typeof getter === "function") {
|
|
return (getter as () => MockBaileysSocket)();
|
|
}
|
|
if (!getter) {
|
|
throw new Error("Baileys mock not initialized");
|
|
}
|
|
throw new Error("Invalid Baileys socket getter");
|
|
}
|