mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-19 05:01:15 +00:00
* Telegram: stabilize Area 2 DM and model callbacks * Telegram: fix dispatch test deps wiring * Telegram: stabilize area2 test harness and gate flaky sticker e2e * Telegram: address review feedback on config reload and tests * Telegram tests: use plugin-sdk reply dispatcher import * Telegram tests: add routing reload regression and track sticker skips * Telegram: add polling-session backoff regression test * Telegram tests: mock loadWebMedia through plugin-sdk path * Telegram: refresh native and callback routing config * Telegram tests: fix compact callback config typing
253 lines
8.7 KiB
TypeScript
253 lines
8.7 KiB
TypeScript
import path from "node:path";
|
|
import type { OpenClawConfig } from "openclaw/plugin-sdk/config-runtime";
|
|
import { resetInboundDedupe } from "openclaw/plugin-sdk/reply-runtime";
|
|
import type { GetReplyOptions, MsgContext } from "openclaw/plugin-sdk/reply-runtime";
|
|
import { beforeEach, vi, type Mock } from "vitest";
|
|
import type { TelegramBotDeps } from "./bot-deps.js";
|
|
|
|
type TelegramBotRuntimeForTest = NonNullable<
|
|
Parameters<typeof import("./bot.js").setTelegramBotRuntimeForTest>[0]
|
|
>;
|
|
type DispatchReplyWithBufferedBlockDispatcherFn =
|
|
typeof import("openclaw/plugin-sdk/reply-runtime").dispatchReplyWithBufferedBlockDispatcher;
|
|
type DispatchReplyHarnessParams = Parameters<DispatchReplyWithBufferedBlockDispatcherFn>[0];
|
|
type FetchRemoteMediaFn = typeof import("openclaw/plugin-sdk/media-runtime").fetchRemoteMedia;
|
|
|
|
export const useSpy: Mock = vi.fn();
|
|
export const middlewareUseSpy: Mock = vi.fn();
|
|
export const onSpy: Mock = vi.fn();
|
|
export const stopSpy: Mock = vi.fn();
|
|
export const sendChatActionSpy: Mock = vi.fn();
|
|
|
|
function defaultUndiciFetch(input: RequestInfo | URL, init?: RequestInit) {
|
|
return globalThis.fetch(input, init);
|
|
}
|
|
|
|
export const undiciFetchSpy: Mock = vi.fn(defaultUndiciFetch);
|
|
|
|
export function resetUndiciFetchMock() {
|
|
undiciFetchSpy.mockReset();
|
|
undiciFetchSpy.mockImplementation(defaultUndiciFetch);
|
|
}
|
|
|
|
async function defaultFetchRemoteMedia(
|
|
params: Parameters<FetchRemoteMediaFn>[0],
|
|
): ReturnType<FetchRemoteMediaFn> {
|
|
if (!params.fetchImpl) {
|
|
throw new Error(`Missing fetchImpl for ${params.url}`);
|
|
}
|
|
const response = await params.fetchImpl(params.url, { redirect: "manual" });
|
|
if (!response.ok) {
|
|
throw new Error(
|
|
`Failed to fetch media from ${params.url}: HTTP ${response.status} ${response.statusText}`,
|
|
);
|
|
}
|
|
const arrayBuffer = await response.arrayBuffer();
|
|
return {
|
|
buffer: Buffer.from(arrayBuffer),
|
|
contentType: response.headers.get("content-type") ?? undefined,
|
|
fileName: params.filePathHint ? path.basename(params.filePathHint) : undefined,
|
|
} as Awaited<ReturnType<FetchRemoteMediaFn>>;
|
|
}
|
|
|
|
export const fetchRemoteMediaSpy: Mock = vi.fn(defaultFetchRemoteMedia);
|
|
|
|
export function resetFetchRemoteMediaMock() {
|
|
fetchRemoteMediaSpy.mockReset();
|
|
fetchRemoteMediaSpy.mockImplementation(defaultFetchRemoteMedia);
|
|
}
|
|
|
|
async function defaultSaveMediaBuffer(buffer: Buffer, contentType?: string) {
|
|
return {
|
|
id: "media",
|
|
path: "/tmp/telegram-media",
|
|
size: buffer.byteLength,
|
|
contentType: contentType ?? "application/octet-stream",
|
|
};
|
|
}
|
|
|
|
const saveMediaBufferSpy: Mock = vi.fn(defaultSaveMediaBuffer);
|
|
|
|
export function setNextSavedMediaPath(params: {
|
|
path: string;
|
|
id?: string;
|
|
contentType?: string;
|
|
size?: number;
|
|
}) {
|
|
saveMediaBufferSpy.mockImplementationOnce(
|
|
async (buffer: Buffer, detectedContentType?: string) => ({
|
|
id: params.id ?? "media",
|
|
path: params.path,
|
|
size: params.size ?? buffer.byteLength,
|
|
contentType: params.contentType ?? detectedContentType ?? "application/octet-stream",
|
|
}),
|
|
);
|
|
}
|
|
|
|
export function resetSaveMediaBufferMock() {
|
|
saveMediaBufferSpy.mockReset();
|
|
saveMediaBufferSpy.mockImplementation(defaultSaveMediaBuffer);
|
|
}
|
|
|
|
type ApiStub = {
|
|
config: { use: (arg: unknown) => void };
|
|
sendChatAction: Mock;
|
|
sendMessage: Mock;
|
|
setMyCommands: (commands: Array<{ command: string; description: string }>) => Promise<void>;
|
|
};
|
|
|
|
const apiStub: ApiStub = {
|
|
config: { use: useSpy },
|
|
sendChatAction: sendChatActionSpy,
|
|
sendMessage: vi.fn(async () => ({ message_id: 1 })),
|
|
setMyCommands: vi.fn(async () => undefined),
|
|
};
|
|
|
|
const throttlerSpy = vi.fn(() => "throttler");
|
|
|
|
export const telegramBotRuntimeForTest: TelegramBotRuntimeForTest = {
|
|
Bot: class {
|
|
api = apiStub;
|
|
use = middlewareUseSpy;
|
|
on = onSpy;
|
|
command = vi.fn();
|
|
stop = stopSpy;
|
|
catch = vi.fn();
|
|
constructor(public token: string) {}
|
|
} as unknown as TelegramBotRuntimeForTest["Bot"],
|
|
sequentialize: (() => vi.fn()) as TelegramBotRuntimeForTest["sequentialize"],
|
|
apiThrottler: (() => throttlerSpy()) as unknown as TelegramBotRuntimeForTest["apiThrottler"],
|
|
};
|
|
|
|
const mediaHarnessReplySpy = vi.hoisted(() =>
|
|
vi.fn(async (_ctx: MsgContext, opts?: GetReplyOptions) => {
|
|
await opts?.onReplyStart?.();
|
|
return undefined;
|
|
}),
|
|
);
|
|
|
|
const mediaHarnessDispatchReplyWithBufferedBlockDispatcher = vi.hoisted(() =>
|
|
vi.fn<DispatchReplyWithBufferedBlockDispatcherFn>(async (params: DispatchReplyHarnessParams) => {
|
|
await params.dispatcherOptions.typingCallbacks?.onReplyStart?.();
|
|
const reply = await mediaHarnessReplySpy(params.ctx, params.replyOptions);
|
|
const payloads = reply === undefined ? [] : Array.isArray(reply) ? reply : [reply];
|
|
for (const payload of payloads) {
|
|
await params.dispatcherOptions?.deliver?.(payload, { kind: "final" });
|
|
}
|
|
return {
|
|
queuedFinal: payloads.length > 0,
|
|
counts: { block: 0, final: payloads.length, tool: 0 },
|
|
};
|
|
}),
|
|
);
|
|
|
|
export const telegramBotDepsForTest: TelegramBotDeps = {
|
|
loadConfig: (() =>
|
|
({
|
|
channels: { telegram: { dmPolicy: "open", allowFrom: ["*"] } },
|
|
}) as OpenClawConfig) as TelegramBotDeps["loadConfig"],
|
|
resolveStorePath: vi.fn(
|
|
(storePath?: string) => storePath ?? "/tmp/telegram-media-sessions.json",
|
|
) as TelegramBotDeps["resolveStorePath"],
|
|
readChannelAllowFromStore: vi.fn(async () => []) as TelegramBotDeps["readChannelAllowFromStore"],
|
|
upsertChannelPairingRequest: vi.fn(async () => ({
|
|
code: "PAIRCODE",
|
|
created: true,
|
|
})) as TelegramBotDeps["upsertChannelPairingRequest"],
|
|
enqueueSystemEvent: vi.fn() as TelegramBotDeps["enqueueSystemEvent"],
|
|
dispatchReplyWithBufferedBlockDispatcher: mediaHarnessDispatchReplyWithBufferedBlockDispatcher,
|
|
buildModelsProviderData: vi.fn(async () => ({
|
|
byProvider: new Map<string, Set<string>>(),
|
|
providers: [],
|
|
resolvedDefault: { provider: "openai", model: "gpt-4.1" },
|
|
})) as TelegramBotDeps["buildModelsProviderData"],
|
|
listSkillCommandsForAgents: vi.fn(() => []) as TelegramBotDeps["listSkillCommandsForAgents"],
|
|
wasSentByBot: vi.fn(() => false) as TelegramBotDeps["wasSentByBot"],
|
|
};
|
|
|
|
beforeEach(() => {
|
|
resetInboundDedupe();
|
|
resetSaveMediaBufferMock();
|
|
resetUndiciFetchMock();
|
|
resetFetchRemoteMediaMock();
|
|
});
|
|
|
|
vi.doMock("./bot.runtime.js", () => ({
|
|
...telegramBotRuntimeForTest,
|
|
}));
|
|
|
|
vi.mock("undici", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("undici")>();
|
|
return {
|
|
...actual,
|
|
fetch: (...args: Parameters<typeof undiciFetchSpy>) => undiciFetchSpy(...args),
|
|
};
|
|
});
|
|
|
|
export async function mockMediaRuntimeModuleForTest(
|
|
importOriginal: () => Promise<typeof import("openclaw/plugin-sdk/media-runtime")>,
|
|
) {
|
|
const actual = await importOriginal();
|
|
const mockModule = Object.create(null) as Record<string, unknown>;
|
|
Object.defineProperties(mockModule, Object.getOwnPropertyDescriptors(actual));
|
|
Object.defineProperty(mockModule, "fetchRemoteMedia", {
|
|
configurable: true,
|
|
enumerable: true,
|
|
writable: true,
|
|
value: (...args: Parameters<typeof fetchRemoteMediaSpy>) => fetchRemoteMediaSpy(...args),
|
|
});
|
|
Object.defineProperty(mockModule, "saveMediaBuffer", {
|
|
configurable: true,
|
|
enumerable: true,
|
|
writable: true,
|
|
value: (...args: Parameters<typeof saveMediaBufferSpy>) => saveMediaBufferSpy(...args),
|
|
});
|
|
return mockModule;
|
|
}
|
|
|
|
vi.mock("openclaw/plugin-sdk/media-runtime", mockMediaRuntimeModuleForTest);
|
|
|
|
vi.doMock("openclaw/plugin-sdk/config-runtime", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("openclaw/plugin-sdk/config-runtime")>();
|
|
return {
|
|
...actual,
|
|
loadConfig: telegramBotDepsForTest.loadConfig,
|
|
updateLastRoute: vi.fn(async () => undefined),
|
|
};
|
|
});
|
|
|
|
vi.doMock("openclaw/plugin-sdk/agent-runtime", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("openclaw/plugin-sdk/agent-runtime")>();
|
|
return {
|
|
...actual,
|
|
findModelInCatalog: vi.fn(() => undefined),
|
|
loadModelCatalog: vi.fn(async () => []),
|
|
modelSupportsVision: vi.fn(() => false),
|
|
resolveDefaultModelForAgent: vi.fn(() => ({
|
|
provider: "openai",
|
|
model: "gpt-test",
|
|
})),
|
|
};
|
|
});
|
|
|
|
vi.doMock("openclaw/plugin-sdk/conversation-runtime", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("openclaw/plugin-sdk/conversation-runtime")>();
|
|
return {
|
|
...actual,
|
|
readChannelAllowFromStore: telegramBotDepsForTest.readChannelAllowFromStore,
|
|
upsertChannelPairingRequest: vi.fn(async () => ({
|
|
code: "PAIRCODE",
|
|
created: true,
|
|
})),
|
|
};
|
|
});
|
|
|
|
vi.doMock("openclaw/plugin-sdk/reply-runtime", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("openclaw/plugin-sdk/reply-runtime")>();
|
|
return {
|
|
...actual,
|
|
getReplyFromConfig: mediaHarnessReplySpy,
|
|
__replySpy: mediaHarnessReplySpy,
|
|
};
|
|
});
|