284 lines
8.7 KiB
Markdown
284 lines
8.7 KiB
Markdown
# Playwright Utils Overview
|
|
|
|
## Principle
|
|
|
|
Use production-ready, fixture-based utilities from `@seontechnologies/playwright-utils` for common Playwright testing patterns. Build test helpers as pure functions first, then wrap in framework-specific fixtures for composability and reuse.
|
|
|
|
## Rationale
|
|
|
|
Writing Playwright utilities from scratch for every project leads to:
|
|
|
|
- Duplicated code across test suites
|
|
- Inconsistent patterns and quality
|
|
- Maintenance burden when Playwright APIs change
|
|
- Missing advanced features (schema validation, HAR recording, auth persistence)
|
|
|
|
`@seontechnologies/playwright-utils` provides:
|
|
|
|
- **Production-tested utilities**: Used at SEON Technologies in production
|
|
- **Functional-first design**: Core logic as pure functions, fixtures for convenience
|
|
- **Composable fixtures**: Use `mergeTests` to combine utilities
|
|
- **TypeScript support**: Full type safety with generic types
|
|
- **Comprehensive coverage**: API requests, auth, network, logging, file handling, burn-in
|
|
|
|
## Installation
|
|
|
|
```bash
|
|
npm install -D @seontechnologies/playwright-utils
|
|
```
|
|
|
|
**Peer Dependencies:**
|
|
|
|
- `@playwright/test` >= 1.54.1 (required)
|
|
- `ajv` >= 8.0.0 (optional - for JSON Schema validation)
|
|
- `zod` >= 3.0.0 (optional - for Zod schema validation)
|
|
|
|
## Available Utilities
|
|
|
|
### Core Testing Utilities
|
|
|
|
| Utility | Purpose | Test Context |
|
|
| -------------------------- | ------------------------------------------ | ------------- |
|
|
| **api-request** | Typed HTTP client with schema validation | API tests |
|
|
| **network-recorder** | HAR record/playback for offline testing | UI tests |
|
|
| **auth-session** | Token persistence, multi-user auth | Both UI & API |
|
|
| **recurse** | Cypress-style polling for async conditions | Both UI & API |
|
|
| **intercept-network-call** | Network spy/stub with auto JSON parsing | UI tests |
|
|
| **log** | Playwright report-integrated logging | Both UI & API |
|
|
| **file-utils** | CSV/XLSX/PDF/ZIP reading & validation | Both UI & API |
|
|
| **burn-in** | Smart test selection with git diff | CI/CD |
|
|
| **network-error-monitor** | Automatic HTTP 4xx/5xx detection | UI tests |
|
|
|
|
## Design Patterns
|
|
|
|
### Pattern 1: Functional Core, Fixture Shell
|
|
|
|
**Context**: All utilities follow the same architectural pattern - pure function as core, fixture as wrapper.
|
|
|
|
**Implementation**:
|
|
|
|
```typescript
|
|
// Direct import (pass Playwright context explicitly)
|
|
import { apiRequest } from '@seontechnologies/playwright-utils';
|
|
|
|
test('direct usage', async ({ request }) => {
|
|
const { status, body } = await apiRequest({
|
|
request, // Must pass request context
|
|
method: 'GET',
|
|
path: '/api/users',
|
|
});
|
|
});
|
|
|
|
// Fixture import (context injected automatically)
|
|
import { test } from '@seontechnologies/playwright-utils/fixtures';
|
|
|
|
test('fixture usage', async ({ apiRequest }) => {
|
|
const { status, body } = await apiRequest({
|
|
// No need to pass request context
|
|
method: 'GET',
|
|
path: '/api/users',
|
|
});
|
|
});
|
|
```
|
|
|
|
**Key Points**:
|
|
|
|
- Pure functions testable without Playwright running
|
|
- Fixtures inject framework dependencies automatically
|
|
- Choose direct import (more control) or fixture (convenience)
|
|
|
|
### Pattern 2: Subpath Imports for Tree-Shaking
|
|
|
|
**Context**: Import only what you need to keep bundle sizes small.
|
|
|
|
**Implementation**:
|
|
|
|
```typescript
|
|
// Import specific utility
|
|
import { apiRequest } from '@seontechnologies/playwright-utils/api-request';
|
|
|
|
// Import specific fixture
|
|
import { test } from '@seontechnologies/playwright-utils/api-request/fixtures';
|
|
|
|
// Import everything (use sparingly)
|
|
import { apiRequest, recurse, log } from '@seontechnologies/playwright-utils';
|
|
```
|
|
|
|
**Key Points**:
|
|
|
|
- Subpath imports enable tree-shaking
|
|
- Keep bundle sizes minimal
|
|
- Import from specific paths for production builds
|
|
|
|
### Pattern 3: Fixture Composition with mergeTests
|
|
|
|
**Context**: Combine multiple playwright-utils fixtures with your own custom fixtures.
|
|
|
|
**Implementation**:
|
|
|
|
```typescript
|
|
// playwright/support/merged-fixtures.ts
|
|
import { mergeTests } from '@playwright/test';
|
|
import { test as apiRequestFixture } from '@seontechnologies/playwright-utils/api-request/fixtures';
|
|
import { test as authFixture } from '@seontechnologies/playwright-utils/auth-session/fixtures';
|
|
import { test as recurseFixture } from '@seontechnologies/playwright-utils/recurse/fixtures';
|
|
import { test as logFixture } from '@seontechnologies/playwright-utils/log/fixtures';
|
|
|
|
// Merge all fixtures into one test object
|
|
export const test = mergeTests(apiRequestFixture, authFixture, recurseFixture, logFixture);
|
|
|
|
export { expect } from '@playwright/test';
|
|
```
|
|
|
|
```typescript
|
|
// In your tests
|
|
import { test, expect } from '../support/merged-fixtures';
|
|
|
|
test('all utilities available', async ({ apiRequest, authToken, recurse, log }) => {
|
|
await log.step('Making authenticated API request');
|
|
|
|
const { body } = await apiRequest({
|
|
method: 'GET',
|
|
path: '/api/protected',
|
|
headers: { Authorization: `Bearer ${authToken}` },
|
|
});
|
|
|
|
await recurse(
|
|
() => apiRequest({ method: 'GET', path: `/status/${body.id}` }),
|
|
(res) => res.body.ready === true,
|
|
);
|
|
});
|
|
```
|
|
|
|
**Key Points**:
|
|
|
|
- `mergeTests` combines multiple fixtures without conflicts
|
|
- Create one merged-fixtures.ts file per project
|
|
- Import test object from your merged fixtures in all tests
|
|
- All utilities available in single test signature
|
|
|
|
## Integration with Existing Tests
|
|
|
|
### Gradual Adoption Strategy
|
|
|
|
**1. Start with logging** (zero breaking changes):
|
|
|
|
```typescript
|
|
import { log } from '@seontechnologies/playwright-utils';
|
|
|
|
test('existing test', async ({ page }) => {
|
|
await log.step('Navigate to page'); // Just add logging
|
|
await page.goto('/dashboard');
|
|
// Rest of test unchanged
|
|
});
|
|
```
|
|
|
|
**2. Add API utilities** (for API tests):
|
|
|
|
```typescript
|
|
import { test } from '@seontechnologies/playwright-utils/api-request/fixtures';
|
|
|
|
test('API test', async ({ apiRequest }) => {
|
|
const { status, body } = await apiRequest({
|
|
method: 'GET',
|
|
path: '/api/users',
|
|
});
|
|
|
|
expect(status).toBe(200);
|
|
});
|
|
```
|
|
|
|
**3. Expand to network utilities** (for UI tests):
|
|
|
|
```typescript
|
|
import { test } from '@seontechnologies/playwright-utils/fixtures';
|
|
|
|
test('UI with network control', async ({ page, interceptNetworkCall }) => {
|
|
const usersCall = interceptNetworkCall({
|
|
url: '**/api/users',
|
|
});
|
|
|
|
await page.goto('/dashboard');
|
|
const { responseJson } = await usersCall;
|
|
|
|
expect(responseJson).toHaveLength(10);
|
|
});
|
|
```
|
|
|
|
**4. Full integration** (merged fixtures):
|
|
|
|
Create merged-fixtures.ts and use across all tests.
|
|
|
|
## Related Fragments
|
|
|
|
- `api-request.md` - HTTP client with schema validation
|
|
- `network-recorder.md` - HAR-based offline testing
|
|
- `auth-session.md` - Token management
|
|
- `intercept-network-call.md` - Network interception
|
|
- `recurse.md` - Polling patterns
|
|
- `log.md` - Logging utility
|
|
- `file-utils.md` - File operations
|
|
- `fixtures-composition.md` - Advanced mergeTests patterns
|
|
|
|
## Anti-Patterns
|
|
|
|
**❌ Don't mix direct and fixture imports in same test:**
|
|
|
|
```typescript
|
|
import { apiRequest } from '@seontechnologies/playwright-utils';
|
|
import { test } from '@seontechnologies/playwright-utils/auth-session/fixtures';
|
|
|
|
test('bad', async ({ request, authToken }) => {
|
|
// Confusing - mixing direct (needs request) and fixture (has authToken)
|
|
await apiRequest({ request, method: 'GET', path: '/api/users' });
|
|
});
|
|
```
|
|
|
|
**✅ Use consistent import style:**
|
|
|
|
```typescript
|
|
import { test } from '../support/merged-fixtures';
|
|
|
|
test('good', async ({ apiRequest, authToken }) => {
|
|
// Clean - all from fixtures
|
|
await apiRequest({ method: 'GET', path: '/api/users' });
|
|
});
|
|
```
|
|
|
|
**❌ Don't import everything when you need one utility:**
|
|
|
|
```typescript
|
|
import * as utils from '@seontechnologies/playwright-utils'; // Large bundle
|
|
```
|
|
|
|
**✅ Use subpath imports:**
|
|
|
|
```typescript
|
|
import { apiRequest } from '@seontechnologies/playwright-utils/api-request'; // Small bundle
|
|
```
|
|
|
|
## Reference Implementation
|
|
|
|
The official `@seontechnologies/playwright-utils` repository provides working examples of all patterns described in these fragments.
|
|
|
|
**Repository:** <https://github.com/seontechnologies/playwright-utils>
|
|
|
|
**Key resources:**
|
|
|
|
- **Test examples:** `playwright/tests` - All utilities in action
|
|
- **Framework setup:** `playwright.config.ts`, `playwright/support/merged-fixtures.ts`
|
|
- **CI patterns:** `.github/workflows/` - GitHub Actions with sharding, parallelization
|
|
|
|
**Quick start:**
|
|
|
|
```bash
|
|
git clone https://github.com/seontechnologies/playwright-utils.git
|
|
cd playwright-utils
|
|
nvm use
|
|
npm install
|
|
npm run test:pw-ui # Explore tests with Playwright UI
|
|
npm run test:pw
|
|
```
|
|
|
|
All patterns in TEA fragments are production-tested in this repository.
|