一键导入
e2e-testing
// Run, debug, and fix Playwright E2E tests locally. Use when writing new E2E tests, debugging failing tests, or iterating on test fixes before pushing to CI.
// Run, debug, and fix Playwright E2E tests locally. Use when writing new E2E tests, debugging failing tests, or iterating on test fixes before pushing to CI.
Connect to local or staging PostgreSQL database. Use when you need to inspect tables, run queries, check migration status, or debug database issues.
Railway CLI operations for authentication, context management, deployment, monitoring, and troubleshooting. Use when working with Railway infrastructure, deploying services, or managing environments.
| name | E2E Testing |
| description | Run, debug, and fix Playwright E2E tests locally. Use when writing new E2E tests, debugging failing tests, or iterating on test fixes before pushing to CI. |
The E2E tests use a separate iota_erp_e2e database. Create it if missing:
# Check DB connection params from .env (DB_HOST, DB_PORT, DB_PASSWORD)
grep '^DB_' .env
# Create the database (adjust host/port from .env)
PGPASSWORD=postgres psql -U postgres -h localhost -p 5438 -c "CREATE DATABASE iota_erp_e2e;"
# Run migrations
just e2e migrate up
The server must run with test endpoints enabled:
PORT=3201 ORIGIN='http://localhost:3201' DB_NAME=iota_erp_e2e ENABLE_TEST_ENDPOINTS=true \
go run -tags e2e ./cmd/server/main.go
Or use just e2e dev (uses air hot-reload, same env vars).
Verify it's running:
curl -s -o /dev/null -w "%{http_code}" http://localhost:3201/login
# Should return 200
kill $(lsof -ti:3201) 2>/dev/null
# Run a specific test file
cd e2e && npx playwright test tests/roles/roles.spec.ts --reporter=line
# Run a specific test by line number
cd e2e && npx playwright test tests/roles/roles.spec.ts:26 --reporter=line
# Run all E2E tests (headless)
just e2e ci
# Run with Playwright UI (interactive debugging)
just e2e run
When fixing a failing E2E test, follow this loop:
e2e/test-results/ after failures.webm) for timing/animation issues# Quick iteration: run just the failing test
cd e2e && npx playwright test tests/module/test.spec.ts:LINE --reporter=line
# After fixing, run the full suite
cd e2e && npx playwright test tests/module/test.spec.ts --reporter=line
# View failure screenshot
# (path shown in test output, use Read tool to view)
The <dialog> top-layer positioning confuses Playwright's elementFromPoint() hit testing
in headless Chromium. The bottom action bar (sticky footer) intercepts pointer events even
though the dialog is visually on top.
Workaround: After verifying the dialog is visible, trigger the htmx form submit directly instead of clicking the confirm button:
// Open the confirmation dialog
await page.locator('[data-test-id="delete-btn"]').click();
// Wait for dialog to appear
const dialog = page.locator('[data-test-id="delete-confirmation-dialog"]');
await expect(dialog).toBeVisible();
// Trigger htmx delete directly (bypasses dialog hit-test issue)
await page.evaluate(() => {
const form = document.getElementById('delete-form') as HTMLFormElement;
(window as any).htmx.trigger(form, 'submit');
});
await page.waitForURL(/\/expected-path$/);
Prefer data-test-id attributes over text-based or structural selectors:
// Good
page.locator('[data-test-id="save-role-btn"]')
page.locator('[data-test-id="dialog-confirm-btn"]')
// Fragile
page.locator('button').filter({ hasText: /Save/i })
page.getByRole('button', { name: /confirm/i })
import { waitForAlpine } from '../../fixtures/auth';
await waitForAlpine(page);
Tests use beforeAll to reset and seed:
import { resetTestDatabase, seedScenario } from '../../fixtures/test-data';
test.beforeAll(async ({ request }) => {
await resetTestDatabase(request, { reseedMinimal: false });
await seedScenario(request, 'comprehensive');
});
e2e/
├── fixtures/
│ ├── auth.ts # login, logout, waitForAlpine helpers
│ └── test-data.ts # resetTestDatabase, seedScenario
├── tests/
│ ├── roles/ # Role management tests
│ ├── users/ # User management tests
│ └── ...
├── playwright.config.ts # Playwright config (1280x720 viewport)
└── test-results/ # Screenshots, videos, traces from failures
http://localhost:3201