test(e2e): add login and impersonation test coverage (GRO-77) #101

Merged
lint-roller-qa[bot] merged 4 commits from feat/e2e-login-impersonation-gro-77 into main 2026-03-22 15:43:01 +00:00
3 changed files with 24 additions and 20 deletions
Showing only changes of commit 355f11fdaa - Show all commits
+17 -18
View File
@@ -19,48 +19,48 @@ const MOCK_SESSION = {
test.describe("ImpersonationBanner", () => {
test.beforeEach(async ({ page }) => {
await page.route("**/api/impersonation/session", (route) =>
await page.route("**/api/impersonation/sessions/session-1", (route) =>
route.fulfill({ json: MOCK_SESSION })
);
await page.route("**/api/impersonation/session/end", (route) =>
await page.route("**/api/impersonation/sessions/session-1/end", (route) =>
route.fulfill({ json: { status: "ended" } })
);
await page.route("**/api/impersonation/session/extend", (route) =>
await page.route("**/api/impersonation/sessions/session-1/extend", (route) =>
route.fulfill({ json: { ...MOCK_SESSION, expiresAt: new Date(Date.now() + 60 * 60 * 1000).toISOString() } })
);
await page.route("**/api/impersonation/audit/**", (route) =>
await page.route("**/api/impersonation/sessions/session-1/audit-log", (route) =>
route.fulfill({ json: { logs: [] } })
);
});
test("banner displays when session is active", async ({ page }) => {
await page.goto("/");
await expect(page.locator(".bg-amber-500")).toBeVisible();
await page.goto("/?sessionId=session-1");
await expect(page.locator("[data-testid=\"impersonation-banner\"]")).toBeVisible();
await expect(page.getByText("STAFF VIEW")).toBeVisible();
});
test("banner shows reason when session has reason", async ({ page }) => {
await page.goto("/");
await page.goto("/?sessionId=session-1");
await expect(page.getByText(/Reason: Testing customer booking flow/)).toBeVisible();
});
test("banner shows started time", async ({ page }) => {
await page.goto("/");
await expect(page.getByText(/Started \d{1,2}:\d{2}/)).toBeVisible();
await page.goto("/?sessionId=session-1");
await expect(page.getByText(/Started/)).toBeVisible();
});
test("End Session button is visible", async ({ page }) => {
await page.goto("/");
await page.goto("/?sessionId=session-1");
await expect(page.getByRole("button", { name: /End Session/ })).toBeVisible();
});
test("Audit button is visible", async ({ page }) => {
await page.goto("/");
await page.goto("/?sessionId=session-1");
await expect(page.getByRole("button", { name: /Audit/ })).toBeVisible();
});
test("clicking End Session calls API and redirects", async ({ page }) => {
await page.goto("/");
await page.goto("/?sessionId=session-1");
await page.getByRole("button", { name: /End Session/ }).click();
await expect(page.getByText("STAFF VIEW")).not.toBeVisible();
});
@@ -70,17 +70,16 @@ test.describe("ImpersonationBanner", () => {
...MOCK_SESSION,
expiresAt: new Date(Date.now() + 3 * 60 * 1000).toISOString(),
};
await page.route("**/api/impersonation/session", (route) =>
await page.route("**/api/impersonation/sessions/session-1", (route) =>
route.fulfill({ json: lowTimeSession })
);
await page.goto("/");
await page.waitForTimeout(1100);
await page.goto("/?sessionId=session-1");
await expect(page.getByRole("button", { name: /Extend/ })).toBeVisible();
});
test("URL is cleaned when session ends", async ({ page }) => {
await page.goto("/?impersonation=session-1");
await page.goto("/?sessionId=session-1");
await page.getByRole("button", { name: /End Session/ }).click();
await expect(page).not.toHaveURL(/impersonation=session-1/);
await expect(page).not.toHaveURL(/sessionId=session-1/);
});
});
});
+6 -1
View File
@@ -14,6 +14,11 @@ test.describe("DevLoginSelector", () => {
});
test("shows loading state while fetching users", async ({ page }) => {
await page.route("**/api/dev/users", async (route) => {
await page.waitForResponse((res) => res.url().includes("/api/dev/users"));
await new Promise((r) => setTimeout(r, 100));
await route.fulfill({ json: { staff: [], clients: [] } });
});
await page.goto("/login");
await expect(page.getByText("Loading users...")).toBeVisible();
});
@@ -66,4 +71,4 @@ test.describe("DevLoginSelector", () => {
await expect(page.getByText("Staff")).toBeVisible();
await expect(page.getByText("Clients")).toBeVisible();
});
});
});
+1 -1
View File
@@ -35,7 +35,7 @@ export function ImpersonationBanner({ session, isExtended, onEnd, onExtend, onSh
}, [session.expiresAt, onEnd]);
return (
<div className="sticky top-0 z-40 bg-amber-500 text-amber-950 px-4 py-2.5 flex flex-wrap items-center gap-x-4 gap-y-1 text-sm font-medium shadow-md">
<div data-testid="impersonation-banner" className="sticky top-0 z-40 bg-amber-500 text-amber-950 px-4 py-2.5 flex flex-wrap items-center gap-x-4 gap-y-1 text-sm font-medium shadow-md">
<span className="flex items-center gap-1.5">
<Eye size={16} />
STAFF VIEW