106 lines
3.3 KiB
Markdown
106 lines
3.3 KiB
Markdown
---
|
|
id: testing-snapshot-patterns
|
|
title: Snapshot Test — 적절한 사용처
|
|
category: Coding
|
|
status: draft
|
|
source_trust_level: B
|
|
verification_status: conceptual
|
|
created_at: 2026-05-09
|
|
updated_at: 2026-05-09
|
|
tags: [testing, snapshot, jest, vibe-coding]
|
|
tech_stack: { language: "Jest / Vitest", applicable_to: ["Web", "Backend"] }
|
|
applied_in: []
|
|
aliases: [toMatchSnapshot, inline snapshot, golden file]
|
|
---
|
|
|
|
# Snapshot Test
|
|
|
|
> 의도적으로 사용하면 강력하지만 무지성으로 쓰면 "test 가 그냥 현재 출력을 freeze" 함. 변경 → snapshot 갱신 → 의미 없는 통과. **diff 검토 문화** 가 핵심.
|
|
|
|
## 📖 핵심 개념
|
|
- 첫 실행: 출력 기록.
|
|
- 이후: 기록과 다르면 fail.
|
|
- Inline snapshot: test 파일에 직접 (변화 즉시 보임).
|
|
- File snapshot: 별도 `.snap` 파일.
|
|
|
|
## 💻 코드 패턴
|
|
|
|
### React 컴포넌트 — 보수적 사용
|
|
```ts
|
|
import { render } from '@testing-library/react';
|
|
|
|
test('Header renders title', () => {
|
|
const { container } = render(<Header title="Hello" />);
|
|
expect(container).toMatchSnapshot();
|
|
});
|
|
```
|
|
|
|
⚠️ 큰 컴포넌트는 변화 잦음. inline + 핵심만 권장.
|
|
|
|
### Inline snapshot — diff 즉시 visible
|
|
```ts
|
|
import { test, expect } from 'vitest';
|
|
|
|
test('format duration', () => {
|
|
expect(formatDuration(95_000)).toMatchInlineSnapshot(`"1m 35s"`);
|
|
// jest --ci 면 fail / 로컬 --updateSnapshot 으로 갱신
|
|
});
|
|
```
|
|
|
|
### Serializer — 핵심만 비교
|
|
```ts
|
|
expect.addSnapshotSerializer({
|
|
test: (val) => val instanceof Date,
|
|
print: () => '"<Date>"',
|
|
});
|
|
|
|
// timestamp 같이 매번 다른 값은 mask
|
|
test('user object', () => {
|
|
expect(makeUser()).toMatchSnapshot();
|
|
// {"id": "<Date>", "createdAt": "<Date>", ...}
|
|
});
|
|
```
|
|
|
|
### Property snapshot — 동적 값 마스킹
|
|
```ts
|
|
expect(user).toMatchSnapshot({
|
|
id: expect.any(String),
|
|
createdAt: expect.any(Date),
|
|
});
|
|
```
|
|
|
|
### Golden file — 큰 출력
|
|
```ts
|
|
test('SQL builder generates expected query', async () => {
|
|
const sql = build({ table: 'users', filters: { age: 18 } });
|
|
await expect(sql).toMatchFileSnapshot('./golden/users.sql');
|
|
});
|
|
```
|
|
|
|
## 🤔 의사결정 기준
|
|
| 사용처 | snapshot 적합 |
|
|
|---|---|
|
|
| 작은 안정적 컴포넌트 | ✅ 단 inline + 짧게 |
|
|
| 큰 페이지 컴포넌트 | ❌ — assertion 으로 행위 검증 |
|
|
| 빌더 / 직렬화 결과 | ✅ — golden file |
|
|
| API 응답 형식 | ❌ — schema 검증 (zod) 권장 |
|
|
| CSS-in-JS 출력 | ❌ — 자주 바뀜 |
|
|
| 에러 메시지 텍스트 | ✅ inline |
|
|
|
|
## ❌ 안티패턴
|
|
- **`-u` 무지성**: CI 빨갛다고 자동 update. 의미 없는 통과.
|
|
- **거대 snapshot**: diff 못 읽음. 작은 단위로 분할.
|
|
- **timestamp / random / uuid 안 마스킹**: 매번 fail. property matcher.
|
|
- **snapshot only test**: 비즈니스 행위 검증 0. assertion 결합.
|
|
- **inline 안 쓰고 매번 .snap 파일 열기**: 흐름 깨짐. 작은 출력은 inline.
|
|
- **eslint-plugin-jest 의 `no-large-snapshots` 무시**: 50줄 넘는 snapshot 은 검토 신호.
|
|
- **PR 에서 snapshot 변경 review 안 함**: 사일런트 회귀.
|
|
|
|
## 🤖 LLM 활용 힌트
|
|
- "snapshot 보단 명시적 assertion 우선. snapshot 은 builder/format/error 메시지 같은 안정 출력만" 명시.
|
|
- 동적 값은 property matcher 로 마스킹.
|
|
|
|
## 🔗 관련 문서
|
|
- [[Testing_Test_Pyramid]]
|
|
- [[Testing_Faker_and_Builders]]
|