193 lines
5.7 KiB
TypeScript
193 lines
5.7 KiB
TypeScript
/**
|
|
* Settings Pages E2E Tests
|
|
*
|
|
* Tests the Social, SEO, and Email settings sub-pages.
|
|
* These are form-based pages that load settings from the API and save them.
|
|
*
|
|
* The primary class of bug we're catching: API response shape mismatches
|
|
* that crash the page on load, or save mutations that silently fail.
|
|
*/
|
|
|
|
import { test, expect } from "../fixtures";
|
|
|
|
// API patterns
|
|
const SETTINGS_API_PATTERN = /\/api\/settings$/;
|
|
|
|
test.describe("Social Settings", () => {
|
|
test.beforeEach(async ({ admin }) => {
|
|
await admin.devBypassAuth();
|
|
});
|
|
|
|
test("page renders with heading and form", async ({ admin, page }) => {
|
|
await admin.goto("/settings/social");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
// Page heading
|
|
await expect(page.locator("h1")).toContainText("Social Links");
|
|
|
|
// Should show the social profiles section
|
|
await expect(page.locator("text=Social Profiles")).toBeVisible({ timeout: 10000 });
|
|
});
|
|
|
|
test("displays all social input fields", async ({ admin, page }) => {
|
|
await admin.goto("/settings/social");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
// Each social field should have a visible input with its label
|
|
for (const label of ["Twitter", "GitHub", "Facebook", "Instagram", "LinkedIn", "YouTube"]) {
|
|
await expect(page.locator(`label:has-text("${label}")`)).toBeVisible({ timeout: 5000 });
|
|
}
|
|
|
|
// Save button should exist
|
|
await expect(page.locator("button", { hasText: "Save Social Links" })).toBeVisible();
|
|
});
|
|
|
|
test("saves a social link and persists across reload", async ({ admin, page }) => {
|
|
await admin.goto("/settings/social");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
const testHandle = `@e2e-test-${Date.now()}`;
|
|
|
|
// Fill the first social input field (Twitter)
|
|
const firstInput = page.locator("form input").first();
|
|
await firstInput.fill(testHandle);
|
|
|
|
// Wait for the save response
|
|
const saveResponse = page.waitForResponse(
|
|
(res) =>
|
|
SETTINGS_API_PATTERN.test(res.url()) &&
|
|
res.request().method() === "POST" &&
|
|
res.status() === 200,
|
|
{ timeout: 15000 },
|
|
);
|
|
|
|
// Click save
|
|
await page.locator("button", { hasText: "Save Social Links" }).click();
|
|
await saveResponse;
|
|
|
|
// Success banner should appear
|
|
await expect(page.locator("text=Social links saved")).toBeVisible({ timeout: 5000 });
|
|
|
|
// Reload the page
|
|
await admin.goto("/settings/social");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
// The value should persist
|
|
const firstInputAfterReload = page.locator("form input").first();
|
|
await expect(firstInputAfterReload).toHaveValue(testHandle, { timeout: 10000 });
|
|
});
|
|
});
|
|
|
|
test.describe("SEO Settings", () => {
|
|
test.beforeEach(async ({ admin }) => {
|
|
await admin.devBypassAuth();
|
|
});
|
|
|
|
test("page renders with heading and form", async ({ admin, page }) => {
|
|
await admin.goto("/settings/seo");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
// Page heading
|
|
await expect(page.locator("h1")).toContainText("SEO Settings");
|
|
|
|
// Should show the SEO section
|
|
await expect(page.locator("text=Search Engine Optimization")).toBeVisible({ timeout: 10000 });
|
|
});
|
|
|
|
test("displays expected SEO fields", async ({ admin, page }) => {
|
|
await admin.goto("/settings/seo");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
// Expected fields from SeoSettings.tsx
|
|
for (const label of [
|
|
"Title Separator",
|
|
"Google Verification",
|
|
"Bing Verification",
|
|
"robots.txt",
|
|
]) {
|
|
await expect(page.locator(`label:has-text("${label}")`)).toBeVisible({ timeout: 5000 });
|
|
}
|
|
|
|
// Save button
|
|
await expect(page.locator("button", { hasText: "Save SEO Settings" })).toBeVisible();
|
|
});
|
|
|
|
test("saves SEO settings and persists across reload", async ({ admin, page }) => {
|
|
await admin.goto("/settings/seo");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
const testVerification = `e2e-verify-${Date.now()}`;
|
|
|
|
// Fill the Google Verification field
|
|
const googleInput = page
|
|
.locator("label:has-text('Google Verification')")
|
|
.locator("..")
|
|
.locator("input");
|
|
await googleInput.fill(testVerification);
|
|
|
|
// Wait for save response
|
|
const saveResponse = page.waitForResponse(
|
|
(res) =>
|
|
SETTINGS_API_PATTERN.test(res.url()) &&
|
|
res.request().method() === "POST" &&
|
|
res.status() === 200,
|
|
{ timeout: 15000 },
|
|
);
|
|
|
|
// Click save
|
|
await page.locator("button", { hasText: "Save SEO Settings" }).click();
|
|
await saveResponse;
|
|
|
|
// Success banner
|
|
await expect(page.locator("text=SEO settings saved")).toBeVisible({ timeout: 5000 });
|
|
|
|
// Reload
|
|
await admin.goto("/settings/seo");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
// Value should persist
|
|
const googleInputAfterReload = page
|
|
.locator("label:has-text('Google Verification')")
|
|
.locator("..")
|
|
.locator("input");
|
|
await expect(googleInputAfterReload).toHaveValue(testVerification, { timeout: 10000 });
|
|
});
|
|
});
|
|
|
|
test.describe("Email Settings", () => {
|
|
test.beforeEach(async ({ admin }) => {
|
|
await admin.devBypassAuth();
|
|
});
|
|
|
|
test("page renders with heading and pipeline status", async ({ admin, page }) => {
|
|
await admin.goto("/settings/email");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
// Page heading
|
|
await expect(page.locator("h1")).toContainText("Email Settings");
|
|
|
|
// Should show the Email Pipeline section
|
|
await expect(page.locator("text=Email Pipeline")).toBeVisible({ timeout: 10000 });
|
|
});
|
|
|
|
test("shows pipeline section without crashing", async ({ admin, page }) => {
|
|
await admin.goto("/settings/email");
|
|
await admin.waitForShell();
|
|
await admin.waitForLoading();
|
|
|
|
// The Email Pipeline section heading should be visible
|
|
await expect(page.getByRole("heading", { name: "Email Pipeline" })).toBeVisible({
|
|
timeout: 10000,
|
|
});
|
|
});
|
|
});
|