frontend/testing-guide
This Claude Code skill provides comprehensive frontend testing guidelines covering unit tests, integration tests, and end-to-end tests for React applications. It specifies a testing pyramid structure requiring approximately 70% unit tests for components and hooks, 20% integration tests for component interactions, and 10% mandatory E2E tests for user workflows. Use this resource when establishing testing standards for frontend development teams or implementing test coverage requirements across React projects.
git clone --depth 1 https://github.com/echoVic/boss-skill /tmp/frontend-testing-guide && cp -r /tmp/frontend-testing-guide/skill/skills/frontend/testing-guide ~/.claude/skills/frontend-testing-guideSKILL.md
# 前端测试编写指南
## 测试要求(强制)
> **职责边界**:Frontend Agent 是测试的**编写者**,QA Agent 是测试的**验证者**。
### 测试金字塔
| 测试类型 | 占比 | 要求 |
|----------|------|------|
| **单元测试** | ~70% | 每个组件/Hook 必须有测试 |
| **集成测试** | ~20% | 组件交互、状态管理测试 |
| **E2E 测试** | ~10% | **必须编写**,覆盖用户流程 |
## 单元测试编写
### 组件渲染测试
```typescript
// Button.test.tsx
import { render, screen } from '@testing-library/react';
import { Button } from './Button';
describe('Button', () => {
it('renders with text', () => {
render(<Button>Click me</Button>);
expect(screen.getByText('Click me')).toBeInTheDocument();
});
it('calls onClick when clicked', () => {
const handleClick = jest.fn();
render(<Button onClick={handleClick}>Click</Button>);
screen.getByText('Click').click();
expect(handleClick).toHaveBeenCalledTimes(1);
});
it('is disabled when disabled prop is true', () => {
render(<Button disabled>Click</Button>);
expect(screen.getByText('Click')).toBeDisabled();
});
});
```
### Hook 测试
```typescript
// useCounter.test.ts
import { renderHook, act } from '@testing-library/react';
import { useCounter } from './useCounter';
describe('useCounter', () => {
it('initializes with default value', () => {
const { result } = renderHook(() => useCounter());
expect(result.current.count).toBe(0);
});
it('increments count', () => {
const { result } = renderHook(() => useCounter());
act(() => {
result.current.increment();
});
expect(result.current.count).toBe(1);
});
});
```
### 表单验证测试
```typescript
// LoginForm.test.tsx
import { render, screen, waitFor } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import { LoginForm } from './LoginForm';
describe('LoginForm', () => {
it('shows validation error for invalid email', async () => {
render(<LoginForm />);
const emailInput = screen.getByLabelText('Email');
await userEvent.type(emailInput, 'invalid-email');
await userEvent.tab(); // Trigger blur
await waitFor(() => {
expect(screen.getByText('Invalid email format')).toBeInTheDocument();
});
});
it('submits form with valid data', async () => {
const onSubmit = jest.fn();
render(<LoginForm onSubmit={onSubmit} />);
await userEvent.type(screen.getByLabelText('Email'), 'user@example.com');
await userEvent.type(screen.getByLabelText('Password'), 'password123');
await userEvent.click(screen.getByRole('button', { name: 'Login' }));
await waitFor(() => {
expect(onSubmit).toHaveBeenCalledWith({
email: 'user@example.com',
password: 'password123',
});
});
});
});
```
## 集成测试编写
### 组件交互测试
```typescript
// UserList.test.tsx
import { render, screen, waitFor } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import { UserList } from './UserList';
import { UserProvider } from './UserContext';
describe('UserList integration', () => {
it('adds new user to list', async () => {
render(
<UserProvider>
<UserList />
</UserProvider>
);
// 打开添加用户表单
await userEvent.click(screen.getByText('Add User'));
// 填写表单
await userEvent.type(screen.getByLabelText('Name'), 'John Doe');
await userEvent.type(screen.getByLabelText('Email'), 'john@example.com');
// 提交
await userEvent.click(screen.getByRole('button', { name: 'Submit' }));
// 验证用户出现在列表中
await waitFor(() => {
expect(screen.getByText('John Doe')).toBeInTheDocument();
expect(screen.getByText('john@example.com')).toBeInTheDocument();
});
});
});
```
### 状态管理测试
```typescript
// store.test.ts
import { renderHook, act } from '@testing-library/react';
import { useStore } from './store';
describe('Store integration', () => {
it('updates user state across components', () => {
const { result } = renderHook(() => useStore());
act(() => {
result.current.setUser({ id: '1', name: 'Alice' });
});
expect(result.current.user).toEqual({ id: '1', name: 'Alice' });
act(() => {
result.current.updateUserName('Bob');
});
expect(result.current.user?.name).toBe('Bob');
});
});
```
## E2E 测试编写(必须)
> **完整 Playwright 方法论**:详见 `Skill(skill: "qa/e2e-playwright")`,包含项目初始化、Page Object Model、认证复用、API Mock、视觉回归、多浏览器测试、CI 集成和调试技巧。
### E2E 测试必须覆盖
- ✅ 创建流程(如:添加数据)
- ✅ 编辑流程(如:修改数据)
- ✅ 删除流程(如:删除数据)
- ✅ 列表展示(如:查看列表)
- ✅ 核心业务流程
### Playwright 示例(Page Object Model)
```typescript
// e2e/pages/user-list.page.ts
import { type Page, type Locator } from '@playwright/test';
export class UserListPage {
private readonly addButton: Locator;
private readonly table: Locator;
constructor(private readonly page: Page) {
this.addButton = page.getByRole('button', { name: '添加用户' });
this.table = page.getByRole('table');
}
async goto() { await this.page.goto('/users'); }
async clickAddUser() { await this.addButton.click(); }
getTable() { return this.table; }
async editUser(name: string) {
await this.page.getByRole('row', { name }).getByRole('button', { name: '编辑' }).click();
}
async deleteUser(name: string) {
await this.page.getByRole('row', { name }).getByRole('button', { name: '删除' }).click();
}
async confirmDelete() {
await this.page.getByRole('button', { name: '确认' }).click();
}
}
```
```typescript
// e2e/specs/crud/user-management.spec.ts
import { test, expect } from '@playwright/test';
import { UserListPage } from '../../pages/user-list.page';
test.describe('用户管理 CRUD', () => {
let userList: UserListPage;
test.beforeEach(async ({ page }) => {
userList = new UserListPage(page);
await userList.goto();
});
test('创建 → 编辑 → 删除完整流程', async ({ page }) => {
// 创建
await userList.clickAddUser();
await page.getByLabel('姓名').fill('测试用户');
await page.getByLabel('邮箱').fill('test@example.com');
await page.getByRole('button', { name: '提交' }).click();
await expect(page.getB|
系统架构设计方法论,包含架构模式选择、系统分层、目录结构设计
数据模型和API设计方法论,包含ERD设计、数据字典、RESTful API规范
技术调研方法论,通过系统性调研和对比分析,为技术选型提供数据支持
后端API开发方法论,包括RESTful/GraphQL设计、请求验证、错误处理和安全实现
后端测试编写指南,包括单元测试、集成测试和E2E测试的编写方法和最佳实践
|
自动生成 CHANGELOG,基于 git 提交历史和 pipeline 产物信息,遵循 Conventional Commits 和 Keep a Changelog 规范