3.6 KiB
Code Reuse Thinking Guide
Purpose: Stop and think before creating new code - does it already exist?
The Problem
Duplicated code is the #1 source of inconsistency bugs.
When you copy-paste or rewrite existing logic:
- Bug fixes don't propagate
- Behavior diverges over time
- Codebase becomes harder to understand
Before Writing New Code
Step 1: Search First
# Search for similar function names
grep -r "functionName" .
# Search for similar logic
grep -r "keyword" .
Step 2: Ask These Questions
| Question | If Yes... |
|---|---|
| Does a similar function exist? | Use or extend it |
| Is this pattern used elsewhere? | Follow the existing pattern |
| Could this be a shared utility? | Create it in the right place |
| Am I copying code from another file? | STOP - extract to shared |
Common Duplication Patterns
Pattern 1: Copy-Paste Functions
Bad: Copying a validation function to another file
Good: Extract to shared utilities, import where needed
Pattern 2: Similar Components
Bad: Creating a new component that's 80% similar to existing
Good: Extend existing component with props/variants
Pattern 3: Repeated Constants
Bad: Defining the same constant in multiple files
Good: Single source of truth, import everywhere
When to Abstract
Abstract when:
- Same code appears 3+ times
- Logic is complex enough to have bugs
- Multiple people might need this
Don't abstract when:
- Only used once
- Trivial one-liner
- Abstraction would be more complex than duplication
After Batch Modifications
When you've made similar changes to multiple files:
- Review: Did you catch all instances?
- Search: Run grep to find any missed
- Consider: Should this be abstracted?
Gotcha: Asymmetric Mechanisms Producing Same Output
Problem: When two different mechanisms must produce the same file set (e.g., recursive directory copy for init vs. manual files.set() for update), structural changes (renaming, moving, adding subdirectories) only propagate through the automatic mechanism. The manual one silently drifts.
Symptom: Init works perfectly, but update creates files at wrong paths or misses files entirely.
Prevention checklist:
- When migrating directory structures, search for ALL code paths that reference the old structure
- If one path is auto-derived (glob/copy) and another is manually listed, the manual one needs updating
- Add a regression test that compares outputs from both mechanisms
Gotcha: Rename Without Propagation
Problem: When renaming a function, variable, or component, IDE refactoring tools update all references automatically. But if you manually rename (even with search-and-replace), you may miss some references.
Symptom: Code compiles/runs but behavior is broken because some code still references the old name.
Example: Renaming debouncedMeasure to handleResize but forgetting to update the call site in a ResizeObserver callback.
Prevention checklist:
- Use IDE rename (F2 in most IDEs) - it finds ALL references
- If manual rename: Search for the old name with grep BEFORE renaming to find all occurrences
- If manual rename: Search for the new name AFTER renaming to verify all references were updated
- Test the feature that uses the renamed entity
Checklist Before Commit
- Searched for existing similar code
- No copy-pasted logic that should be shared
- Constants defined in one place
- Similar patterns follow same structure