Testing with Playwright
Quick Start
import { defineConfig, devices } from "@playwright/test";
export default defineConfig({
testDir: "./tests/e2e",
fullyParallel: true,
retries: process.env.CI ? 2 : 0,
reporter: [["list"], ["html"]],
use: {
baseURL: "http://localhost:3000",
trace: "on-first-retry",
screenshot: "only-on-failure",
},
projects: [
{ name: "chromium", use: { ...devices["Desktop Chrome"] } },
{ name: "firefox", use: { ...devices["Desktop Firefox"] } },
{ name: "mobile", use: { ...devices["iPhone 12"] } },
],
webServer: { command: "npm run dev", url: "http://localhost:3000" },
});
Features
| Feature |
Description |
Reference |
| Page Object Model |
Maintainable test architecture pattern |
POM Guide |
| Auto-Waiting |
Built-in waiting for elements and assertions |
Auto-Waiting |
| Network Mocking |
Intercept and mock API responses |
Network |
| Visual Testing |
Screenshot comparison for regression testing |
Visual Comparisons |
| Cross-Browser |
Chrome, Firefox, Safari, mobile devices |
Browsers |
| Trace Viewer |
Debug failing tests with timeline |
Trace Viewer |
Common Patterns
Page Object Model
import { Page, Locator, expect } from "@playwright/test";
export class LoginPage {
readonly emailInput: Locator;
readonly passwordInput: Locator;
readonly submitButton: Locator;
constructor(private page: Page) {
this.emailInput = page.getByLabel("Email");
this.passwordInput = page.getByLabel("Password");
this.submitButton = page.getByRole("button", { name: "Sign in" });
}
async login(email: string, password: string) {
await this.emailInput.fill(email);
await this.passwordInput.fill(password);
await this.submitButton.click();
}
async expectError(message: string) {
await expect(this.page.getByRole("alert")).toContainText(message);
}
}
API Mocking
import { test, expect } from "@playwright/test";
test("mock API response", async ({ page }) => {
await page.route("**/api/users", (route) =>
route.fulfill({
status: 200,
contentType: "application/json",
body: JSON.stringify({ users: [{ id: 1, name: "John" }] }),
})
);
await page.goto("/users");
await expect(page.getByText("John")).toBeVisible();
});
test("capture network requests", async ({ page }) => {
const requestPromise = page.waitForRequest("**/api/analytics");
await page.goto("/dashboard");
const request = await requestPromise;
expect(request.postDataJSON()).toMatchObject({ event: "page_view" });
});
Authentication Fixture
import { test as base } from "@playwright/test";
import { LoginPage } from "../pages/login.page";
export const test = base.extend<{ authenticatedPage: Page }>({
authenticatedPage: async ({ page }, use) => {
const response = await page.request.post("/api/auth/login", {
data: { email: "test@example.com", password: "password" },
});
const { token } = await response.json();
await page.context().addCookies([
{ name: "auth_token", value: token, domain: "localhost", path: "/" },
]);
await page.goto("/dashboard");
await use(page);
},
});
Visual Regression Testing
test("visual snapshot", async ({ page }) => {
await page.goto("/");
await page.addStyleTag({
content: "*, *::before, *::after { animation-duration: 0s !important; }",
});
await expect(page).toHaveScreenshot("homepage.png", {
fullPage: true,
maxDiffPixels: 100,
});
await expect(page).toHaveScreenshot("dashboard.png", {
mask: [page.getByTestId("timestamp"), page.getByTestId("avatar")],
});
});
Best Practices
| Do |
Avoid |
| Use Page Object Model for maintainability |
Fragile CSS selectors |
| Prefer user-facing locators (getByRole, getByLabel) |
Relying on arbitrary waits |
| Use API auth for faster test setup |
Sharing state between tests |
| Enable traces and screenshots for debugging |
Testing third-party services directly |
| Run tests in parallel for speed |
Skipping flaky tests without fixing |
| Mock external APIs for reliability |
Hardcoding test data |
References