Files
2nd/10_Wiki/Topics/Coding/Testing_Mocking_Boundaries.md
T
2026-05-09 21:08:02 +09:00

3.9 KiB

id, title, category, status, source_trust_level, verification_status, created_at, updated_at, tags, tech_stack, applied_in, aliases
id title category status source_trust_level verification_status created_at updated_at tags tech_stack applied_in aliases
testing-mocking-boundaries Mock 경계 — 어디서 모킹할까 Coding draft B conceptual 2026-05-09 2026-05-09
testing
mocking
boundaries
vibe-coding
language applicable_to
TypeScript / Jest / Vitest
Backend
Web
test double
fake
stub
spy
msw

Mock 경계

모킹은 외부 시스템 경계 에서. 자기 코드 모킹 = test 가 코드 모양을 검증할 뿐 행위 안 검증. 외부 의존성 (HTTP, DB, time, fs) 은 모킹 OK 하되 integration test 에서는 진짜 사용 보완.

📖 핵심 개념

Test double 5종:

  • Dummy: 자리만 채움.
  • Stub: 정해진 값 반환.
  • Spy: 호출 기록 + 정해진 값.
  • Mock: 호출 기대 검증.
  • Fake: 진짜 동작 흉내 (in-memory DB).

💻 코드 패턴

MSW — HTTP 모킹 (web/node 통합)

import { setupServer } from 'msw/node';
import { http, HttpResponse } from 'msw';

const server = setupServer(
  http.get('https://api.example.com/users/:id', ({ params }) => {
    return HttpResponse.json({ id: params.id, email: `user${params.id}@a.com` });
  })
);

beforeAll(() => server.listen());
afterEach(() => server.resetHandlers());
afterAll(() => server.close());

test('fetch user', async () => {
  const u = await fetchUser('1');
  expect(u.email).toBe('user1@a.com');
});

Time 모킹

import { vi } from 'vitest';

beforeEach(() => vi.useFakeTimers());
afterEach(() => vi.useRealTimers());

test('debounce 300ms', () => {
  const fn = vi.fn();
  const debounced = debounce(fn, 300);
  debounced(); debounced(); debounced();
  vi.advanceTimersByTime(300);
  expect(fn).toHaveBeenCalledTimes(1);
});

의존성 주입 — fake repository

interface UserRepo {
  find(id: string): Promise<User | null>;
}

class InMemoryUserRepo implements UserRepo {
  private store = new Map<string, User>();
  async find(id: string) { return this.store.get(id) ?? null; }
  seed(u: User) { this.store.set(u.id, u); }
}

test('service uses repo', async () => {
  const repo = new InMemoryUserRepo();
  repo.seed({ id: '1', email: 'a@a.com' });
  const svc = new UserService(repo);
  expect(await svc.greet('1')).toBe('Hi a@a.com');
});

Spy on existing module

import * as mailer from './mailer';

test('sends welcome email', async () => {
  const sendSpy = vi.spyOn(mailer, 'sendEmail').mockResolvedValue();
  await registerUser({ email: 'a@a.com' });
  expect(sendSpy).toHaveBeenCalledWith({ to: 'a@a.com', template: 'welcome' });
});

🤔 의사결정 기준

의존성 권장
HTTP (외부 API) MSW (request-level)
DB testcontainers (real Postgres) — unit 에서는 in-memory fake
Time fake timers
Random seeded faker
Filesystem memfs 또는 tmpdir
자기 비즈니스 로직 모킹 X. 진짜 호출
Network 인접 (queue producer) spy on enqueue
큰 외부 SDK (AWS, Stripe) 외부 wrapper + wrapper interface 모킹

안티패턴

  • 자기 함수 모킹 (jest.mock('./foo')): 통합 안 됨. test 가 implementation 모양만 검증.
  • 모든 곳 mock: 가짜 통과. integration test 안 함.
  • mock 상태 누수: beforeEach reset 누락. 다음 test 영향.
  • MSW + jest.fn() 양쪽 사용: 일관성 깨짐. HTTP 는 MSW 한 군데로.
  • 모킹된 인터페이스가 실제와 다른 형태: deploy 후 production 에서 폭사. integration 검증.
  • 시간을 Date.now() 직접 모킹: 부작용 큼. fake timers.
  • "that's mocked" 라고 말하면서 진짜 호출: jest.spyOn vs jest.mock 차이 명확.

🤖 LLM 활용 힌트

  • 외부 API = MSW. DB = testcontainers. 자기 코드 = 모킹 X.
  • 의존성 주입 + in-memory fake 가 가장 깔끔한 unit test.

🔗 관련 문서