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

161 lines
6.0 KiB
Markdown

---
id: optimistic-concurrency-control
title: 낙관적 동시성 제어 (Optimistic Concurrency Control)
category: Coding
status: draft
canonical_id: optimistic-concurrency-control
aliases: [OCC, optimistic locking, version field, ETag, 낙관적 락]
duplicate_of: null
source_trust_level: B
confidence_score: 0.9
verification_status: conceptual
created_at: 2026-05-09
updated_at: 2026-05-09
last_reinforced: 2026-05-09
review_reason: ""
merge_history: []
tags: [coding, concurrency, database, http, vibe-coding]
raw_sources: ["P-Reinforce session 2026-05-09 — bulk Coding seed batch 1"]
tech_stack:
language: "SQL / TypeScript / HTTP"
applicable_to: ["Backend", "REST API", "DB write paths"]
applied_in: []
---
# 낙관적 동시성 제어
> 충돌은 드물다고 가정하고 락 없이 진행. 충돌이 발생하면 **버전 비교로 감지하고 재시도하거나 사용자에게 알린다**. 비관적 락보다 throughput 높지만 충돌이 잦으면 retry 폭주.
## 📖 핵심 개념
두 사용자가 같은 레코드를 동시에 수정할 때:
- **비관적(pessimistic) 락**: 한 명이 잠금 → 다른 명은 대기. throughput 낮음, deadlock 위험.
- **낙관적(optimistic)**: 둘 다 진행, 마지막 commit 시점에 "내가 읽은 버전이 아직도 유효한가?" 확인. 깨졌으면 reject.
낙관적 락은 다음 둘 중 하나로 구현:
1. **Version 컬럼** (정수 또는 timestamp): UPDATE 시 `WHERE version = ?` 조건. 변경 시 version+1.
2. **HTTP ETag / If-Match**: 서버가 ETag 발급 → 클라이언트가 PUT 시 `If-Match` 헤더로 보냄.
## 💻 코드 패턴
### 1. Version 컬럼 (SQL)
```sql
ALTER TABLE orders ADD COLUMN version INT NOT NULL DEFAULT 0;
```
```ts
async function updateShipping(orderId: string, addr: Address, expectedVersion: number) {
const result = await db.query(
`UPDATE orders
SET shipping = $1, version = version + 1
WHERE id = $2 AND version = $3`,
[addr, orderId, expectedVersion]
);
if (result.rowCount === 0) {
// 누군가 이미 수정했거나 주문이 없음
throw new ConflictError('Concurrent update detected — re-read and try again');
}
}
```
### 2. HTTP ETag
```ts
// GET — ETag 발급
app.get('/api/orders/:id', async (req, res) => {
const order = await db.findOrder(req.params.id);
res.setHeader('ETag', `"${order.version}"`);
res.json(order);
});
// PUT — If-Match 검증
app.put('/api/orders/:id', async (req, res) => {
const ifMatch = req.header('If-Match');
if (!ifMatch) return res.status(428).json({ error: 'If-Match required' });
const expected = parseInt(ifMatch.replace(/"/g, ''), 10);
try {
await updateShipping(req.params.id, req.body.shipping, expected);
res.status(204).end();
} catch (e) {
if (e instanceof ConflictError) return res.status(412).json({ error: 'Version mismatch' });
throw e;
}
});
```
### 3. 자동 재시도 wrapper
```ts
async function retryOnConflict<T>(
op: () => Promise<T>,
{ maxAttempts = 3, backoffMs = 50 }: { maxAttempts?: number; backoffMs?: number } = {}
): Promise<T> {
let lastErr: unknown;
for (let i = 0; i < maxAttempts; i++) {
try {
return await op();
} catch (e) {
if (!(e instanceof ConflictError)) throw e;
lastErr = e;
await new Promise(r => setTimeout(r, backoffMs * Math.pow(2, i)));
}
}
throw lastErr;
}
```
`op()` 안에서 항상 **다시 read → 변경 → write** 흐름이어야 한다. 같은 expectedVersion 으로 재시도하면 영원히 실패.
### 4. Document DB (MongoDB) — findOneAndUpdate with filter
```ts
const result = await orders.findOneAndUpdate(
{ _id: orderId, version: expectedVersion },
{ $set: { shipping: addr }, $inc: { version: 1 } },
{ returnDocument: 'after' }
);
if (!result.value) throw new ConflictError('version mismatch');
```
## 🤔 의사결정 기준
| 상황 | OCC 적합 | 비관적 락 적합 |
|---|---|---|
| 충돌 빈도 < 5% | ✅ | ❌ |
| 충돌 빈도 > 30% | ❌ (retry 폭주) | ✅ |
| 트랜잭션이 짧음 | ✅ | — |
| 트랜잭션이 길고 사용자 대기 중 | ✅ (사용자에게 충돌 알림) | ❌ (deadlock) |
| 분산 환경 (DB 여러 대) | ✅ | 어려움 |
| 잔액 / 재고 같은 strict invariant | OCC 가능하지만 SELECT FOR UPDATE 또는 `UPDATE ... WHERE balance >= ?` 사용 | ✅ |
## ❌ 안티패턴
- **버전 비교 없이 `UPDATE SET ... WHERE id = ?`**: lost update. 마지막 쓰기가 이전 쓰기를 조용히 덮어씀.
- **클라이언트가 보낸 version 을 그대로 신뢰 후 검증 안 함**: 클라이언트가 의도적으로 옛 version 보내면 항상 통과.
- **재시도 시 같은 expectedVersion 사용**: 영원히 실패. 재시도 안에서 **다시 read** 해야 함.
- **충돌 시 묵묵히 무시**: 사용자는 자기 변경이 적용되었다고 믿음. 실제로 사라짐. 명시적 4xx 응답 + UI 알림.
- **Version 을 timestamp 로**: 같은 ms 에 두 update 가 들어오면 둘 다 통과. 정수 시퀀스가 안전.
- **모든 곳에 자동 retry**: 사용자가 form 에 입력한 값을 자동 재시도 = 사용자가 본 데이터와 다를 수 있음. UI 가 "다른 사람이 수정했어요. 다시 보시겠어요?" 묻기.
- **OCC 와 application-level 캐시 결합**: 캐시가 stale version 을 들고 있으면 영원히 충돌. 캐시 invalidation 정책 필수.
## 🤖 LLM 활용 힌트
- LLM에게 SQL UPDATE 작성: "**WHERE 조건에 version = expected 추가, rowCount 0 이면 ConflictError throw**" 명시.
- REST API: "**GET 응답에 ETag, PUT 요청에 If-Match 강제 (없으면 428)**" 패턴 요청.
- retry wrapper: "**read → modify → write 흐름이어야 retry 의미 있음. 같은 expected 로 재시도 금지**" 강조.
## 🧪 검증 상태
- verification_status: `conceptual`
- JPA `@Version`, Hibernate optimistic locking, REST API ETag 표준 (RFC 7232) 의 핵심 패턴.
- 적용 사례 발견 시 `applied_in` 추가.
## 🔗 관련 문서
- [[Idempotent_Operations]]
- [[Backpressure_Patterns]]
- [[Error_Handling_Result_vs_Throw]]