6.8 KiB
6.8 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 | |||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| db-lock-analysis | Postgres Lock 분석 — Deadlock / Wait / 진단 | Coding | draft | B | conceptual | 2026-05-09 | 2026-05-09 |
|
|
|
Postgres Lock 분석
"느린 query" 가 사실 lock wait.
pg_stat_activity+pg_locks로 누가 누구를 차단. lock_timeout / statement_timeout 으로 무한 hang 방지.
📖 핵심 개념
- Row-level lock: SELECT FOR UPDATE / DELETE / UPDATE.
- Table-level lock: ALTER / DROP / VACUUM FULL.
- Advisory lock: 직접 잡음.
- Deadlock: 양방 lock — Postgres 가 1개 abort.
💻 코드 패턴
Locking SELECT
-- Row lock (다른 transaction 의 같은 row UPDATE 차단)
SELECT * FROM accounts WHERE id = $1 FOR UPDATE;
-- Skip locked (큐 패턴)
SELECT * FROM jobs WHERE status = 'pending' ORDER BY id LIMIT 1
FOR UPDATE SKIP LOCKED;
-- No wait (즉시 fail)
SELECT * FROM accounts WHERE id = $1 FOR UPDATE NOWAIT;
-- Share (read 차단 X but write 차단)
SELECT * FROM accounts WHERE id = $1 FOR SHARE;
Lock conflicts
SELECT : ACCESS SHARE
SELECT FOR UPDATE / SHARE : ROW SHARE → ROW EXCLUSIVE
INSERT/UPDATE/DELETE : ROW EXCLUSIVE
CREATE INDEX : SHARE
CREATE INDEX CONCURRENTLY : SHARE UPDATE EXCLUSIVE (덜 차단)
ALTER TABLE : ACCESS EXCLUSIVE (모두 차단)
→ ALTER TABLE = 모든 query 차단. Concurrent 가능한 변경 사용.
Blocked queries 진단
SELECT
blocked.pid AS blocked_pid,
blocked.usename AS blocked_user,
blocked.query AS blocked_query,
blocking.pid AS blocking_pid,
blocking.usename AS blocking_user,
blocking.query AS blocking_query,
age(NOW(), blocked.query_start) AS blocked_for
FROM pg_stat_activity blocked
JOIN pg_stat_activity blocking ON blocking.pid = ANY(pg_blocking_pids(blocked.pid))
WHERE NOT blocked.pid = blocking.pid;
→ "이 query 가 저 query 를 대기 중".
Lock timeout
SET lock_timeout = '5s';
-- 5초 안 못 잡으면 ERROR
-- Statement timeout
SET statement_timeout = '30s';
-- 30초 안 안 끝나면 cancel
// App 에서 — 매 connection
await pool.query("SET statement_timeout = '30s'");
await pool.query("SET lock_timeout = '5s'");
또는 connection string:
postgresql://...?options=-c%20statement_timeout=30s
Deadlock
Tx A: UPDATE a → wait for b (held by Tx B)
Tx B: UPDATE b → wait for a (held by Tx A)
→ deadlock — Postgres 가 1개 abort
-- log
log_lock_waits = on
deadlock_timeout = 1s -- 1초 후 deadlock 검사
→ Deadlock 자주 = 코드에서 lock 순서 통일.
// ❌ 다른 순서
async function transferA(from, to) {
await update(from, ...); // lock from
await update(to, ...); // lock to
}
async function transferB(from, to) {
await update(to, ...); // lock to
await update(from, ...); // lock from — deadlock 가능
}
// ✅ 항상 ID 순
async function transfer(from, to) {
const [first, second] = [from, to].sort();
await update(first, ...);
await update(second, ...);
}
Long-running transaction
-- 30분+ transaction 검출
SELECT pid, usename, state, query, age(NOW(), xact_start) AS tx_age
FROM pg_stat_activity
WHERE state != 'idle' AND age(NOW(), xact_start) > INTERVAL '30 minutes';
// 강제 cancel
await db.query('SELECT pg_cancel_backend($1)', [pid]);
// 또는 더 강력
await db.query('SELECT pg_terminate_backend($1)', [pid]);
Idle in transaction (큰 문제)
App 이 BEGIN → 외부 API call → 1분 hang → 다른 트랜잭션 차단
-- 검출
SELECT * FROM pg_stat_activity WHERE state = 'idle in transaction';
-- 한도
ALTER SYSTEM SET idle_in_transaction_session_timeout = '60s';
SELECT pg_reload_conf();
→ 60초 후 자동 cancel.
Migration safety
-- ❌ Big table 에 column 추가 (default = NULL OK, default value 면 lock)
ALTER TABLE orders ADD COLUMN x INT DEFAULT 0;
-- PG 11+ = fast (metadata only). 옛 PG = full rewrite.
-- ❌ NOT NULL 추가
ALTER TABLE orders ADD COLUMN x INT NOT NULL DEFAULT 0;
-- 점진적 migration
-- ✅ Safer pattern (아래 DB_Migration_Safety 참조)
Index — concurrently
-- ❌ Lock 모든 write
CREATE INDEX orders_user ON orders (user_id);
-- ✅ Lock 없이 (느리지만 안전)
CREATE INDEX CONCURRENTLY orders_user ON orders (user_id);
Advisory lock (app 레벨)
-- 단일 instance 가 작업
SELECT pg_advisory_lock(42); -- bigint key
-- 작업
SELECT pg_advisory_unlock(42);
-- Transaction-scoped (자동 release)
SELECT pg_advisory_xact_lock(hashtext('process-orders'));
COMMIT;
→ Cron / single-leader 패턴.
Lock 통계
SELECT mode, count(*) FROM pg_locks GROUP BY mode;
-- 어떤 종류 lock 가 많은지
-- Wait 시간
SELECT
query, wait_event_type, wait_event, count(*)
FROM pg_stat_activity
WHERE wait_event IS NOT NULL
GROUP BY 1, 2, 3 ORDER BY count DESC;
App 측 transaction 짧게
// ❌ 트랜잭션 안 외부 호출
await db.transaction(async (tx) => {
const order = await tx.orders.find(id);
const result = await fetch(externalApi); // 1초+
await tx.orders.update(...);
}); // 1초 lock
// ✅
const order = await db.orders.find(id);
const result = await fetch(externalApi);
await db.transaction(async (tx) => {
await tx.orders.update(...); // ms
});
Optimistic vs Pessimistic
-- Pessimistic — lock
SELECT * FROM orders WHERE id = $1 FOR UPDATE;
-- 작업
-- Optimistic — version check
UPDATE orders SET ..., version = version + 1
WHERE id = $1 AND version = $expected_version;
-- 0 row affected = 다른 process 가 변경 — retry
→ Lock contention 적음 = optimistic.
🤔 의사결정 기준
| 상황 | 추천 |
|---|---|
| 큰 query lock 차단 | lock_timeout |
| 일반 prod | statement_timeout 30s |
| Idle in transaction | idle_in_transaction_session_timeout |
| Cron leader | Advisory lock |
| 큰 table ALTER | concurrent migration patterns |
| Update conflict | Optimistic version |
❌ 안티패턴
- Lock timeout 없음: hang. 5-30s 항상.
- Long transaction (30분+): autovacuum / 다른 query 차단.
- Idle in transaction 무한: timeout 설정.
- Lock 순서 다름: deadlock.
- ALTER TABLE prod + 큰 table: 모두 차단.
- CREATE INDEX (non-concurrent) prod: write 차단.
- App 이 외부 API 후 commit: tx 길어짐.
🤖 LLM 활용 힌트
- statement_timeout + lock_timeout 항상.
- pg_blocking_pids 로 진단.
- 트랜잭션 안 외부 호출 금지.