claude-mpm 5.4.59__py3-none-any.whl → 5.4.62__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of claude-mpm might be problematic. Click here for more details.
- claude_mpm/VERSION +1 -1
- claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +5 -0
- claude_mpm/scripts/start_activity_logging.py +0 -0
- claude_mpm/services/agents/deployment/agent_template_builder.py +8 -0
- claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
- claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
- claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
- claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
- claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
- claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
- claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
- claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
- claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
- claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
- claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
- claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
- claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
- claude_mpm/skills/bundled/pm/pm-delegation-patterns/SKILL.md +167 -0
- claude_mpm/skills/bundled/pm/pm-git-file-tracking/SKILL.md +113 -0
- claude_mpm/skills/bundled/pm/pm-pr-workflow/SKILL.md +124 -0
- claude_mpm/skills/bundled/pm/pm-ticketing-integration/SKILL.md +154 -0
- claude_mpm/skills/bundled/pm/pm-verification-protocols/SKILL.md +198 -0
- claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
- claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
- claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
- claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
- claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
- claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
- claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
- claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
- claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
- claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
- claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
- claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
- claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
- claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
- claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
- claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
- claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
- claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
- claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
- claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
- {claude_mpm-5.4.59.dist-info → claude_mpm-5.4.62.dist-info}/METADATA +1 -1
- {claude_mpm-5.4.59.dist-info → claude_mpm-5.4.62.dist-info}/RECORD +127 -10
- {claude_mpm-5.4.59.dist-info → claude_mpm-5.4.62.dist-info}/WHEEL +0 -0
- {claude_mpm-5.4.59.dist-info → claude_mpm-5.4.62.dist-info}/entry_points.txt +0 -0
- {claude_mpm-5.4.59.dist-info → claude_mpm-5.4.62.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-5.4.59.dist-info → claude_mpm-5.4.62.dist-info}/licenses/LICENSE-FAQ.md +0 -0
- {claude_mpm-5.4.59.dist-info → claude_mpm-5.4.62.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,543 @@
|
|
|
1
|
+
# TDD Anti-Patterns
|
|
2
|
+
|
|
3
|
+
> **Part of**: [Test-Driven Development](../SKILL.md)
|
|
4
|
+
> **Category**: testing
|
|
5
|
+
> **Reading Level**: Intermediate
|
|
6
|
+
|
|
7
|
+
## Purpose
|
|
8
|
+
|
|
9
|
+
Common TDD mistakes, rationalizations, and red flags to avoid. Learn to recognize when you're violating test-driven development principles.
|
|
10
|
+
|
|
11
|
+
## Red Flags - Recognize and Stop
|
|
12
|
+
|
|
13
|
+
### "I'll Write Tests After to Verify It Works"
|
|
14
|
+
|
|
15
|
+
**What It Sounds Like:**
|
|
16
|
+
- "Let me confirm this works first, then I'll test"
|
|
17
|
+
- "I'll add tests once the feature is done"
|
|
18
|
+
- "Tests are part of cleanup, I'll do them later"
|
|
19
|
+
|
|
20
|
+
**Why It's Wrong:**
|
|
21
|
+
- Tests passing immediately prove nothing
|
|
22
|
+
- Didn't watch test fail = don't know if test works
|
|
23
|
+
- Implementation may be hard to test (too late to redesign)
|
|
24
|
+
- "Later" often becomes "never"
|
|
25
|
+
|
|
26
|
+
**Reality Check:**
|
|
27
|
+
```typescript
|
|
28
|
+
// Test-after: Passes immediately
|
|
29
|
+
test('validates email', () => {
|
|
30
|
+
expect(validateEmail('test@example.com')).toBe(true);
|
|
31
|
+
});
|
|
32
|
+
// ✓ PASS
|
|
33
|
+
|
|
34
|
+
// But you never saw it fail!
|
|
35
|
+
// Does it actually test anything? Unknown.
|
|
36
|
+
```
|
|
37
|
+
|
|
38
|
+
**What to Do Instead:**
|
|
39
|
+
```typescript
|
|
40
|
+
// Test-first: Watch it fail
|
|
41
|
+
test('validates email', () => {
|
|
42
|
+
expect(validateEmail('test@example.com')).toBe(true);
|
|
43
|
+
});
|
|
44
|
+
// ✗ FAIL: validateEmail is not defined
|
|
45
|
+
|
|
46
|
+
// Implement
|
|
47
|
+
function validateEmail(email: string): boolean {
|
|
48
|
+
return email.includes('@');
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// ✓ PASS: Now you know test works
|
|
52
|
+
```
|
|
53
|
+
|
|
54
|
+
### "Keep as Reference, Write Tests First"
|
|
55
|
+
|
|
56
|
+
**What It Sounds Like:**
|
|
57
|
+
- "I'll keep this code as a reference while writing tests"
|
|
58
|
+
- "I'll adapt this implementation while test-driving"
|
|
59
|
+
- "Let me look at what I built to write better tests"
|
|
60
|
+
|
|
61
|
+
**Why It's Wrong:**
|
|
62
|
+
- You WILL adapt existing code (human nature)
|
|
63
|
+
- "Adapt" = testing after, not testing first
|
|
64
|
+
- Reference biases your test design
|
|
65
|
+
- Defeats purpose of TDD
|
|
66
|
+
|
|
67
|
+
**Reality Check:**
|
|
68
|
+
```typescript
|
|
69
|
+
// You wrote this "reference":
|
|
70
|
+
function processUser(user: any) {
|
|
71
|
+
const valid = user.email && user.age >= 18 && user.name;
|
|
72
|
+
if (!valid) throw new Error('Invalid');
|
|
73
|
+
return { id: generateId(), ...user };
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
// Now you write "test-first" while looking at it:
|
|
77
|
+
test('processes valid user', () => {
|
|
78
|
+
const user = { email: 'test@example.com', age: 25, name: 'Test' };
|
|
79
|
+
const result = processUser(user);
|
|
80
|
+
expect(result.id).toBeDefined();
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
// You just tested what you built, not what it should do
|
|
84
|
+
```
|
|
85
|
+
|
|
86
|
+
**What to Do Instead:**
|
|
87
|
+
```
|
|
88
|
+
Delete the code. Seriously. Delete it.
|
|
89
|
+
|
|
90
|
+
Then write test describing what SHOULD happen:
|
|
91
|
+
test('creates user with required fields', () => {
|
|
92
|
+
const user = createUser({
|
|
93
|
+
email: 'test@example.com',
|
|
94
|
+
age: 25,
|
|
95
|
+
name: 'Test User'
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
expect(user).toMatchObject({
|
|
99
|
+
id: expect.any(String),
|
|
100
|
+
email: 'test@example.com',
|
|
101
|
+
age: 25,
|
|
102
|
+
name: 'Test User'
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
Now implement fresh from test.
|
|
107
|
+
```
|
|
108
|
+
|
|
109
|
+
### "Already Spent X Hours, Deleting Is Wasteful"
|
|
110
|
+
|
|
111
|
+
**What It Sounds Like:**
|
|
112
|
+
- "I can't throw away 4 hours of work"
|
|
113
|
+
- "Let me just add tests to verify what I built"
|
|
114
|
+
- "Rewriting would take too long"
|
|
115
|
+
|
|
116
|
+
**Why It's Wrong:**
|
|
117
|
+
- Sunk cost fallacy
|
|
118
|
+
- Time already gone, can't recover it
|
|
119
|
+
- Keeping unverified code = technical debt
|
|
120
|
+
- Will spend more time debugging
|
|
121
|
+
|
|
122
|
+
**Reality Check:**
|
|
123
|
+
```
|
|
124
|
+
Time spent: 4 hours
|
|
125
|
+
Code quality: Unknown
|
|
126
|
+
Test coverage: None
|
|
127
|
+
Technical debt: High
|
|
128
|
+
|
|
129
|
+
Option A: Keep it and test after
|
|
130
|
+
Time: 4 hours (sunk) + 30 min (weak tests) + 120 min (future debugging)
|
|
131
|
+
Total: 6.5 hours
|
|
132
|
+
Quality: Low
|
|
133
|
+
|
|
134
|
+
Option B: Delete and TDD
|
|
135
|
+
Time: 4 hours (sunk) + 2 hours (TDD rewrite)
|
|
136
|
+
Total: 6 hours
|
|
137
|
+
Quality: High
|
|
138
|
+
|
|
139
|
+
Option B is objectively better.
|
|
140
|
+
```
|
|
141
|
+
|
|
142
|
+
**What to Do Instead:**
|
|
143
|
+
- Accept sunk cost
|
|
144
|
+
- Treat it as learning time
|
|
145
|
+
- Delete code
|
|
146
|
+
- Rewrite with TDD (will be faster, you know the problem now)
|
|
147
|
+
|
|
148
|
+
### "Tests After Achieve the Same Purpose"
|
|
149
|
+
|
|
150
|
+
**What It Sounds Like:**
|
|
151
|
+
- "I'll have 100% coverage either way"
|
|
152
|
+
- "Tests-after verify correctness just as well"
|
|
153
|
+
- "It's about the spirit, not the ritual"
|
|
154
|
+
- "Being pragmatic means adapting"
|
|
155
|
+
|
|
156
|
+
**Why It's Wrong:**
|
|
157
|
+
- Coverage ≠ quality
|
|
158
|
+
- Tests-after verify what you remembered
|
|
159
|
+
- Can't achieve TDD spirit without TDD practice
|
|
160
|
+
- Tests-after is optimistic gambling, not pragmatic
|
|
161
|
+
|
|
162
|
+
**Reality Check:**
|
|
163
|
+
```typescript
|
|
164
|
+
// Test-after: 100% coverage, useless
|
|
165
|
+
function add(a: number, b: number): number {
|
|
166
|
+
return a - b; // BUG
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
test('add function', () => {
|
|
170
|
+
add(2, 3); // 100% coverage!
|
|
171
|
+
// But doesn't verify result
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
// Test-first: Forces verification
|
|
175
|
+
test('adds two numbers', () => {
|
|
176
|
+
expect(add(2, 3)).toBe(5); // Must verify behavior
|
|
177
|
+
});
|
|
178
|
+
```
|
|
179
|
+
|
|
180
|
+
See [Philosophy](philosophy.md) for detailed explanation.
|
|
181
|
+
|
|
182
|
+
### "This Is Different Because..."
|
|
183
|
+
|
|
184
|
+
**What It Sounds Like:**
|
|
185
|
+
- "This is too simple for TDD"
|
|
186
|
+
- "This is too complex for TDD"
|
|
187
|
+
- "This is exploratory code"
|
|
188
|
+
- "This is just a quick bug fix"
|
|
189
|
+
- "This is legacy code (can't test)"
|
|
190
|
+
|
|
191
|
+
**Why It's Wrong:**
|
|
192
|
+
- Every situation has an excuse
|
|
193
|
+
- All excuses are invalid
|
|
194
|
+
- "Just this once" becomes "always"
|
|
195
|
+
|
|
196
|
+
**Reality Check:**
|
|
197
|
+
|
|
198
|
+
| Excuse | Truth |
|
|
199
|
+
|--------|-------|
|
|
200
|
+
| "Too simple" | Simple code breaks too |
|
|
201
|
+
| "Too complex" | Complex code NEEDS tests |
|
|
202
|
+
| "Exploratory" | Exploration teaches what to test |
|
|
203
|
+
| "Quick fix" | Quick fixes need regression protection |
|
|
204
|
+
| "Legacy code" | Characterization tests enable refactoring |
|
|
205
|
+
|
|
206
|
+
**What to Do Instead:**
|
|
207
|
+
- Stop rationalizing
|
|
208
|
+
- Follow TDD process
|
|
209
|
+
- Every. Single. Time.
|
|
210
|
+
|
|
211
|
+
## Common Anti-Patterns
|
|
212
|
+
|
|
213
|
+
### Pattern: Testing Implementation, Not Behavior
|
|
214
|
+
|
|
215
|
+
**Bad:**
|
|
216
|
+
```typescript
|
|
217
|
+
test('uses regex to validate email', () => {
|
|
218
|
+
const validator = new EmailValidator();
|
|
219
|
+
expect(validator.pattern).toBe(/^[^\s@]+@[^\s@]+\.[^\s@]+$/);
|
|
220
|
+
});
|
|
221
|
+
```
|
|
222
|
+
|
|
223
|
+
**Why Wrong:** Tests HOW, not WHAT. Prevents refactoring.
|
|
224
|
+
|
|
225
|
+
**Good:**
|
|
226
|
+
```typescript
|
|
227
|
+
test('accepts valid email format', () => {
|
|
228
|
+
expect(validateEmail('user@example.com')).toBe(true);
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
test('rejects email without @ symbol', () => {
|
|
232
|
+
expect(validateEmail('userexample.com')).toBe(false);
|
|
233
|
+
});
|
|
234
|
+
```
|
|
235
|
+
|
|
236
|
+
**Why Right:** Tests behavior. Implementation can change freely.
|
|
237
|
+
|
|
238
|
+
### Pattern: Mocking Everything
|
|
239
|
+
|
|
240
|
+
**Bad:**
|
|
241
|
+
```typescript
|
|
242
|
+
test('processes user data', () => {
|
|
243
|
+
const mockValidator = jest.fn().mockReturnValue(true);
|
|
244
|
+
const mockRepository = jest.fn().mockResolvedValue({ id: 1 });
|
|
245
|
+
const mockLogger = jest.fn();
|
|
246
|
+
|
|
247
|
+
const service = new UserService(mockValidator, mockRepository, mockLogger);
|
|
248
|
+
|
|
249
|
+
service.createUser({ name: 'Test' });
|
|
250
|
+
|
|
251
|
+
expect(mockValidator).toHaveBeenCalled();
|
|
252
|
+
expect(mockRepository).toHaveBeenCalled();
|
|
253
|
+
expect(mockLogger).toHaveBeenCalled();
|
|
254
|
+
});
|
|
255
|
+
```
|
|
256
|
+
|
|
257
|
+
**Why Wrong:**
|
|
258
|
+
- Tests mock interactions, not real code
|
|
259
|
+
- Doesn't verify actual behavior
|
|
260
|
+
- Brittle (breaks on implementation changes)
|
|
261
|
+
|
|
262
|
+
**Good:**
|
|
263
|
+
```typescript
|
|
264
|
+
test('creates user with valid data', async () => {
|
|
265
|
+
const service = new UserService();
|
|
266
|
+
|
|
267
|
+
const user = await service.createUser({
|
|
268
|
+
name: 'Test User',
|
|
269
|
+
email: 'test@example.com'
|
|
270
|
+
});
|
|
271
|
+
|
|
272
|
+
expect(user.id).toBeDefined();
|
|
273
|
+
expect(user.name).toBe('Test User');
|
|
274
|
+
expect(user.email).toBe('test@example.com');
|
|
275
|
+
});
|
|
276
|
+
```
|
|
277
|
+
|
|
278
|
+
**Why Right:** Tests real behavior using real objects.
|
|
279
|
+
|
|
280
|
+
**When to Mock:** Only external dependencies you don't control (APIs, databases, file system).
|
|
281
|
+
|
|
282
|
+
### Pattern: One Giant Test
|
|
283
|
+
|
|
284
|
+
**Bad:**
|
|
285
|
+
```typescript
|
|
286
|
+
test('user registration flow', async () => {
|
|
287
|
+
// Test validates email
|
|
288
|
+
expect(validateEmail('test@example.com')).toBe(true);
|
|
289
|
+
|
|
290
|
+
// Test validates password
|
|
291
|
+
expect(validatePassword('Pass123')).toBe(true);
|
|
292
|
+
|
|
293
|
+
// Test creates user
|
|
294
|
+
const user = await createUser({...});
|
|
295
|
+
expect(user.id).toBeDefined();
|
|
296
|
+
|
|
297
|
+
// Test sends welcome email
|
|
298
|
+
expect(emailsSent).toContain('welcome');
|
|
299
|
+
|
|
300
|
+
// Test logs activity
|
|
301
|
+
expect(logs).toContain('user_created');
|
|
302
|
+
});
|
|
303
|
+
```
|
|
304
|
+
|
|
305
|
+
**Why Wrong:**
|
|
306
|
+
- Tests multiple behaviors
|
|
307
|
+
- If fails, unclear what broke
|
|
308
|
+
- Hard to understand what's being tested
|
|
309
|
+
- Violates "one test, one behavior"
|
|
310
|
+
|
|
311
|
+
**Good:**
|
|
312
|
+
```typescript
|
|
313
|
+
test('validates email format', () => {
|
|
314
|
+
expect(validateEmail('test@example.com')).toBe(true);
|
|
315
|
+
});
|
|
316
|
+
|
|
317
|
+
test('validates password strength', () => {
|
|
318
|
+
expect(validatePassword('Pass123')).toBe(true);
|
|
319
|
+
});
|
|
320
|
+
|
|
321
|
+
test('creates user with valid data', async () => {
|
|
322
|
+
const user = await createUser({...});
|
|
323
|
+
expect(user.id).toBeDefined();
|
|
324
|
+
});
|
|
325
|
+
|
|
326
|
+
test('sends welcome email after registration', async () => {
|
|
327
|
+
await createUser({...});
|
|
328
|
+
expect(emailsSent).toContain('welcome');
|
|
329
|
+
});
|
|
330
|
+
```
|
|
331
|
+
|
|
332
|
+
**Why Right:** Each test has single, clear purpose.
|
|
333
|
+
|
|
334
|
+
### Pattern: Vague Assertions
|
|
335
|
+
|
|
336
|
+
**Bad:**
|
|
337
|
+
```typescript
|
|
338
|
+
test('processes array', () => {
|
|
339
|
+
const result = processArray([1, 2, 3]);
|
|
340
|
+
expect(result).toBeDefined();
|
|
341
|
+
expect(result.length).toBeGreaterThan(0);
|
|
342
|
+
});
|
|
343
|
+
```
|
|
344
|
+
|
|
345
|
+
**Why Wrong:**
|
|
346
|
+
- Doesn't verify actual behavior
|
|
347
|
+
- Many wrong implementations would pass
|
|
348
|
+
- Provides false confidence
|
|
349
|
+
|
|
350
|
+
**Good:**
|
|
351
|
+
```typescript
|
|
352
|
+
test('doubles each element in array', () => {
|
|
353
|
+
const result = processArray([1, 2, 3]);
|
|
354
|
+
expect(result).toEqual([2, 4, 6]);
|
|
355
|
+
});
|
|
356
|
+
```
|
|
357
|
+
|
|
358
|
+
**Why Right:** Precise assertion. Only correct implementation passes.
|
|
359
|
+
|
|
360
|
+
### Pattern: Skipping Verification Steps
|
|
361
|
+
|
|
362
|
+
**Bad:**
|
|
363
|
+
```typescript
|
|
364
|
+
// Write test
|
|
365
|
+
test('retries on failure', async () => {
|
|
366
|
+
const result = await withRetry(operation);
|
|
367
|
+
expect(result).toBe('success');
|
|
368
|
+
});
|
|
369
|
+
|
|
370
|
+
// Implement immediately without running test
|
|
371
|
+
async function withRetry(fn) {
|
|
372
|
+
try {
|
|
373
|
+
return await fn();
|
|
374
|
+
} catch (e) {
|
|
375
|
+
return await fn();
|
|
376
|
+
}
|
|
377
|
+
}
|
|
378
|
+
|
|
379
|
+
// Run both tests together
|
|
380
|
+
// PASS: Both pass
|
|
381
|
+
|
|
382
|
+
// You never saw test fail!
|
|
383
|
+
```
|
|
384
|
+
|
|
385
|
+
**Why Wrong:**
|
|
386
|
+
- Don't know if test actually tests anything
|
|
387
|
+
- Implementation might have already existed
|
|
388
|
+
- Test might be broken
|
|
389
|
+
|
|
390
|
+
**Good:**
|
|
391
|
+
```typescript
|
|
392
|
+
// Write test
|
|
393
|
+
test('retries on failure', async () => {
|
|
394
|
+
const result = await withRetry(operation);
|
|
395
|
+
expect(result).toBe('success');
|
|
396
|
+
});
|
|
397
|
+
|
|
398
|
+
// MANDATORY: Run test FIRST
|
|
399
|
+
// FAIL: withRetry is not defined
|
|
400
|
+
// ✓ Good - test works
|
|
401
|
+
|
|
402
|
+
// Implement
|
|
403
|
+
async function withRetry(fn) {
|
|
404
|
+
try {
|
|
405
|
+
return await fn();
|
|
406
|
+
} catch (e) {
|
|
407
|
+
return await fn();
|
|
408
|
+
}
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
// Run test AGAIN
|
|
412
|
+
// PASS
|
|
413
|
+
// ✓ Good - implementation works
|
|
414
|
+
```
|
|
415
|
+
|
|
416
|
+
**Why Right:** Verified test fails, verified test passes. Confidence high.
|
|
417
|
+
|
|
418
|
+
## Rationalization Detection
|
|
419
|
+
|
|
420
|
+
### Self-Assessment Questions
|
|
421
|
+
|
|
422
|
+
Ask yourself:
|
|
423
|
+
|
|
424
|
+
**Before skipping RED phase:**
|
|
425
|
+
- [ ] Am I making excuses?
|
|
426
|
+
- [ ] Do I think "just this once"?
|
|
427
|
+
- [ ] Am I under time pressure? (More reason for TDD!)
|
|
428
|
+
- [ ] Do I feel the process is "too slow"? (It's faster overall)
|
|
429
|
+
|
|
430
|
+
**After writing implementation first:**
|
|
431
|
+
- [ ] Did I write code before test?
|
|
432
|
+
- [ ] Am I now writing test to verify existing code?
|
|
433
|
+
- [ ] Will my test pass immediately?
|
|
434
|
+
- [ ] Did I watch the test fail first?
|
|
435
|
+
|
|
436
|
+
**When reviewing tests:**
|
|
437
|
+
- [ ] Did I watch each test fail before implementing?
|
|
438
|
+
- [ ] Does each test verify behavior, not implementation?
|
|
439
|
+
- [ ] Could I refactor without breaking tests?
|
|
440
|
+
- [ ] Do tests use real code (not excessive mocks)?
|
|
441
|
+
|
|
442
|
+
### If Any Answer Is "No"
|
|
443
|
+
|
|
444
|
+
**STOP. You're not doing TDD.**
|
|
445
|
+
|
|
446
|
+
Options:
|
|
447
|
+
1. Delete implementation, start with test
|
|
448
|
+
2. Admit you're not doing TDD, accept consequences
|
|
449
|
+
3. Ask for help understanding how to TDD this
|
|
450
|
+
|
|
451
|
+
## Recovery from Anti-Patterns
|
|
452
|
+
|
|
453
|
+
### If You Wrote Code First
|
|
454
|
+
|
|
455
|
+
**Option 1: Characterization Tests (if code works)**
|
|
456
|
+
```typescript
|
|
457
|
+
// Write tests for current behavior
|
|
458
|
+
test('current behavior for case A', () => {
|
|
459
|
+
expect(existingFunction(inputA)).toBe(currentOutputA);
|
|
460
|
+
});
|
|
461
|
+
|
|
462
|
+
// Then refactor with test protection
|
|
463
|
+
// Then delete and rewrite with TDD for new features
|
|
464
|
+
```
|
|
465
|
+
|
|
466
|
+
**Option 2: Delete and Restart (if code not verified)**
|
|
467
|
+
```typescript
|
|
468
|
+
// Delete implementation
|
|
469
|
+
// Write failing test
|
|
470
|
+
// Implement from test
|
|
471
|
+
```
|
|
472
|
+
|
|
473
|
+
### If Tests Are Too Broad
|
|
474
|
+
|
|
475
|
+
**Refactor tests:**
|
|
476
|
+
```typescript
|
|
477
|
+
// Before: One giant test
|
|
478
|
+
test('user registration', () => {
|
|
479
|
+
// 50 lines of test code
|
|
480
|
+
});
|
|
481
|
+
|
|
482
|
+
// After: Multiple focused tests
|
|
483
|
+
describe('User Registration', () => {
|
|
484
|
+
test('validates email format', () => {...});
|
|
485
|
+
test('validates password strength', () => {...});
|
|
486
|
+
test('creates user record', () => {...});
|
|
487
|
+
test('sends welcome email', () => {...});
|
|
488
|
+
});
|
|
489
|
+
```
|
|
490
|
+
|
|
491
|
+
### If Tests Are Brittle
|
|
492
|
+
|
|
493
|
+
**Refactor to test behavior:**
|
|
494
|
+
```typescript
|
|
495
|
+
// Before: Tests implementation
|
|
496
|
+
test('uses bcrypt with 10 rounds', () => {
|
|
497
|
+
expect(hasher.algorithm).toBe('bcrypt');
|
|
498
|
+
expect(hasher.rounds).toBe(10);
|
|
499
|
+
});
|
|
500
|
+
|
|
501
|
+
// After: Tests behavior
|
|
502
|
+
test('hashes password securely', () => {
|
|
503
|
+
const hash1 = hashPassword('password');
|
|
504
|
+
const hash2 = hashPassword('password');
|
|
505
|
+
|
|
506
|
+
expect(hash1).not.toBe('password'); // Actually hashed
|
|
507
|
+
expect(hash1).not.toBe(hash2); // Salted
|
|
508
|
+
expect(verifyPassword('password', hash1)).toBe(true); // Verifiable
|
|
509
|
+
});
|
|
510
|
+
```
|
|
511
|
+
|
|
512
|
+
## Summary
|
|
513
|
+
|
|
514
|
+
**Red Flags:**
|
|
515
|
+
- Code before test
|
|
516
|
+
- Test passes immediately
|
|
517
|
+
- "I'll test after"
|
|
518
|
+
- "Keep as reference"
|
|
519
|
+
- "Deleting is wasteful"
|
|
520
|
+
- "Tests-after are equivalent"
|
|
521
|
+
- "This is different because..."
|
|
522
|
+
|
|
523
|
+
**Anti-Patterns:**
|
|
524
|
+
- Testing implementation not behavior
|
|
525
|
+
- Mocking everything
|
|
526
|
+
- One giant test
|
|
527
|
+
- Vague assertions
|
|
528
|
+
- Skipping verification steps
|
|
529
|
+
|
|
530
|
+
**Recovery:**
|
|
531
|
+
- Recognize the anti-pattern
|
|
532
|
+
- STOP immediately
|
|
533
|
+
- Delete or refactor
|
|
534
|
+
- Restart with proper TDD
|
|
535
|
+
|
|
536
|
+
**Remember:** Proper TDD requires discipline but saves time. Shortcuts feel faster but waste time.
|
|
537
|
+
|
|
538
|
+
## Related References
|
|
539
|
+
|
|
540
|
+
- [Workflow](workflow.md): Correct TDD process
|
|
541
|
+
- [Examples](examples.md): Real-world TDD practice
|
|
542
|
+
- [Philosophy](philosophy.md): Why TDD works
|
|
543
|
+
- [Integration](integration.md): TDD with other skills
|