Testing Guide
编写高质量测试的约束规范,确保测试真正验证行为而非仅仅覆盖代码。
Language
- Accept questions in both Chinese and English
- Always respond in Chinese
Trigger Conditions
- 用户正在编写测试代码
- 用户需要测试策略指导
- 用户提到测试覆盖率、断言、变异测试
- 用户要求分析代码分支覆盖情况
- Code Review 中涉及测试代码
- 关键词:"分支分析"、"branch analysis"、"未测试路径"、"覆盖盲区"
核心理念
测试的目的不是"覆盖代码",而是"验证行为"。
测试依据来自需求,不是来自代码。
测试质量金字塔
Level 3: 测试有效性 ─ 变异得分≥80%, 需求追溯100%
Level 2: 断言质量 ─ 禁止弱断言, 验证行为非实现
Level 1: 代码覆盖 ─ 行/分支覆盖≥80% (必要非充分)
详细说明见 templates/core-concepts.md
Constraints
单元测试约束
详细指南见 templates/unit-testing.md
覆盖率约束
变异测试约束(推荐)
配置见 templates/mutation-testing.md
集成测试约束
详细指南见 templates/integration-testing.md
E2E 测试约束
详细指南见 templates/e2e-testing.md
需求追溯约束
模板见 templates/traceability-matrix.md
测试骨架生成
AI 驱动的自顶向下开发:先生成测试骨架,后填充实现。
标注规范
测试代码必须包含追溯标注,用于 /devdocs-sync 扫描:
/**
* @verifies AC-XXX - 验收标准描述
* @testcase UT/IT/E2E-XXX
*/
test('测试名称', () => {
// 测试代码
});
| 标注 | 用途 | 必须性 |
|---|
@verifies AC-XXX | 关联验收标准 | 必须 |
@testcase UT/IT/E2E-XXX | 测试用例编号 | 必须 |
骨架生成流程
03-test-cases.md (测试用例设计)
│
▼
生成测试骨架
├── describe 结构(按功能点分组)
├── test.skip() 占位(每个测试用例)
├── @verifies/@testcase 标注
└── // TODO: 实现测试 注释
│
▼
逐个实现测试
├── 移除 skip
├── 编写 AAA 结构
└── 添加具体断言
骨架示例
// tests/user.service.test.ts
describe('UserService', () => {
describe('createUser', () => {
/**
* @verifies AC-001 - 邮箱格式校验
* @testcase UT-001
*/
test.skip('应该拒绝无效邮箱格式', () => {
// TODO: 实现测试
// Arrange: 准备无效邮箱
// Act: 调用 createUser
// Assert: 验证抛出 ValidationError
});
/**
* @verifies AC-002 - 密码强度校验
* @testcase UT-002
*/
test.skip('应该拒绝弱密码', () => {
// TODO: 实现测试
});
/**
* @verifies AC-003 - 用户名唯一性
* @testcase UT-003
*/
test.skip('应该拒绝重复用户名', () => {
// TODO: 实现测试
});
});
});
骨架生成约束
与 DevDocs 协作
| 阶段 | Skill | 输入 | 输出 |
|---|
| 测试设计 | /devdocs-test-cases | 需求文档 | 测试用例矩阵 |
| 骨架生成 | /devdocs-dev-tasks | 测试用例 | 测试骨架代码 |
| 测试实现 | /testing-guide | 骨架代码 | 完整测试 |
| 追溯同步 | /devdocs-sync | 代码标注 | 更新矩阵 |
代码分支覆盖分析(AI 驱动)
代码实现完成后,AI 分析所有代码分支,找出未被测试覆盖的路径,生成补充测试用例。
这是对需求驱动测试(AC → 测试)的补充,不是替代。
定位
需求驱动测试(AC → 测试) ← 主路径,覆盖业务规则
+
代码分支分析(代码 → 测试) ← 补充路径,覆盖防御性逻辑
=
完整测试覆盖
适用场景
| 场景 | 说明 |
|---|
| 代码包含防御性逻辑 | 参数校验、null 检查、类型保护 |
| 复杂条件分支 | switch/case、多条件组合 |
| 覆盖率工具显示分支不足 | 行覆盖≥80% 但分支覆盖<80% |
| 变异测试发现存活变异体 | 补充测试以杀死变异体 |
分析流程
Step 1: 分析代码分支
├── 条件语句(if/else, switch, 三元, &&/||)
├── 异常处理路径(try/catch, throw)
├── 早返回(guard clause)
└── 循环边界(空集合、单元素、多元素)
│
▼
Step 2: 映射现有测试
├── 匹配 @verifies 标注的 AC 覆盖范围
├── 分析每个测试实际触发的分支
└── 标记已覆盖/未覆盖分支
│
▼
Step 3: 生成补充测试
├── 为未覆盖分支生成测试用例
├── 使用 BCA-XXX 编号
├── 添加 @covers-branch 标注
└── 遵循 AAA 结构和断言质量约束
标注规范
/**
* @covers-branch createUser:null-email-guard
* @testcase BCA-001
*/
test('createUser 应该抛出错误当 email 为 null', () => {
// Arrange
const dto = { email: null, password: 'Strong1234' };
// Act & Assert
expect(() => createUser(dto)).toThrow('Email is required');
});
| 标注 | 用途 | 必须性 |
|---|
@covers-branch <函数>:<分支描述> | 标记覆盖的代码分支 | 必须 |
@testcase BCA-XXX | 分支补充测试编号 | 必须 |
分支覆盖分析约束
详细分析流程和示例见 templates/branch-coverage-analysis.md
Quick Reference
测试命名
[被测方法] 应该 [预期行为] 当 [条件]
测试结构 (AAA)
Arrange → Act → Assert (具体断言)
禁止的弱断言
// ❌ 禁止
expect(result).toBeDefined();
expect(result).toBeTruthy();
expect(result).not.toBeNull();
// ✅ 要求
expect(result.status).toBe('success');
expect(result.items).toHaveLength(3);
常用命令
# 覆盖率
npm test -- --coverage # Jest
pytest --cov=src # pytest
go test -cover ./... # Go
# 变异测试
npx stryker run # JS/TS
mutmut run # Python
mvn pitest:mutationCoverage # Java
模板索引
核心指南
工具配置
语言最佳实践
与其他 Skills 协作
| 场景 | Skill |
|---|
| 测试用例设计 | /devdocs-test-cases |
| 代码可测试性 | /code-quality |
| 重构前测试 | /refactor |
| 分支覆盖分析 | /devdocs-dev-workflow — 完成检查阶段可选调用 |