7f2ef09df5
* test: expand e2e app shell coverage and stabilize flaky scenarios * fix(e2e): stabilize dashboard flow and frontend ci gates
256 lines
9.6 KiB
TypeScript
256 lines
9.6 KiB
TypeScript
import type { Page } from "@playwright/test";
|
||
import {
|
||
authFile,
|
||
createMedicationViaAPI,
|
||
deleteAllMedicationsViaAPI,
|
||
expect,
|
||
navigateTo,
|
||
type TestMedication,
|
||
test,
|
||
} from "./fixtures";
|
||
|
||
/**
|
||
* Helper: navigate to planner, wait for page to be ready, click Calculate,
|
||
* and wait for results to appear.
|
||
*/
|
||
async function calculatePlanner(page: Page): Promise<void> {
|
||
await page.waitForLoadState("networkidle");
|
||
await page.locator('form.planner button[type="submit"]').click();
|
||
// Wait for the results table to appear (more reliable than waitForResponse
|
||
// since 429 responses would satisfy waitForResponse but not populate results)
|
||
await expect(page.locator(".table")).toBeVisible({ timeout: 15000 });
|
||
}
|
||
|
||
/**
|
||
* Planner with Medication Data E2E Tests
|
||
*
|
||
* Creates medications via API, then verifies the demand calculator
|
||
* produces correct results with status chips and usage data.
|
||
*/
|
||
test.describe("Planner with medications", () => {
|
||
test.use({ storageState: authFile });
|
||
test.describe.configure({ timeout: 60000 });
|
||
|
||
const MED_HIGH = "PlanData HighStock";
|
||
const MED_LOW = "PlanData LowStock";
|
||
|
||
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 createdMeds: TestMedication[] = [];
|
||
|
||
test.beforeAll(async () => {
|
||
// Clean up any leftover medications from previous test runs
|
||
await deleteAllMedicationsViaAPI();
|
||
// Medication with plenty of stock (60 pills)
|
||
createdMeds.push(
|
||
await createMedicationViaAPI({
|
||
name: MED_HIGH,
|
||
packageType: "blister",
|
||
packCount: 2,
|
||
blistersPerPack: 3,
|
||
pillsPerBlister: 10,
|
||
looseTablets: 0,
|
||
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
|
||
})
|
||
);
|
||
// Medication with very low stock (3 pills)
|
||
createdMeds.push(
|
||
await createMedicationViaAPI({
|
||
name: MED_LOW,
|
||
packageType: "blister",
|
||
packCount: 1,
|
||
blistersPerPack: 1,
|
||
pillsPerBlister: 3,
|
||
looseTablets: 0,
|
||
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
|
||
})
|
||
);
|
||
});
|
||
|
||
test.afterAll(async () => {
|
||
await deleteAllMedicationsViaAPI();
|
||
});
|
||
|
||
test("should show results table after calculating", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
});
|
||
|
||
test("should show medication names in results", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
|
||
await expect(resultsTable.getByText(MED_HIGH)).toBeVisible();
|
||
await expect(resultsTable.getByText(MED_LOW)).toBeVisible();
|
||
});
|
||
|
||
test("should show status chips in results", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
|
||
const statusChips = resultsTable.locator(".status-chip");
|
||
expect(await statusChips.count()).toBeGreaterThanOrEqual(2);
|
||
});
|
||
|
||
test("should show correct usage values in results rows", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
|
||
const rows = resultsTable.locator(".table-row");
|
||
expect(await rows.count()).toBeGreaterThanOrEqual(2);
|
||
|
||
// Each medication has usage=1, every=1 → plannerUsage should reflect the period
|
||
// Verify the usage column contains a numeric <strong> value and "pill(s)"
|
||
for (const row of await rows.all()) {
|
||
const usageCell = row.locator("[data-label]").nth(1); // Usage is 2nd column
|
||
const usageStrong = usageCell.locator("strong");
|
||
await expect(usageStrong).toBeVisible();
|
||
const usageText = await usageStrong.textContent();
|
||
expect(Number(usageText)).toBeGreaterThan(0);
|
||
}
|
||
});
|
||
|
||
test("should show danger status for low-stock medication over 90 days", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
|
||
// Set the "until" date to 90 days from now
|
||
const dateInputs = page.locator('form.planner input[type="datetime-local"]');
|
||
const untilInput = dateInputs.last();
|
||
const fromValue = await dateInputs.first().inputValue();
|
||
const fromDate = new Date(fromValue);
|
||
const untilDate = new Date(fromDate.getTime() + 90 * 24 * 60 * 60 * 1000);
|
||
const pad = (n: number) => n.toString().padStart(2, "0");
|
||
const untilValue = `${untilDate.getFullYear()}-${pad(untilDate.getMonth() + 1)}-${pad(untilDate.getDate())}T${pad(untilDate.getHours())}:${pad(untilDate.getMinutes())}`;
|
||
await untilInput.fill(untilValue);
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
|
||
// Low-stock med (3 pills, usage 1/day, 90 days) should have danger status
|
||
const dangerChips = resultsTable.locator(".status-chip.danger");
|
||
expect(await dangerChips.count()).toBeGreaterThanOrEqual(1);
|
||
|
||
// Find the low-stock med row and verify its usage value ~90 pills
|
||
const lowStockRow = resultsTable.locator(".table-row", { hasText: MED_LOW });
|
||
await expect(lowStockRow).toBeVisible();
|
||
const lowUsage = await lowStockRow.locator("[data-label] strong").first().textContent();
|
||
expect(Number(lowUsage)).toBeGreaterThanOrEqual(85); // ~90 pills needed
|
||
expect(Number(lowUsage)).toBeLessThanOrEqual(95);
|
||
});
|
||
|
||
test("should show Enough status for well-stocked medication over 7 days", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
|
||
// Set a short date range: 7 days
|
||
const dateInputs = page.locator('form.planner input[type="datetime-local"]');
|
||
const untilInput = dateInputs.last();
|
||
const fromValue = await dateInputs.first().inputValue();
|
||
const fromDate = new Date(fromValue);
|
||
const untilDate = new Date(fromDate.getTime() + 7 * 24 * 60 * 60 * 1000);
|
||
const pad = (n: number) => n.toString().padStart(2, "0");
|
||
const untilValue = `${untilDate.getFullYear()}-${pad(untilDate.getMonth() + 1)}-${pad(untilDate.getDate())}T${pad(untilDate.getHours())}:${pad(untilDate.getMinutes())}`;
|
||
await untilInput.fill(untilValue);
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
|
||
// High-stock med (60 pills, usage 1/day, 7 days → needs ~7, has 60) should be "Enough"
|
||
const highStockRow = resultsTable.locator(".table-row", { hasText: MED_HIGH });
|
||
await expect(highStockRow).toBeVisible();
|
||
const highStatus = highStockRow.locator(".status-chip.success");
|
||
await expect(highStatus).toBeVisible();
|
||
|
||
// Verify usage is ~7 pills for the 7-day range
|
||
const highUsage = await highStockRow.locator("[data-label] strong").first().textContent();
|
||
expect(Number(highUsage)).toBeGreaterThanOrEqual(5);
|
||
expect(Number(highUsage)).toBeLessThanOrEqual(10);
|
||
});
|
||
|
||
test("should show table header with correct columns", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
|
||
const tableHead = resultsTable.locator(".table-head");
|
||
await expect(tableHead).toBeVisible();
|
||
await expect(tableHead.getByText(/Medication/i)).toBeVisible();
|
||
await expect(tableHead.getByText(/Usage/i)).toBeVisible();
|
||
await expect(tableHead.getByText(/Status/i)).toBeVisible();
|
||
});
|
||
|
||
test("should display available stock for each medication", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
|
||
// High-stock med should show a blister + loose-pill stock breakdown
|
||
const highStockRow = resultsTable.locator(".table-row", { hasText: MED_HIGH });
|
||
await expect(highStockRow).toBeVisible();
|
||
const highStockText = await highStockRow.textContent();
|
||
expect(highStockText).toMatch(/\d+\s*(blisters|Blister)/i);
|
||
expect(highStockText).toMatch(/\d+\s*(pill|pills|Tablette|Tabletten)/i);
|
||
|
||
// Low-stock med: 1 pack × 1 blister × 3 pills = 3 pills = 0 full blisters + 3 loose
|
||
const lowStockRow = resultsTable.locator(".table-row", { hasText: MED_LOW });
|
||
await expect(lowStockRow).toBeVisible();
|
||
const lowStockText = await lowStockRow.textContent();
|
||
// The exact loose-pill amount can vary due already-taken doses; ensure stock details are still rendered.
|
||
expect(lowStockText).toMatch(/\d+\s*×\s*\d+/i);
|
||
expect(lowStockText).toMatch(/\d+\s*(pill|pills|Tablette|Tabletten)/i);
|
||
});
|
||
|
||
test("should reset form and clear results", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
|
||
// Click Reset
|
||
await page.locator("form.planner button.ghost").click();
|
||
|
||
// Results should be cleared
|
||
await expect(resultsTable).not.toBeVisible({ timeout: 5000 });
|
||
});
|
||
|
||
test("should make results rows clickable for medication detail", async ({ page }) => {
|
||
await navigateTo(page, "/planner");
|
||
await calculatePlanner(page);
|
||
|
||
const resultsTable = page.locator(".table");
|
||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||
|
||
// Click on a results row
|
||
await resultsTable.locator(".table-row").first().click();
|
||
|
||
const modal = page.locator(".modal-overlay");
|
||
await expect(modal).toBeVisible({ timeout: 5000 });
|
||
|
||
await page.locator("button.modal-close").click();
|
||
await expect(modal).not.toBeVisible();
|
||
});
|
||
});
|