456 lines
13 KiB
TypeScript
456 lines
13 KiB
TypeScript
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
|
|
let currentPage: Record<string, unknown> | null = null;
|
|
let currentRefLocator: Record<string, unknown> | null = null;
|
|
let pageState: {
|
|
console: unknown[];
|
|
armIdUpload: number;
|
|
armIdDialog: number;
|
|
armIdDownload: number;
|
|
};
|
|
|
|
const sessionMocks = vi.hoisted(() => ({
|
|
getPageForTargetId: vi.fn(async () => {
|
|
if (!currentPage) throw new Error("missing page");
|
|
return currentPage;
|
|
}),
|
|
ensurePageState: vi.fn(() => pageState),
|
|
refLocator: vi.fn(() => {
|
|
if (!currentRefLocator) throw new Error("missing locator");
|
|
return currentRefLocator;
|
|
}),
|
|
}));
|
|
|
|
vi.mock("./pw-session.js", () => sessionMocks);
|
|
|
|
async function importModule() {
|
|
return await import("./pw-tools-core.js");
|
|
}
|
|
|
|
describe("pw-tools-core", () => {
|
|
beforeEach(() => {
|
|
currentPage = null;
|
|
currentRefLocator = null;
|
|
pageState = {
|
|
console: [],
|
|
armIdUpload: 0,
|
|
armIdDialog: 0,
|
|
armIdDownload: 0,
|
|
};
|
|
for (const fn of Object.values(sessionMocks)) fn.mockClear();
|
|
});
|
|
|
|
it("screenshots an element selector", async () => {
|
|
const elementScreenshot = vi.fn(async () => Buffer.from("E"));
|
|
currentPage = {
|
|
locator: vi.fn(() => ({
|
|
first: () => ({ screenshot: elementScreenshot }),
|
|
})),
|
|
screenshot: vi.fn(async () => Buffer.from("P")),
|
|
};
|
|
|
|
const mod = await importModule();
|
|
const res = await mod.takeScreenshotViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
element: "#main",
|
|
type: "png",
|
|
});
|
|
|
|
expect(res.buffer.toString()).toBe("E");
|
|
expect(sessionMocks.getPageForTargetId).toHaveBeenCalled();
|
|
expect(
|
|
currentPage.locator as ReturnType<typeof vi.fn>,
|
|
).toHaveBeenCalledWith("#main");
|
|
expect(elementScreenshot).toHaveBeenCalledWith({ type: "png" });
|
|
});
|
|
|
|
it("screenshots a ref locator", async () => {
|
|
const refScreenshot = vi.fn(async () => Buffer.from("R"));
|
|
currentRefLocator = { screenshot: refScreenshot };
|
|
currentPage = {
|
|
locator: vi.fn(),
|
|
screenshot: vi.fn(async () => Buffer.from("P")),
|
|
};
|
|
|
|
const mod = await importModule();
|
|
const res = await mod.takeScreenshotViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
ref: "76",
|
|
type: "jpeg",
|
|
});
|
|
|
|
expect(res.buffer.toString()).toBe("R");
|
|
expect(sessionMocks.refLocator).toHaveBeenCalledWith(currentPage, "76");
|
|
expect(refScreenshot).toHaveBeenCalledWith({ type: "jpeg" });
|
|
});
|
|
|
|
it("rejects fullPage for element or ref screenshots", async () => {
|
|
currentRefLocator = { screenshot: vi.fn(async () => Buffer.from("R")) };
|
|
currentPage = {
|
|
locator: vi.fn(() => ({
|
|
first: () => ({ screenshot: vi.fn(async () => Buffer.from("E")) }),
|
|
})),
|
|
screenshot: vi.fn(async () => Buffer.from("P")),
|
|
};
|
|
|
|
const mod = await importModule();
|
|
|
|
await expect(
|
|
mod.takeScreenshotViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
element: "#x",
|
|
fullPage: true,
|
|
}),
|
|
).rejects.toThrow(/fullPage is not supported/i);
|
|
|
|
await expect(
|
|
mod.takeScreenshotViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
ref: "1",
|
|
fullPage: true,
|
|
}),
|
|
).rejects.toThrow(/fullPage is not supported/i);
|
|
});
|
|
|
|
it("arms the next file chooser and sets files (default timeout)", async () => {
|
|
const fileChooser = { setFiles: vi.fn(async () => {}) };
|
|
const waitForEvent = vi.fn(
|
|
async (_event: string, _opts: unknown) => fileChooser,
|
|
);
|
|
currentPage = {
|
|
waitForEvent,
|
|
keyboard: { press: vi.fn(async () => {}) },
|
|
};
|
|
|
|
const mod = await importModule();
|
|
await mod.armFileUploadViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
paths: ["/tmp/a.txt"],
|
|
});
|
|
|
|
// waitForEvent is awaited immediately; handler continues async.
|
|
await Promise.resolve();
|
|
|
|
expect(waitForEvent).toHaveBeenCalledWith("filechooser", {
|
|
timeout: 120_000,
|
|
});
|
|
expect(fileChooser.setFiles).toHaveBeenCalledWith(["/tmp/a.txt"]);
|
|
});
|
|
|
|
it("arms the next file chooser and escapes if no paths provided", async () => {
|
|
const fileChooser = { setFiles: vi.fn(async () => {}) };
|
|
const press = vi.fn(async () => {});
|
|
const waitForEvent = vi.fn(async () => fileChooser);
|
|
currentPage = {
|
|
waitForEvent,
|
|
keyboard: { press },
|
|
};
|
|
|
|
const mod = await importModule();
|
|
await mod.armFileUploadViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
paths: [],
|
|
});
|
|
await Promise.resolve();
|
|
|
|
expect(fileChooser.setFiles).not.toHaveBeenCalled();
|
|
expect(press).toHaveBeenCalledWith("Escape");
|
|
});
|
|
|
|
it("last file-chooser arm wins", async () => {
|
|
let resolve1: ((value: unknown) => void) | null = null;
|
|
let resolve2: ((value: unknown) => void) | null = null;
|
|
|
|
const fc1 = { setFiles: vi.fn(async () => {}) };
|
|
const fc2 = { setFiles: vi.fn(async () => {}) };
|
|
|
|
const waitForEvent = vi
|
|
.fn()
|
|
.mockImplementationOnce(
|
|
() =>
|
|
new Promise((r) => {
|
|
resolve1 = r;
|
|
}) as Promise<unknown>,
|
|
)
|
|
.mockImplementationOnce(
|
|
() =>
|
|
new Promise((r) => {
|
|
resolve2 = r;
|
|
}) as Promise<unknown>,
|
|
);
|
|
|
|
currentPage = {
|
|
waitForEvent,
|
|
keyboard: { press: vi.fn(async () => {}) },
|
|
};
|
|
|
|
const mod = await importModule();
|
|
await mod.armFileUploadViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
paths: ["/tmp/1"],
|
|
});
|
|
await mod.armFileUploadViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
paths: ["/tmp/2"],
|
|
});
|
|
|
|
resolve1?.(fc1);
|
|
resolve2?.(fc2);
|
|
await Promise.resolve();
|
|
|
|
expect(fc1.setFiles).not.toHaveBeenCalled();
|
|
expect(fc2.setFiles).toHaveBeenCalledWith(["/tmp/2"]);
|
|
});
|
|
|
|
it("arms the next dialog and accepts/dismisses (default timeout)", async () => {
|
|
const accept = vi.fn(async () => {});
|
|
const dismiss = vi.fn(async () => {});
|
|
const dialog = { accept, dismiss };
|
|
const waitForEvent = vi.fn(async () => dialog);
|
|
currentPage = {
|
|
waitForEvent,
|
|
};
|
|
|
|
const mod = await importModule();
|
|
await mod.armDialogViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
accept: true,
|
|
promptText: "x",
|
|
});
|
|
await Promise.resolve();
|
|
|
|
expect(waitForEvent).toHaveBeenCalledWith("dialog", { timeout: 120_000 });
|
|
expect(accept).toHaveBeenCalledWith("x");
|
|
expect(dismiss).not.toHaveBeenCalled();
|
|
|
|
accept.mockClear();
|
|
dismiss.mockClear();
|
|
waitForEvent.mockClear();
|
|
|
|
await mod.armDialogViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
accept: false,
|
|
});
|
|
await Promise.resolve();
|
|
|
|
expect(waitForEvent).toHaveBeenCalledWith("dialog", { timeout: 120_000 });
|
|
expect(dismiss).toHaveBeenCalled();
|
|
expect(accept).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("waits for selector, url, load state, and function", async () => {
|
|
const waitForSelector = vi.fn(async () => {});
|
|
const waitForURL = vi.fn(async () => {});
|
|
const waitForLoadState = vi.fn(async () => {});
|
|
const waitForFunction = vi.fn(async () => {});
|
|
const waitForTimeout = vi.fn(async () => {});
|
|
|
|
currentPage = {
|
|
locator: vi.fn(() => ({
|
|
first: () => ({ waitFor: waitForSelector }),
|
|
})),
|
|
waitForURL,
|
|
waitForLoadState,
|
|
waitForFunction,
|
|
waitForTimeout,
|
|
getByText: vi.fn(() => ({ first: () => ({ waitFor: vi.fn() }) })),
|
|
};
|
|
|
|
const mod = await importModule();
|
|
await mod.waitForViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
selector: "#main",
|
|
url: "**/dash",
|
|
loadState: "networkidle",
|
|
fn: "window.ready===true",
|
|
timeoutMs: 1234,
|
|
timeMs: 50,
|
|
});
|
|
|
|
expect(waitForTimeout).toHaveBeenCalledWith(50);
|
|
expect(
|
|
currentPage.locator as ReturnType<typeof vi.fn>,
|
|
).toHaveBeenCalledWith("#main");
|
|
expect(waitForSelector).toHaveBeenCalledWith({
|
|
state: "visible",
|
|
timeout: 1234,
|
|
});
|
|
expect(waitForURL).toHaveBeenCalledWith("**/dash", { timeout: 1234 });
|
|
expect(waitForLoadState).toHaveBeenCalledWith("networkidle", {
|
|
timeout: 1234,
|
|
});
|
|
expect(waitForFunction).toHaveBeenCalledWith("window.ready===true", {
|
|
timeout: 1234,
|
|
});
|
|
});
|
|
|
|
it("waits for the next download and saves it", async () => {
|
|
let downloadHandler: ((download: unknown) => void) | undefined;
|
|
const on = vi.fn((event: string, handler: (download: unknown) => void) => {
|
|
if (event === "download") downloadHandler = handler;
|
|
});
|
|
const off = vi.fn();
|
|
|
|
const saveAs = vi.fn(async () => {});
|
|
const download = {
|
|
url: () => "https://example.com/file.bin",
|
|
suggestedFilename: () => "file.bin",
|
|
saveAs,
|
|
};
|
|
|
|
currentPage = { on, off };
|
|
|
|
const mod = await importModule();
|
|
const p = mod.waitForDownloadViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
path: "/tmp/file.bin",
|
|
timeoutMs: 1000,
|
|
});
|
|
|
|
await Promise.resolve();
|
|
expect(downloadHandler).toBeDefined();
|
|
downloadHandler?.(download);
|
|
|
|
const res = await p;
|
|
expect(saveAs).toHaveBeenCalledWith("/tmp/file.bin");
|
|
expect(res.path).toBe("/tmp/file.bin");
|
|
});
|
|
|
|
it("clicks a ref and saves the resulting download", async () => {
|
|
let downloadHandler: ((download: unknown) => void) | undefined;
|
|
const on = vi.fn((event: string, handler: (download: unknown) => void) => {
|
|
if (event === "download") downloadHandler = handler;
|
|
});
|
|
const off = vi.fn();
|
|
|
|
const click = vi.fn(async () => {});
|
|
currentRefLocator = { click };
|
|
|
|
const saveAs = vi.fn(async () => {});
|
|
const download = {
|
|
url: () => "https://example.com/report.pdf",
|
|
suggestedFilename: () => "report.pdf",
|
|
saveAs,
|
|
};
|
|
|
|
currentPage = { on, off };
|
|
|
|
const mod = await importModule();
|
|
const p = mod.downloadViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
ref: "e12",
|
|
path: "/tmp/report.pdf",
|
|
timeoutMs: 1000,
|
|
});
|
|
|
|
await Promise.resolve();
|
|
expect(downloadHandler).toBeDefined();
|
|
expect(click).toHaveBeenCalledWith({ timeout: 1000 });
|
|
|
|
downloadHandler?.(download);
|
|
|
|
const res = await p;
|
|
expect(saveAs).toHaveBeenCalledWith("/tmp/report.pdf");
|
|
expect(res.path).toBe("/tmp/report.pdf");
|
|
});
|
|
|
|
it("waits for a matching response and returns its body", async () => {
|
|
let responseHandler: ((resp: unknown) => void) | undefined;
|
|
const on = vi.fn((event: string, handler: (resp: unknown) => void) => {
|
|
if (event === "response") responseHandler = handler;
|
|
});
|
|
const off = vi.fn();
|
|
currentPage = { on, off };
|
|
|
|
const resp = {
|
|
url: () => "https://example.com/api/data",
|
|
status: () => 200,
|
|
headers: () => ({ "content-type": "application/json" }),
|
|
text: async () => '{"ok":true,"value":123}',
|
|
};
|
|
|
|
const mod = await importModule();
|
|
const p = mod.responseBodyViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
url: "**/api/data",
|
|
timeoutMs: 1000,
|
|
maxChars: 10,
|
|
});
|
|
|
|
await Promise.resolve();
|
|
expect(responseHandler).toBeDefined();
|
|
responseHandler?.(resp);
|
|
|
|
const res = await p;
|
|
expect(res.url).toBe("https://example.com/api/data");
|
|
expect(res.status).toBe(200);
|
|
expect(res.body).toBe('{"ok":true');
|
|
expect(res.truncated).toBe(true);
|
|
});
|
|
|
|
it("rewrites strict mode violations into snapshot hints", async () => {
|
|
const click = vi.fn(async () => {
|
|
throw new Error(
|
|
'Error: strict mode violation: locator("aria-ref=1") resolved to 2 elements',
|
|
);
|
|
});
|
|
currentRefLocator = { click };
|
|
currentPage = {};
|
|
|
|
const mod = await importModule();
|
|
await expect(
|
|
mod.clickViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
ref: "1",
|
|
}),
|
|
).rejects.toThrow(/Run a new snapshot/i);
|
|
});
|
|
|
|
it("rewrites not-visible timeouts into snapshot hints", async () => {
|
|
const click = vi.fn(async () => {
|
|
throw new Error(
|
|
'Timeout 5000ms exceeded. waiting for locator("aria-ref=1") to be visible',
|
|
);
|
|
});
|
|
currentRefLocator = { click };
|
|
currentPage = {};
|
|
|
|
const mod = await importModule();
|
|
await expect(
|
|
mod.clickViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
ref: "1",
|
|
}),
|
|
).rejects.toThrow(/not found or not visible/i);
|
|
});
|
|
|
|
it("rewrites covered/hidden errors into interactable hints", async () => {
|
|
const click = vi.fn(async () => {
|
|
throw new Error(
|
|
"Element is not receiving pointer events because another element intercepts pointer events",
|
|
);
|
|
});
|
|
currentRefLocator = { click };
|
|
currentPage = {};
|
|
|
|
const mod = await importModule();
|
|
await expect(
|
|
mod.clickViaPlaywright({
|
|
cdpUrl: "http://127.0.0.1:18792",
|
|
targetId: "T1",
|
|
ref: "1",
|
|
}),
|
|
).rejects.toThrow(/not interactable/i);
|
|
});
|
|
});
|