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

205 lines
5.4 KiB
Markdown

---
id: cs-crdt-patterns
title: CRDT — 자동 conflict 해결 / 협업
category: Coding
status: draft
source_trust_level: B
verification_status: conceptual
created_at: 2026-05-09
updated_at: 2026-05-09
tags: [cs, crdt, collab, vibe-coding]
tech_stack: { language: "TS / yjs / automerge", applicable_to: ["Frontend", "Backend"] }
applied_in: []
aliases: [CRDT, Yjs, Automerge, conflict-free, eventually consistent, OT vs CRDT]
---
# CRDT (Conflict-free Replicated Data Type)
> 여러 replica 가 다른 변경 후 자동 merge — conflict 없이. **실시간 협업 (Notion, Figma, Google Docs), local-first, offline sync**. **Yjs / Automerge**.
## 📖 핵심 개념
- 모든 operation 가 commutative + idempotent.
- Merge 결과 = operation 순서 무관.
- State-based vs Operation-based.
- OT (Operational Transform) vs CRDT — CRDT 가 modern.
## 💻 코드 패턴
### Yjs (가장 인기, 작음)
```ts
import * as Y from 'yjs';
import { WebsocketProvider } from 'y-websocket';
import { IndexeddbPersistence } from 'y-indexeddb';
const ydoc = new Y.Doc();
// 다른 client 와 동기화
const wsProvider = new WebsocketProvider('ws://localhost:1234', 'doc-1', ydoc);
// 로컬 영속
const persistence = new IndexeddbPersistence('doc-1', ydoc);
// Y.Map (object)
const ymap = ydoc.getMap('config');
ymap.set('theme', 'dark');
ymap.observe((event) => console.log('changed:', event.changes));
// Y.Array
const yarray = ydoc.getArray('items');
yarray.push(['item1']);
yarray.insert(0, ['first']);
// Y.Text (rich text)
const ytext = ydoc.getText('doc');
ytext.insert(0, 'Hello');
```
### Tiptap + Yjs (collab editor)
```tsx
import Collaboration from '@tiptap/extension-collaboration';
import CollaborationCursor from '@tiptap/extension-collaboration-cursor';
const editor = useEditor({
extensions: [
StarterKit.configure({ history: false }), // Yjs 가 history 관리
Collaboration.configure({ document: ydoc }),
CollaborationCursor.configure({
provider: wsProvider,
user: { name: 'Alice', color: '#f0f' },
}),
],
});
```
### Awareness (presence)
```ts
const awareness = wsProvider.awareness;
awareness.setLocalStateField('user', { name: 'Alice', cursor: { x: 100, y: 200 } });
awareness.on('change', () => {
const states = Array.from(awareness.getStates().values());
// 다른 사용자 보임
});
```
### Automerge (다른 popular CRDT)
```ts
import * as A from '@automerge/automerge';
let doc = A.from({ items: [], title: 'Untitled' });
doc = A.change(doc, 'add item', d => {
d.items.push({ id: '1', text: 'first' });
});
// Sync (binary)
const change = A.getLastLocalChange(doc);
sendToPeer(change);
// 다른 peer
let doc2 = A.from({ items: [], title: 'Untitled' });
doc2 = A.applyChanges(doc2, [change])[0];
```
### Merge (자동)
```ts
// Alice
docA.text.insert(0, 'A');
// Bob (offline)
docB.text.insert(0, 'B');
// 동기화 → 둘 다 'AB' 또는 'BA' (deterministic)
```
### Server provider
```ts
// Hocuspocus (Yjs 전용 server)
import { Server } from '@hocuspocus/server';
const server = Server.configure({
port: 1234,
async onAuthenticate({ token }) {
const user = verifyJwt(token);
return { user };
},
async onLoadDocument({ documentName }) {
const ydoc = new Y.Doc();
const persisted = await db.docs.get(documentName);
if (persisted) Y.applyUpdate(ydoc, persisted);
return ydoc;
},
async onStoreDocument({ documentName, document }) {
await db.docs.upsert(documentName, Y.encodeStateAsUpdate(document));
},
});
server.listen();
```
### LiveBlocks (managed)
```tsx
import { useStorage, useMutation } from '@/liveblocks.config';
const items = useStorage(root => root.items);
const addItem = useMutation(({ storage }, text: string) => {
storage.get('items').push({ id: nanoid(), text });
}, []);
```
→ Yjs / 자체 server 관리 안 해도 됨.
### Local-first
```
1. 로컬 변경 → IndexedDB 저장
2. 네트워크 OK → server 와 sync
3. Offline → 로컬만 — 정상 동작
4. Reconnect → 자동 merge
```
→ Notion / Linear 의 UX.
### 단점
```
- Document size 시간 따라 커짐 (history 누적).
- Garbage collection 필요.
- Schema 변경 어려움.
- Server-side validation 어려움 (CRDT 가 자유 변경).
```
### 사용 예
- 협업 editor: Yjs + Tiptap / Lexical.
- Real-time dashboard: Yjs + React.
- Offline-first app: Automerge / Yjs IndexedDB.
- Multi-cursor: Awareness.
## 🤔 의사결정 기준
| 상황 | 추천 |
|---|---|
| Real-time editor | Yjs + ProseMirror/Tiptap |
| 작은 data + JSON-like | Automerge |
| Managed | LiveBlocks / Liveblocks |
| 단순 sync (last-write-wins) | 직접 구현 |
| Strong consistency | DB transaction (CRDT X) |
| 1 사용자 / 1 device | CRDT 불필요 |
## ❌ 안티패턴
- **Server-side validation 강 가정**: CRDT 가 client 자유. 별도 룰 + reject.
- **Offline 무한 길이**: history 누적. GC.
- **CRDT 안에 secret / PII**: client 가 모든 history 접근.
- **Schema 자주 변경**: 마이그레이션 어려움.
- **너무 큰 document (10MB)**: sync 느림.
- **Custom CRDT 자체 구현**: 어려움. Yjs / Automerge 사용.
- **LWW (last write wins) 만 사용 + 잃음**: CRDT = 양쪽 보존.
## 🤖 LLM 활용 힌트
- Yjs 가 가장 인기 + 강력.
- Hocuspocus 또는 LiveBlocks 가 server 답.
- Awareness 로 cursors / presence.
- Strong validation 필요 시 server-side check 분리.
## 🔗 관련 문서
- [[React_Editor_Slate_Lexical]]
- [[Backend_WebSocket_Scaling]]
- [[DB_Distributed_Locks]]