Files
medassist-ng/frontend/e2e/schedule-data.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

240 lines
7.8 KiB
TypeScript

import {
authFile,
createMedicationViaAPI,
deleteAllMedicationsViaAPI,
deleteMedicationViaAPI,
expect,
navigateTo,
type TestMedication,
test,
} from "./fixtures";
/**
* Schedule & Dose Tracking E2E Tests
*
* Creates medications via API, then verifies the schedule timeline:
* day blocks, dose items, dose tracking, collapse/expand, and toggles.
*/
test.describe("Schedule with medications", () => {
test.use({ storageState: authFile });
test.describe.configure({ timeout: 60000 });
const MED_DAILY = "SchedData DailyMed";
const MED_PAST = "SchedData PastMed";
const MED_WEEKLY = "SchedData WeeklyMed";
const todayMorning = (() => {
const d = new Date();
d.setHours(8, 0, 0, 0);
const pad = (n: number) => n.toString().padStart(2, "0");
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
})();
const threeDaysAgo = (() => {
const d = new Date();
d.setDate(d.getDate() - 3);
d.setHours(9, 0, 0, 0);
const pad = (n: number) => n.toString().padStart(2, "0");
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
})();
const createdMeds: TestMedication[] = [];
test.beforeAll(async () => {
// Clean up any leftover medications from previous test runs
await deleteAllMedicationsViaAPI();
createdMeds.push(
await createMedicationViaAPI({
name: MED_DAILY,
packageType: "blister",
packCount: 1,
blistersPerPack: 2,
pillsPerBlister: 14,
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
})
);
createdMeds.push(
await createMedicationViaAPI({
name: MED_PAST,
packageType: "blister",
packCount: 1,
blistersPerPack: 1,
pillsPerBlister: 30,
intakes: [{ usage: 1, every: 1, start: threeDaysAgo, intakeRemindersEnabled: false }],
})
);
createdMeds.push(
await createMedicationViaAPI({
name: MED_WEEKLY,
packageType: "bottle",
totalPills: 52,
intakes: [{ usage: 1, every: 7, start: todayMorning, intakeRemindersEnabled: false }],
})
);
});
test.afterAll(async () => {
await deleteAllMedicationsViaAPI();
});
test("should show today block with medication names", async ({ page }) => {
await navigateTo(page, "/dashboard");
const todayBlock = page.locator(".day-block.today");
await expect(todayBlock).toBeVisible({ timeout: 15000 });
// Today should have time rows with our medication names
const timeRows = todayBlock.locator(".time-row");
expect(await timeRows.count()).toBeGreaterThanOrEqual(1);
// At least the daily and past medications should show today
await expect(todayBlock.getByText(MED_DAILY)).toBeVisible();
await expect(todayBlock.getByText(MED_PAST)).toBeVisible();
});
test("should show dose items with time info", async ({ page }) => {
await navigateTo(page, "/dashboard");
await page.waitForLoadState("networkidle");
const todayBlock = page.locator(".day-block.today");
await expect(todayBlock).toBeVisible({ timeout: 15000 });
const doseItems = todayBlock.locator(".dose-item");
expect(await doseItems.count()).toBeGreaterThanOrEqual(1);
// Each dose should have a time label
await expect(doseItems.first().locator(".dose-time")).toBeVisible();
});
test("should show day date in day header", async ({ page }) => {
await navigateTo(page, "/dashboard");
const todayBlock = page.locator(".day-block.today");
await expect(todayBlock).toBeVisible({ timeout: 10000 });
const dayDate = todayBlock.locator(".day-date");
await expect(dayDate).toBeVisible();
expect(await dayDate.textContent()).toBeTruthy();
});
test("should collapse and expand a past day block", async ({ page }) => {
await navigateTo(page, "/dashboard");
// First show past days
const pastToggle = page.locator(".past-days-toggle");
await expect(pastToggle).toBeVisible({ timeout: 10000 });
await pastToggle.click();
const pastBlock = page.locator(".day-block.past").first();
await expect(pastBlock).toBeVisible({ timeout: 5000 });
// Click the divider to toggle collapse
const dayDivider = pastBlock.locator(".day-divider");
await dayDivider.click();
// Past blocks start expanded after toggle, so clicking should collapse
// Check that the block has or doesn't have the collapsed class
const classAfterClick = await pastBlock.getAttribute("class");
expect(classAfterClick).toBeTruthy();
});
test("should show past days toggle", async ({ page }) => {
await navigateTo(page, "/dashboard");
// A medication starting 3 days ago should create past day entries
const pastToggle = page.locator(".past-days-toggle");
await expect(pastToggle).toBeVisible({ timeout: 10000 });
});
test("should expand past days when toggle is clicked", async ({ page }) => {
await navigateTo(page, "/dashboard");
const pastToggle = page.locator(".past-days-toggle");
await expect(pastToggle).toBeVisible({ timeout: 10000 });
await pastToggle.click();
const pastBlocks = page.locator(".day-block.past");
await expect(pastBlocks.first()).toBeVisible({ timeout: 5000 });
expect(await pastBlocks.count()).toBeGreaterThanOrEqual(1);
});
test("should show future day blocks", async ({ page }) => {
await navigateTo(page, "/dashboard");
// Wait for timeline to fully load
await page.waitForLoadState("networkidle");
const dayBlocks = page.locator(".day-block:not(.past)");
await expect(dayBlocks.first()).toBeVisible({ timeout: 10000 });
expect(await dayBlocks.count()).toBeGreaterThanOrEqual(1);
});
test("should change schedule range", async ({ page }) => {
await navigateTo(page, "/dashboard");
const daysSelect = page.locator("select.schedule-days-select");
await expect(daysSelect).toBeVisible();
await daysSelect.selectOption("30");
await page.waitForTimeout(500);
const count30 = await page.locator(".day-block").count();
await daysSelect.selectOption("90");
await page.waitForTimeout(500);
const count90 = await page.locator(".day-block").count();
expect(count90).toBeGreaterThanOrEqual(count30);
});
test("should mark dose as taken and show undo", async ({ page }) => {
await navigateTo(page, "/dashboard");
await page.waitForLoadState("networkidle");
const todayBlock = page.locator(".day-block.today");
await expect(todayBlock).toBeVisible({ timeout: 15000 });
const takeBtn = todayBlock.locator("button.dose-btn.take:not([disabled])").first();
if (!(await takeBtn.isVisible().catch(() => false))) return;
await takeBtn.click();
await page.waitForLoadState("networkidle");
await expect(todayBlock.locator("button.dose-btn.undo").first()).toBeVisible({ timeout: 10000 });
});
test("should undo taken doses", async ({ page }) => {
await navigateTo(page, "/dashboard");
await page.waitForLoadState("networkidle");
const todayBlock = page.locator(".day-block.today");
await expect(todayBlock).toBeVisible({ timeout: 15000 });
// Undo any previously taken doses
const undoButtons = todayBlock.locator("button.dose-btn.undo");
const undoCount = await undoButtons.count();
for (let i = 0; i < undoCount; i++) {
const btn = todayBlock.locator("button.dose-btn.undo").first();
if (await btn.isVisible().catch(() => false)) {
await btn.click();
await page.waitForTimeout(300);
}
}
if (undoCount > 0) {
const takeButtons = todayBlock.locator("button.dose-btn.take:not([disabled])");
expect(await takeButtons.count()).toBeGreaterThanOrEqual(1);
}
});
test("should show medication names in timeline rows", async ({ page }) => {
await navigateTo(page, "/dashboard");
await page.waitForLoadState("networkidle");
const todayBlock = page.locator(".day-block.today");
await expect(todayBlock).toBeVisible({ timeout: 15000 });
const medNames = todayBlock.locator(".med-name");
expect(await medNames.count()).toBeGreaterThanOrEqual(1);
});
});