[G1-Sync] Manual knowledge update
This commit is contained in:
@@ -0,0 +1,105 @@
|
||||
---
|
||||
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]]
|
||||
Reference in New Issue
Block a user