Testing infrastructure, fix truncating of dataURLs (#26)

* feat: implement comprehensive testing infrastructure

- Fix image dataURL truncation bug in security.ts with configurable size limits
- Add backend integration tests (22 tests) with Vitest for API validation
- Add frontend unit tests (11 tests) for JSON serialization
- Implement browser-based E2E tests (8 tests) with Playwright
- Create Docker setup for repeatable E2E testing environment
- Add GitHub Actions CI workflow for automated testing
- Update .gitignore for test artifacts and temporary files

Testing Infrastructure:
- Backend: Vitest + Supertest for API integration tests
- Frontend: Vitest + Testing Library for component tests
- E2E: Playwright with Chromium for full browser automation
- CI/CD: GitHub Actions with parallel test execution

Security Improvements:
- Make dataURL size limit configurable (default: 10MB)
- Enhanced validation for image dataURLs
- Block malicious content (javascript:, script tags)

All tests pass: 41 total (22 backend + 11 frontend + 8 E2E)

* feat(tests): add comprehensive E2E tests for dashboard workflows and image persistence
chore(env): update environment variables for consistent API URL usage
fix(api): centralize API request helpers for drawing and collection management
style(DrawingCard): enhance accessibility with ARIA attributes and data-testid for testing

* cleanup/revise documentation

* cleanup/revise documentation

* Add end-to-end tests for drawing CRUD, export/import, search/sort, and theme toggle functionalities

- Implemented E2E tests for drawing creation, editing, and deletion in `drawing-crud.spec.ts`.
- Added tests for export and import features, including JSON and SQLite formats in `export-import.spec.ts`.
- Created tests for searching and sorting drawings by name and date in `search-and-sort.spec.ts`.
- Developed tests for theme toggle functionality to ensure persistence across sessions in `theme-toggle.spec.ts`.

* fix: exclude test files from production build to fix Docker build

* feat: implement comprehensive testing infrastructure (#19)

* bump version 0.1.7

* feat: implement comprehensive testing infrastructure

- Fix image dataURL truncation bug in security.ts with configurable size limits
- Add backend integration tests (22 tests) with Vitest for API validation
- Add frontend unit tests (11 tests) for JSON serialization
- Implement browser-based E2E tests (8 tests) with Playwright
- Create Docker setup for repeatable E2E testing environment
- Add GitHub Actions CI workflow for automated testing
- Update .gitignore for test artifacts and temporary files

Testing Infrastructure:
- Backend: Vitest + Supertest for API integration tests
- Frontend: Vitest + Testing Library for component tests
- E2E: Playwright with Chromium for full browser automation
- CI/CD: GitHub Actions with parallel test execution

Security Improvements:
- Make dataURL size limit configurable (default: 10MB)
- Enhanced validation for image dataURLs
- Block malicious content (javascript:, script tags)

All tests pass: 41 total (22 backend + 11 frontend + 8 E2E)

* feat(tests): add comprehensive E2E tests for dashboard workflows and image persistence
chore(env): update environment variables for consistent API URL usage
fix(api): centralize API request helpers for drawing and collection management
style(DrawingCard): enhance accessibility with ARIA attributes and data-testid for testing

* Add end-to-end tests for drawing CRUD, export/import, search/sort, and theme toggle functionalities

- Implemented E2E tests for drawing creation, editing, and deletion in `drawing-crud.spec.ts`.
- Added tests for export and import features, including JSON and SQLite formats in `export-import.spec.ts`.
- Created tests for searching and sorting drawings by name and date in `search-and-sort.spec.ts`.
- Developed tests for theme toggle functionality to ensure persistence across sessions in `theme-toggle.spec.ts`.

* Update backend/src/__tests__/testUtils.ts

---------

Co-authored-by: Zimeng Xiong <zxzimeng@gmail.com>
* version bump 0.1.8

* fix(ci): consolidate E2E server startup to prevent shell isolation issues

Background processes started with & in separate GitHub Actions run steps
can terminate when those steps complete because each step creates a new
shell. This caused the backend and frontend servers to die before the
E2E tests could run.

Fixed by consolidating server startup and test execution into a single
shell step with:
- Proper PID tracking for cleanup
- Health check loops instead of fixed sleep times
- All processes run in the same shell session

* fix(ci): use absolute database path for E2E tests

* fix(backend): use resolved DATABASE_URL path for export/import endpoints

---------

Co-authored-by: Adrian Acala <adrianacala017@gmail.com>
This commit is contained in:
Zimeng Xiong
2025-12-19 15:09:15 -08:00
committed by GitHub
parent 18c8595c2e
commit 49b413bf07
79 changed files with 7628 additions and 14742 deletions
+182
View File
@@ -0,0 +1,182 @@
import { test, expect } from "@playwright/test";
import type { Locator, Page } from "@playwright/test";
import {
API_URL,
createDrawing,
deleteDrawing,
getDrawing,
listDrawings,
listCollections,
deleteCollection,
} from "./helpers/api";
const searchPlaceholder = "Search drawings...";
async function applyDashboardSearch(page: Page, term: string) {
const searchInput = page.getByPlaceholder(searchPlaceholder);
await searchInput.waitFor();
await searchInput.fill("");
await searchInput.fill(term);
}
async function ensureCardVisible(page: Page, drawingId: string): Promise<Locator> {
const card = page.locator(`#drawing-card-${drawingId}`);
await card.waitFor({ state: "attached" });
await card.scrollIntoViewIfNeeded();
await expect(card).toBeVisible();
return card;
}
async function ensureCardSelected(page: Page, drawingId: string) {
const card = await ensureCardVisible(page, drawingId);
const toggle = card.locator(`[data-testid="select-drawing-${drawingId}"]`);
const pressed = await toggle.getAttribute("aria-pressed");
if (pressed !== "true") {
await card.hover();
await toggle.click();
}
}
test.describe("Dashboard Workflows", () => {
let createdDrawingIds: string[] = [];
let createdCollectionIds: string[] = [];
test.afterEach(async ({ request }) => {
for (const id of createdDrawingIds) {
try {
await deleteDrawing(request, id);
} catch (error) {
// Ignore cleanup failures to keep tests resilient
}
}
createdDrawingIds = [];
for (const id of createdCollectionIds) {
try {
await deleteCollection(request, id);
} catch (error) {
// Ignore cleanup failures to keep tests resilient
}
}
createdCollectionIds = [];
});
test("should move drawing to trash and permanently delete it via bulk controls", async ({ page, request }) => {
const drawingName = `Trash Workflow ${Date.now()}`;
const createdDrawing = await createDrawing(request, { name: drawingName });
createdDrawingIds.push(createdDrawing.id);
await page.goto("/");
await page.waitForLoadState("networkidle");
await applyDashboardSearch(page, drawingName);
const cardLocator = await ensureCardVisible(page, createdDrawing.id);
await ensureCardSelected(page, createdDrawing.id);
await page.getByTitle("Move to Trash").click();
await expect(cardLocator).toHaveCount(0);
await page.getByRole("button", { name: /^Trash$/ }).click();
const trashCard = await ensureCardVisible(page, createdDrawing.id);
await ensureCardSelected(page, createdDrawing.id);
await page.getByTitle("Delete Permanently").click();
await page.getByRole("button", { name: /Delete \d+ Drawings/ }).click();
await expect(trashCard).toHaveCount(0);
const response = await request.get(`${API_URL}/drawings/${createdDrawing.id}`);
expect(response.status()).toBe(404);
createdDrawingIds = createdDrawingIds.filter((id) => id !== createdDrawing.id);
});
test("should create a collection via UI and move drawings using card controls", async ({ page, request }) => {
const drawingName = `Collection Flow ${Date.now()}`;
const createdDrawing = await createDrawing(request, { name: drawingName });
createdDrawingIds.push(createdDrawing.id);
const collectionName = `Team ${Date.now()}`;
await page.goto("/");
await page.waitForLoadState("networkidle");
await applyDashboardSearch(page, drawingName);
await page.getByTitle("New Collection").click();
const collectionInput = page.getByPlaceholder("New Collection...");
await collectionInput.fill(collectionName);
await collectionInput.press("Enter");
await expect(page.getByRole("button", { name: collectionName })).toBeVisible();
const collections = await listCollections(request);
const createdCollection = collections.find((collection) => collection.name === collectionName);
expect(createdCollection).toBeDefined();
if (!createdCollection) {
throw new Error("Failed to locate created collection");
}
createdCollectionIds.push(createdCollection.id);
const cardLocator = await ensureCardVisible(page, createdDrawing.id);
const collectionButton = cardLocator.locator(`[data-testid="collection-picker-${createdDrawing.id}"]`);
await collectionButton.click();
await page.locator(`[data-testid="collection-option-${createdCollection.id}"]`).click();
await expect(collectionButton).toContainText(collectionName);
await expect.poll(async () => {
const updated = await getDrawing(request, createdDrawing.id);
return updated.collectionId;
}).toBe(createdCollection.id);
await page.getByRole("navigation").getByRole("button", { name: collectionName }).click();
await expect(cardLocator).toBeVisible();
await page.getByRole("navigation").getByRole("button", { name: "Unorganized" }).click();
await expect(cardLocator).toHaveCount(0);
});
test("should duplicate multiple drawings and move them to trash via bulk toolbar", async ({ page, request }) => {
const prefix = `Bulk Flow ${Date.now()}`;
const [first, second] = await Promise.all([
createDrawing(request, { name: `${prefix} A` }),
createDrawing(request, { name: `${prefix} B` }),
]);
createdDrawingIds.push(first.id, second.id);
await page.goto("/");
await page.waitForLoadState("networkidle");
await applyDashboardSearch(page, prefix);
await ensureCardSelected(page, first.id);
await ensureCardSelected(page, second.id);
await page.getByTitle("Duplicate Selected").click();
await expect.poll(async () => {
const results = await listDrawings(request, { search: prefix });
return results.length;
}).toBe(4);
const allPrefixDrawings = await listDrawings(request, { search: prefix });
for (const drawing of allPrefixDrawings) {
await ensureCardSelected(page, drawing.id);
}
await page.getByTitle("Move to Trash").click();
await expect.poll(async () => {
const trashed = await listDrawings(request, { search: prefix, collectionId: "trash" });
return trashed.length;
}).toBe(4);
const trashDrawings = await listDrawings(request, { search: prefix, collectionId: "trash" });
for (const drawing of trashDrawings) {
await deleteDrawing(request, drawing.id);
}
const removedIds = new Set(trashDrawings.map((drawing) => drawing.id));
createdDrawingIds = createdDrawingIds.filter((id) => !removedIds.has(id));
await expect.poll(async () => {
const remaining = await listDrawings(request, { search: prefix });
return remaining.length;
}).toBe(0);
});
});