重要前提
安装AI Skills的关键前提是:必须科学上网,且开启TUN模式,这一点至关重要,直接决定安装能否顺利完成,在此郑重提醒三遍:科学上网,科学上网,科学上网。查看完整安装教程 →
tdd-strict by svenja-dev/claude-code-skills
npx skills add https://github.com/svenja-dev/claude-code-skills --skill tdd-strict此技能基于核心原则强制执行 TDD 实践:
"如果你没有看到测试失败,你就不知道它是否测试了正确的东西。"
// 首先:编写测试
describe('calculateOEE', () => {
it('should return 0 when availability is 0', () => {
const result = calculateOEE({ availability: 0, performance: 100, quality: 100 });
expect(result).toBe(0);
});
});
// 测试必须失败:
// Error: calculateOEE is not defined
// 或
// Error: Expected 0 but received undefined
重要:测试必须因正确的原因失败:
// 然后:最小代码
export function calculateOEE(params: OEEParams): number {
if (params.availability === 0) return 0;
// 更多逻辑稍后通过更多测试添加
return 0;
}
:编写使测试通过的最简单代码。
广告位招租
在这里展示您的产品或服务
触达数万 AI 开发者,精准高效
// 经过多个绿色测试后:允许重构
export function calculateOEE({ availability, performance, quality }: OEEParams): number {
return (availability * performance * quality) / 10000;
}
重构规则:
这些借口绝不可接受:
现实:简单的代码需要简单的测试。1 行测试也可以。
it('should add two numbers', () => {
expect(add(2, 3)).toBe(5);
});
现实:"稍后"意味着"永不"。TDD 意味着测试先行。
现实:手动测试不可重现且无法扩展。
现实:测试在调试时节省时间并防止回归。
现实:通过公共 API 测试行为。如果无法测试:重构。
现实:React Testing Library、Playwright、Storybook 正是为此而存在。
现实:简单的逻辑也会变化。测试记录预期行为。
现实:QA 在后期发现 Bug 且成本更高。TDD 从一开始就防止 Bug。
现实:临时代码常常存活数年。测试也保护临时代码。
现实:TDD 长期通过减少调试和回归来加速开发。
现实:在更改前编写特征化测试。逐步改进。
现实:测试你对框架的使用,而不是框架本身。
现实:如果模拟太复杂,那么设计就太复杂了。重构。
每次提交前必须满足:
should [预期结果] when [条件]skip 或 only 测试# 对于 src/utils/oee.ts 中的新函数
touch src/utils/oee.test.ts
// src/utils/oee.test.ts
import { describe, it, expect } from 'vitest';
import { calculateOEE } from './oee';
describe('calculateOEE', () => {
it('should return 85 for sample manufacturing data', () => {
const result = calculateOEE({
availability: 90,
performance: 95,
quality: 99
});
expect(result).toBeCloseTo(84.645, 2);
});
});
npm run test -- --run src/utils/oee.test.ts
# 预期输出:
# Error: Cannot find module './oee'
# 或存根后:
# Error: Expected 84.645 but received undefined
// src/utils/oee.ts
export interface OEEParams {
availability: number;
performance: number;
quality: number;
}
export function calculateOEE(params: OEEParams): number {
return (params.availability * params.performance * params.quality) / 10000;
}
npm run test -- --run src/utils/oee.test.ts
# 预期输出:
# PASS src/utils/oee.test.ts
it('should throw when values exceed 100', () => {
expect(() => calculateOEE({
availability: 101,
performance: 100,
quality: 100
})).toThrow('Values must be between 0 and 100');
});
返回步骤 3(红)-> 步骤 4(绿)-> 重复。
// 错误:先写代码
function add(a: number, b: number): number {
return a + b;
}
// 然后才写测试 - 无效!
// 你不知道测试是否正确
// 错误:在没有测试的情况下实现完整类
class UserService {
async create(user: User) { ... }
async update(id: string, data: Partial<User>) { ... }
async delete(id: string) { ... }
async findById(id: string) { ... }
async findAll(filters: FilterOptions) { ... }
}
// 正确:使用 TDD 逐个方法实现
// 错误:因为实现不同而修改测试
// 之前:expect(result).toBe(100);
// 之后:expect(result).toBe(99.5); // "因为公式这样输出"
// 正确:修正实现或澄清需求
import { render, screen } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
describe('LoginButton', () => {
it('should show loading spinner when clicked', async () => {
render(<LoginButton onLogin={vi.fn()} />);
await userEvent.click(screen.getByRole('button', { name: /login/i }));
expect(screen.getByRole('progressbar')).toBeInTheDocument();
});
});
import request from 'supertest';
import { app } from '../app';
describe('POST /api/analyze', () => {
it('should return 401 without authentication', async () => {
const response = await request(app)
.post('/api/analyze')
.send({ data: 'test' });
expect(response.status).toBe(401);
expect(response.body.error).toBe('Unauthorized');
});
});
describe('fetchUserData', () => {
it('should retry 3 times on network failure', async () => {
const mockFetch = vi.fn()
.mockRejectedValueOnce(new Error('Network'))
.mockRejectedValueOnce(new Error('Network'))
.mockResolvedValueOnce({ id: 1, name: 'Test' });
const result = await fetchUserData(1, { fetch: mockFetch });
expect(mockFetch).toHaveBeenCalledTimes(3);
expect(result.name).toBe('Test');
});
});
# 运行单个测试(验证红)
npm run test -- --run path/to/file.test.ts
# 所有测试(提交前)
npm run test
# 覆盖率报告
npm run test:coverage
# 监视模式(开发期间)
npm run test -- --watch
每周安装次数
47
仓库
GitHub 星标数
32
首次出现
2026年1月24日
安全审计
安装于
opencode43
claude-code41
codex39
gemini-cli39
cursor36
github-copilot34
Dieser Skill erzwingt TDD-Praktiken basierend auf dem Kernprinzip:
"If you didn't watch the test fail, you don't know if it tests the right thing."
// ZUERST: Test schreiben
describe('calculateOEE', () => {
it('should return 0 when availability is 0', () => {
const result = calculateOEE({ availability: 0, performance: 100, quality: 100 });
expect(result).toBe(0);
});
});
// Test MUSS fehlschlagen:
// Error: calculateOEE is not defined
// ODER
// Error: Expected 0 but received undefined
Wichtig : Der Test MUSS aus dem richtigen Grund fehlschlagen:
// DANACH: Minimaler Code
export function calculateOEE(params: OEEParams): number {
if (params.availability === 0) return 0;
// Weitere Logik kommt spaeter durch weitere Tests
return 0;
}
Regel : Schreibe den EINFACHSTEN Code der den Test besteht.
// Nach mehreren gruenen Tests: Refactoring erlaubt
export function calculateOEE({ availability, performance, quality }: OEEParams): number {
return (availability * performance * quality) / 10000;
}
Regeln fuer Refactoring :
Diese Ausreden sind NIEMALS akzeptabel:
Realitaet : Einfacher Code braucht einfache Tests. 1 Zeile Test ist okay.
it('should add two numbers', () => {
expect(add(2, 3)).toBe(5);
});
Realitaet : "Spaeter" bedeutet "nie". TDD bedeutet Test ZUERST.
Realitaet : Manuelle Tests sind nicht reproduzierbar und skalieren nicht.
Realitaet : Tests sparen Zeit bei Debugging und verhindern Regressionen.
Realitaet : Teste das Verhalten durch Public APIs. Wenn nicht testbar: Refactor.
Realitaet : React Testing Library, Playwright, Storybook existieren genau dafuer.
Realitaet : Triviale Logik aendert sich. Tests dokumentieren erwartetes Verhalten.
Realitaet : QA findet Bugs spaeter und teurer. TDD verhindert Bugs von Anfang an.
Realitaet : Temporaerer Code lebt oft Jahre. Tests sichern auch temporaeren Code ab.
Realitaet : TDD beschleunigt langfristig durch weniger Debugging und Regressionen.
Realitaet : Charakterisierungstests vor Aenderungen schreiben. Schrittweise verbessern.
Realitaet : Teste DEINE Nutzung des Frameworks, nicht das Framework selbst.
Realitaet : Wenn Mocking zu komplex ist, ist das Design zu komplex. Refactor.
Vor jedem Commit MUSS gelten:
should [erwartetes Ergebnis] when [Bedingung]skip oder only Tests im Commit# Fuer neue Funktion in src/utils/oee.ts
touch src/utils/oee.test.ts
// src/utils/oee.test.ts
import { describe, it, expect } from 'vitest';
import { calculateOEE } from './oee';
describe('calculateOEE', () => {
it('should return 85 for sample manufacturing data', () => {
const result = calculateOEE({
availability: 90,
performance: 95,
quality: 99
});
expect(result).toBeCloseTo(84.645, 2);
});
});
npm run test -- --run src/utils/oee.test.ts
# Erwartete Ausgabe:
# Error: Cannot find module './oee'
# ODER nach Stub:
# Error: Expected 84.645 but received undefined
// src/utils/oee.ts
export interface OEEParams {
availability: number;
performance: number;
quality: number;
}
export function calculateOEE(params: OEEParams): number {
return (params.availability * params.performance * params.quality) / 10000;
}
npm run test -- --run src/utils/oee.test.ts
# Erwartete Ausgabe:
# PASS src/utils/oee.test.ts
it('should throw when values exceed 100', () => {
expect(() => calculateOEE({
availability: 101,
performance: 100,
quality: 100
})).toThrow('Values must be between 0 and 100');
});
Zurueck zu Schritt 3 (RED) -> Schritt 4 (GREEN) -> Repeat.
// FALSCH: Code zuerst geschrieben
function add(a: number, b: number): number {
return a + b;
}
// Dann erst Test geschrieben - UNGUELTIG!
// Du weisst nicht ob der Test das richtige testet
// FALSCH: Komplette Klasse ohne Tests implementiert
class UserService {
async create(user: User) { ... }
async update(id: string, data: Partial<User>) { ... }
async delete(id: string) { ... }
async findById(id: string) { ... }
async findAll(filters: FilterOptions) { ... }
}
// RICHTIG: Eine Methode nach der anderen mit TDD
// FALSCH: Test geaendert weil Implementation anders ist
// Vorher: expect(result).toBe(100);
// Nachher: expect(result).toBe(99.5); // "Weil die Formel das so ausgibt"
// RICHTIG: Implementation korrigieren ODER Anforderungen klaeren
import { render, screen } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
describe('LoginButton', () => {
it('should show loading spinner when clicked', async () => {
render(<LoginButton onLogin={vi.fn()} />);
await userEvent.click(screen.getByRole('button', { name: /login/i }));
expect(screen.getByRole('progressbar')).toBeInTheDocument();
});
});
import request from 'supertest';
import { app } from '../app';
describe('POST /api/analyze', () => {
it('should return 401 without authentication', async () => {
const response = await request(app)
.post('/api/analyze')
.send({ data: 'test' });
expect(response.status).toBe(401);
expect(response.body.error).toBe('Unauthorized');
});
});
describe('fetchUserData', () => {
it('should retry 3 times on network failure', async () => {
const mockFetch = vi.fn()
.mockRejectedValueOnce(new Error('Network'))
.mockRejectedValueOnce(new Error('Network'))
.mockResolvedValueOnce({ id: 1, name: 'Test' });
const result = await fetchUserData(1, { fetch: mockFetch });
expect(mockFetch).toHaveBeenCalledTimes(3);
expect(result.name).toBe('Test');
});
});
# Einzelnen Test laufen lassen (RED pruefen)
npm run test -- --run path/to/file.test.ts
# Alle Tests (vor Commit)
npm run test
# Coverage Report
npm run test:coverage
# Watch Mode (waehrend Entwicklung)
npm run test -- --watch
Weekly Installs
47
Repository
GitHub Stars
32
First Seen
Jan 24, 2026
Security Audits
Gen Agent Trust HubPassSocketPassSnykPass
Installed on
opencode43
claude-code41
codex39
gemini-cli39
cursor36
github-copilot34
Vue 3 调试指南:解决响应式、计算属性与监听器常见错误
12,200 周安装