f8b21af4be
10_Wiki/Topics 대규모 정리: - 오류 캡처/미완성 stub 문서 227개 제거 - 교차폴더 중복 43클러스터 병합 (63파일 → redirect) - 링크명 정규화: 깨진 링크 수정·redirect 직결·개념 매핑 ~2,400건 - 카테고리 MOC 6개 신규 생성 - Graph 섹션 미해결 related-keyword 링크 10,058건 제거 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
6.2 KiB
6.2 KiB
id, title, category, status, canonical_id, aliases, duplicate_of, source_trust_level, confidence_score, verification_status, tags, raw_sources, last_reinforced, github_commit, tech_stack
| id | title | category | status | canonical_id | aliases | duplicate_of | source_trust_level | confidence_score | verification_status | tags | raw_sources | last_reinforced | github_commit | tech_stack | ||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| wiki-2026-0508-testability-architecture | Testability Architecture | 10_Wiki/Topics | verified | self |
|
none | A | 0.9 | applied |
|
2026-05-10 | pending |
|
Testability Architecture
매 한 줄
"매 testable design 은 testing 의 byproduct 가 아니라 cause". Michael Feathers 의 Working Effectively with Legacy Code 의 seam 개념 + Dependency Inversion 의 결합. 매 unit test 의 어려움 = 매 design problem 의 signal.
매 핵심
매 testability dimensions (Robert Binder 1994)
- Observability: 매 internal state 의 inspection.
- Controllability: 매 input 의 injection.
- Isolability: 매 unit 의 independent execution.
- Predictability: 매 deterministic output.
매 seam types (Feathers)
- Object seam: 매 polymorphism (interface + implementation swap).
- Preprocessor seam: 매 macro / build flag.
- Link seam: 매 link-time symbol 의 replacement.
매 응용
- 매 hard-to-test code = refactor signal — 매 hidden coupling.
- Dependency injection 의 systematic 적용.
- Pure function core + I/O shell (functional core, imperative shell).
💻 패턴
Dependency injection
// Bad: hidden dependency, untestable
class OrderService {
async placeOrder(order: Order) {
const result = await fetch('https://api.stripe.com/charge', { /* ... */ });
await db.query('INSERT INTO orders ...');
}
}
// Good: injected, testable
interface PaymentGateway { charge(amount: number): Promise<ChargeResult>; }
interface OrderRepo { save(order: Order): Promise<void>; }
class OrderService {
constructor(private payments: PaymentGateway, private repo: OrderRepo) {}
async placeOrder(order: Order) {
await this.payments.charge(order.total);
await this.repo.save(order);
}
}
// Test
it('saves order after payment', async () => {
const payments = { charge: vi.fn().mockResolvedValue({ ok: true }) };
const repo = { save: vi.fn() };
const svc = new OrderService(payments, repo);
await svc.placeOrder(makeOrder({ total: 100 }));
expect(repo.save).toHaveBeenCalled();
});
Functional core, imperative shell
// Pure core (easy to test)
export function calculateInvoice(order: Order, taxRules: TaxRule[]): Invoice {
const subtotal = order.items.reduce((s, i) => s + i.price * i.qty, 0);
const tax = applyTaxRules(subtotal, taxRules);
return { subtotal, tax, total: subtotal + tax };
}
// Imperative shell (thin, integration-tested)
export async function generateInvoice(orderId: string) {
const order = await db.orders.findById(orderId);
const rules = await db.taxRules.findActive();
const invoice = calculateInvoice(order, rules); // pure
await db.invoices.save(invoice);
await emailService.send(order.customerId, invoice);
}
Hexagonal port for time
interface Clock { now(): Date; }
class SystemClock implements Clock { now() { return new Date(); } }
class FakeClock implements Clock {
constructor(private fixed: Date) {}
now() { return this.fixed; }
advance(ms: number) { this.fixed = new Date(this.fixed.getTime() + ms); }
}
class TokenService {
constructor(private clock: Clock) {}
isExpired(token: { expiresAt: Date }) {
return this.clock.now() > token.expiresAt;
}
}
Test data builder
class OrderBuilder {
private order: Order = { id: 'o1', items: [], total: 0, status: 'pending' };
withItem(item: Item) { this.order.items.push(item); return this; }
withStatus(s: OrderStatus) { this.order.status = s; return this; }
build() { return { ...this.order }; }
}
// Usage
const order = new OrderBuilder()
.withItem({ sku: 'A', price: 10, qty: 2 })
.withStatus('paid')
.build();
Sprout method (Feathers — adding to legacy)
// Legacy untestable
function processBatch(records: Record[]) {
for (const r of records) {
// ... 200 lines of mess ...
db.update(r);
}
}
// Add new logic via sprout (pure, testable)
export function shouldArchive(record: Record, today: Date): boolean {
return today.getTime() - record.createdAt.getTime() > 365 * 86400_000;
}
function processBatch(records: Record[]) {
const today = new Date();
for (const r of records) {
if (shouldArchive(r, today)) { /* new branch */ }
// ... existing mess unchanged ...
}
}
Humble object pattern
// View has no logic — humble
class CartView {
render(model: CartViewModel) { /* pure rendering */ }
}
// Presenter has all logic — testable without DOM
class CartPresenter {
buildViewModel(cart: Cart): CartViewModel { /* pure */ }
}
매 결정 기준
| 상황 | Approach |
|---|---|
| Side-effect heavy code | Inject dependencies |
| Time-sensitive logic | Clock port |
| Complex object graphs in tests | Test data builder |
| Legacy untouchable | Sprout method/class |
| UI logic | Humble object |
기본값: 매 constructor injection + 매 functional core + 매 ports for I/O.
🔗 Graph
- 부모: Technical-Architecture · Test-Driven_Development
- 변형: Hexagonal Architecture · Clean Architecture
- 응용: Dependency Injection · Test Doubles (테스트 대역)
- Adjacent: Test Automation Pyramid · The Two Hats
🤖 LLM 활용
언제: refactor for testability, seam identification, DI introduction. 언제 X: 매 framework-specific DI container choice — 매 ecosystem convention 의 우선.
❌ 안티패턴
- Mock everything: 매 mock soup — 매 test 가 implementation 의 mirror.
- Static singletons: untestable — 매 global state 의 source.
newin business logic: hidden dependency.- Test private methods: 매 leak — public surface 만 test.
🧪 검증 / 중복
- Verified (Feathers 2004 WELC; Binder 1994 testability metrics).
- 신뢰도 A.
🕓 Changelog
| 날짜 | 변경 |
|---|---|
| 2026-05-08 | Phase 1 |
| 2026-05-10 | Manual cleanup — seams + DI + functional core patterns |