Files
medassist-ng/frontend/e2e/auth.spec.ts
T
Daniel Volz 98939877db feat: expand Playwright E2E coverage (#155)
* feat: comprehensive Playwright E2E test rewrite

Rewrite all E2E tests with correct CSS selectors, add new spec files,
and implement robust auth handling to work within backend rate limits.

Changes:
- Rewrite fixtures/index.ts with JWT-based /auth/me mock to avoid
  10 req/min rate limit on /auth/me during test runs
- Rewrite auth.setup.ts with offline JWT validity check to reuse
  existing auth state across runs (saves login rate-limit budget)
- Rewrite auth.spec.ts (6 tests) - login page, fields, submit,
  redirect guard, invalid credentials, login/register toggle
- Rewrite dashboard.spec.ts (8 tests) - header, nav tabs,
  navigation, overview/schedules sections, days selector, redirect
- Rewrite medications.spec.ts (8 tests) - form fields, stock
  inventory, package type toggle, intake schedule, save/cancel,
  unsaved changes guard
- Rewrite settings.spec.ts (12 tests) - language, notification
  matrix, thresholds, calculation mode, toggle switch, export/import,
  user menu navigation
- Create planner.spec.ts (9 tests) - form, date inputs, calculate,
  reset, checkbox, submit, tab state, eyebrow heading
- Create schedule.spec.ts (12 tests) - timeline, days selector,
  past/future toggles, day blocks, today highlight, collapse/expand,
  overview table, share button
- Update playwright.config.ts: remove mobile projects, enable
  webServer section for CI
- Add .github/workflows/e2e.yml CI workflow for Playwright tests

Total: 57 E2E tests across 6 spec files, all passing consistently
across 5+ consecutive runs without backend restart.

Closes #154

* feat: add comprehensive E2E data tests with medication CRUD, dashboard, planner, schedule

Add 48 new Playwright E2E tests covering real medication data scenarios:
- medication-crud: 14 tests for create/edit/delete/list via UI form
- dashboard-data: 13 tests for overview table, timeline, dose tracking
- planner-data: 9 tests for demand calculator with results/status chips
- schedule-data: 11 tests for timeline, collapse/expand, dose mark/undo

Infrastructure improvements:
- Add API helpers (createMedicationViaAPI, deleteMedicationViaAPI,
  deleteAllMedicationsViaAPI) with retry logic for rate-limit resilience
- Configure chromium-data project for serial execution with retry:1
- Add /auth/me mock to avoid rate-limit exhaustion on auth endpoint
- Increase navigateTo reliability with networkidle waits
- Increase auth token validity threshold from 2 to 10 minutes
- Make backend rate limit configurable via RATE_LIMIT_MAX env var
- Set RATE_LIMIT_MAX=300 in dev docker-compose for E2E test support

Total suite: 57 empty-state + 48 data tests = 105 tests (chromium)

* test: add E2E tests for medication editing, stock status, and share schedule

- medication-edit.spec.ts: 10 tests covering generic name, notes,
  taken-by add/remove, expiry date, refill, intake schedule editing,
  adding intake rows, reminder toggle, and package type changes
- stock-status.spec.ts: 12 tests verifying dashboard shows correct
  status chips (High/Normal/Warning/Danger) for different stock levels,
  overview table, reorder card, detail modal, and planner integration
- share-schedule.spec.ts: 10 tests for taken-by badges, share button,
  share dialog, link generation, shared schedule page navigation,
  dose tracking on shared page, and notes display
- fixtures/index.ts: add createShareTokenViaAPI, updateSettingsViaAPI
  helpers; expand createMedicationViaAPI with takenBy, notes, expiryDate
- playwright.config.ts: update testMatch/testIgnore for new test files
- docker-compose.dev.yml: increase RATE_LIMIT_MAX to 1000 for E2E tests

* docs: refine release-manager instructions for CLI safety and commit-linked release notes

* fix: resolve PR155 CI failures for frontend lint and e2e proxy

* fix: stabilize auth-related e2e checks in CI
2026-02-12 20:06:11 +01:00

114 lines
3.9 KiB
TypeScript

import { expect, type Page, test } from "@playwright/test";
async function isAuthEnabled(page: Page): Promise<boolean> {
try {
const response = await page.request.get("/api/auth/state");
if (!response.ok()) return true;
const state = await response.json();
return state?.authEnabled !== false;
} catch {
return true;
}
}
/**
* Authentication E2E Tests
*
* Tests the login/register UI when not authenticated.
* Uses empty storage state to simulate unauthenticated access.
*
* NOTE: This file intentionally imports `test` from @playwright/test
* (not from fixtures) because auth tests use empty storageState and
* must NOT have the auth-me caching interceptor.
*/
test.describe("Authentication", () => {
test.use({ storageState: { cookies: [], origins: [] } });
test("should show login page for unauthenticated users", async ({ page }) => {
test.skip(!(await isAuthEnabled(page)), "Auth is disabled in this environment");
await page.goto("/");
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
// Should have the app title
await expect(page.locator(".auth-title")).toContainText("MedAssist-ng");
});
test("should have username and password fields", async ({ page }) => {
test.skip(!(await isAuthEnabled(page)), "Auth is disabled in this environment");
await page.goto("/");
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
const usernameField = page.locator("#username");
const passwordField = page.locator("#password");
await expect(usernameField).toBeVisible();
await expect(usernameField).toBeEnabled();
await expect(passwordField).toBeVisible();
await expect(passwordField).toBeEnabled();
});
test("should have a submit button", async ({ page }) => {
test.skip(!(await isAuthEnabled(page)), "Auth is disabled in this environment");
await page.goto("/");
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
const submitButton = page.locator('button.auth-submit[type="submit"]');
await expect(submitButton).toBeVisible();
await expect(submitButton).toBeEnabled();
});
test("should not navigate to dashboard without credentials", async ({ page }) => {
test.skip(!(await isAuthEnabled(page)), "Auth is disabled in this environment");
await page.goto("/dashboard");
// Should NOT show the app header (redirected to login)
await expect(page.locator("header.hero")).not.toBeVisible({ timeout: 10000 });
// Should show auth form instead
await expect(page.locator(".auth-container")).toBeVisible();
});
test("should show error for invalid credentials", async ({ page }) => {
test.skip(!(await isAuthEnabled(page)), "Auth is disabled in this environment");
await page.goto("/");
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
// Fill in invalid credentials
await page.locator("#username").fill("nonexistent-user");
await page.locator("#password").fill("wrongpassword");
await page.locator('button.auth-submit[type="submit"]').click();
// Should show an error message
await expect(page.locator(".auth-error")).toBeVisible({ timeout: 5000 });
});
test("should toggle between login and register forms", async ({ page }) => {
test.skip(!(await isAuthEnabled(page)), "Auth is disabled in this environment");
await page.goto("/");
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
const toggleButton = page.locator("button.auth-link-btn");
test.skip(
!(await toggleButton.isVisible().catch(() => false)),
"Registration toggle is unavailable in this environment"
);
// Check current subtitle text
const subtitle = page.locator(".auth-subtitle");
const initialText = await subtitle.textContent();
// Click the toggle link (Create account / Already have an account)
await toggleButton.click();
// Subtitle should change
const newText = await subtitle.textContent();
expect(newText).not.toBe(initialText);
});
});