- Replace links to `selectors.md` with `locators.md`. - Remove sections that are covered by Locators docs. - Restructure the rest of selectors doc, update them in terms of locators. Fixes #18992.
6.9 KiB
| id | title |
|---|---|
| testing-library | Migrating from Testing Library |
Migration principles
This guide describes migration to Playwright's Experimental Component Testing from DOM Testing Library, React Testing Library, Vue Testing Library and Svelte Testing Library.
:::note
If you use DOM Testing Library in the browser (for example, you bundle end-to-end tests with webpack), you can switch directly to Playwright Test. Examples below are focused on component tests, but for end-to-end test you just need to replace await mount with await page.goto('http://localhost:3000/') to open the page under test.
:::
Cheat Sheet
| Testing Library | Playwright |
|---|---|
| screen | page and component |
| queries | locators |
| async helpers | assertions |
| user events | actions |
await user.click(screen.getByText('Click me')) |
await component.getByText('Click me').click() |
await user.click(await screen.findByText('Click me')) |
await component.getByText('Click me').click() |
await user.type(screen.getByLabel('Password'), 'secret') |
await component.getByLabel('Password').fill('secret') |
expect(screen.getByLabel('Password')).toHaveValue('secret') |
await expect(component.getByLabel('Password')).toHaveValue('secret') |
screen.findByText('...') |
component.getByText('...') |
screen.getByTestId('...') |
component.getByTestId('...') |
screen.queryByPlaceholderText('...') |
component.getByPlaceholder('...') |
screen.getByRole('button', { pressed: true }) |
component.getByRole('button', { pressed: true }) |
Example
Testing Library:
import React from 'react';
import { render, screen } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
test('should sign in', async () => {
// Setup the page.
const user = userEvent.setup();
render(<SignInPage />);
// Perform actions.
await user.type(screen.getByLabel('Username'), 'John');
await user.type(screen.getByLabel('Password'), 'secret');
await user.click(screen.getByText('Sign in'));
// Verify signed in state by waiting until "Welcome" message appears.
await screen.findByText('Welcome, John');
});
Line-by-line migration to Playwright Test:
const { test, expect } = require('@playwright/experimental-ct-react'); // 1
test('should sign in', async ({ page, mount }) => { // 2
// Setup the page.
const component = await mount(<SignInPage />); // 3
// Perform actions.
await component.getByText('Username').fill('John'); // 4
await component.getByText('Password').fill('secret');
await component.getByText('Sign in').click();
// Verify signed in state by waiting until "Welcome" message appears.
await expect(component.getByText('Welcome, John')).toBeVisible(); // 5
});
Migration highlights (see inline comments in the Playwright Test code snippet):
- Import everything from
@playwright/experimental-ct-react(or -vue, -svelte) for component tests, or from@playwright/testfor end-to-end tests. - Test function is given a
pagethat is isolated from other tests, andmountthat renders a component in this page. These are two of the useful fixtures in Playwright Test. - Replace
renderwithmountthat returns a component locator. - Use locators created with [
method: Locator.locator] or [method: Page.locator] to perform most of the actions. - Use assertions to verify the state.
Migrating queries
All queries like getBy..., findBy..., queryBy... and their multi-element counterparts are replaced with component.getBy... locators. Locators always auto-wait and retry when needed, so you don't have to worry about choosing the right method. When you want to do a list operation, e.g. assert a list of texts, Playwright automatically performs multi-element operations.
Replacing waitFor
Playwright includes assertions that automatically wait for the condition, so you don't usually need an explicit waitFor/waitForElementToBeRemoved call.
// Testing Library
await waitFor(() => {
expect(getByText('the lion king')).toBeInTheDocument()
})
await waitForElementToBeRemoved(() => queryByText('the mummy'))
// Playwright
await expect(page.getByText('the lion king')).toBeVisible()
await expect(page.getByText('the mummy')).toBeHidden()
When you cannot find a suitable assertion, use expect.poll instead.
await expect.poll(async () => {
const response = await page.request.get('https://api.example.com');
return response.status();
}).toBe(200);
Replacing within
You can create a locator inside another locator with [method: Locator.locator] method.
// Testing Library
const messages = document.getElementById('messages')
const helloMessage = within(messages).getByText('hello')
// Playwright
const messages = component.locator('id=messages')
const helloMessage = messages.getByText('hello')
Playwright Test Super Powers
Once you're on Playwright Test, you get a lot!
- Full zero-configuration TypeScript support
- Run tests across all web engines (Chrome, Firefox, Safari) on any popular operating system (Windows, macOS, Ubuntu)
- Full support for multiple origins, (i)frames, tabs and contexts
- Run tests in isolation in parallel across multiple browsers
- Built-in test artifact collection: video recording, screenshots and playwright traces
You also get all these ✨ awesome tools ✨ that come bundled with Playwright Test:
- Playwright Inspector
- Playwright Test Code generation
- Playwright Tracing for post-mortem debugging
Further Reading
Learn more about Playwright Test runner: