Files
2nd/10_Wiki/Topics/Coding/DB_Audit_Log_Patterns.md
T
Antigravity Agent f8b21af4be Wiki cleanup: error-doc removal, dedup merge, link normalization
10_Wiki/Topics 대규모 정리:
- 오류 캡처/미완성 stub 문서 227개 제거
- 교차폴더 중복 43클러스터 병합 (63파일 → redirect)
- 링크명 정규화: 깨진 링크 수정·redirect 직결·개념 매핑 ~2,400건
- 카테고리 MOC 6개 신규 생성
- Graph 섹션 미해결 related-keyword 링크 10,058건 제거

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-20 23:52:15 +09:00

4.5 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-audit-log-patterns Audit Log — Trigger / 이벤트 / 변경 추적 Coding draft B conceptual 2026-05-09 2026-05-09
database
audit
log
trigger
vibe-coding
language applicable_to
SQL / Postgres
Backend
audit trail
change data capture
CDC
who changed what
history table

Audit Log

누가 / 언제 / 무엇을 / 왜 변경했는지 추적. Compliance (SOC2/GDPR) + 디버깅. 트리거 / 앱 레벨 / CDC (Debezium) 3가지 방식. 같은 DB / 별도 테이블 / 별도 시스템.

📖 핵심 개념

  • Append-only: 변경 이력은 절대 수정/삭제 X.
  • 4가지 정보: who, when, what (table+pk), how (old → new).
  • WORM: write-once read-many. compliance 요구.

💻 코드 패턴

Trigger (Postgres) — 자동 캡처

CREATE TABLE audit_log (
  id BIGSERIAL PRIMARY KEY,
  table_name TEXT NOT NULL,
  operation TEXT NOT NULL,  -- INSERT/UPDATE/DELETE
  row_id TEXT NOT NULL,
  old_data JSONB,
  new_data JSONB,
  changed_by UUID,
  changed_at TIMESTAMPTZ DEFAULT NOW(),
  ip TEXT
);

CREATE OR REPLACE FUNCTION audit_trigger() RETURNS trigger AS $$
DECLARE
  uid UUID;
BEGIN
  -- 앱에서 SET LOCAL app.user_id 로 주입
  uid := current_setting('app.user_id', true)::UUID;

  IF TG_OP = 'INSERT' THEN
    INSERT INTO audit_log(table_name, operation, row_id, new_data, changed_by)
    VALUES (TG_TABLE_NAME, 'INSERT', NEW.id::TEXT, to_jsonb(NEW), uid);
    RETURN NEW;
  ELSIF TG_OP = 'UPDATE' THEN
    INSERT INTO audit_log(table_name, operation, row_id, old_data, new_data, changed_by)
    VALUES (TG_TABLE_NAME, 'UPDATE', NEW.id::TEXT, to_jsonb(OLD), to_jsonb(NEW), uid);
    RETURN NEW;
  ELSIF TG_OP = 'DELETE' THEN
    INSERT INTO audit_log(table_name, operation, row_id, old_data, changed_by)
    VALUES (TG_TABLE_NAME, 'DELETE', OLD.id::TEXT, to_jsonb(OLD), uid);
    RETURN OLD;
  END IF;
END;
$$ LANGUAGE plpgsql;

CREATE TRIGGER users_audit AFTER INSERT OR UPDATE OR DELETE ON users
  FOR EACH ROW EXECUTE FUNCTION audit_trigger();

앱이 user 주입

// 매 트랜잭션 시작 시
await db.execute(`SET LOCAL app.user_id = '${ctx.userId}'`);

App-level audit (간단 케이스)

async function updateUser(id: string, patch: Partial<User>, by: string) {
  return db.transaction(async (tx) => {
    const before = await tx.user.findUnique({ where: { id } });
    const after = await tx.user.update({ where: { id }, data: patch });
    await tx.auditLog.create({
      data: {
        table: 'users', op: 'UPDATE', rowId: id,
        oldData: before, newData: after, changedBy: by,
      },
    });
    return after;
  });
}

History 테이블 (full row snapshot)

CREATE TABLE users_history (
  history_id BIGSERIAL PRIMARY KEY,
  id UUID NOT NULL,         -- 원래 PK
  email TEXT,
  -- ... users 의 모든 컬럼
  valid_from TIMESTAMPTZ,
  valid_to TIMESTAMPTZ DEFAULT 'infinity',
  changed_by UUID
);

-- 시점별 조회
SELECT * FROM users_history
WHERE id = $1 AND valid_from <= $time AND valid_to > $time;

CDC (Debezium / wal2json)

  • Postgres logical replication slot → Debezium → Kafka → audit service.
  • DB 부하 작음, 별 시스템 분리.
  • Compliance 시스템에 적합.

차이만 저장

-- json diff 만 저장 (적은 공간)
INSERT INTO audit_log(table_name, row_id, diff)
VALUES (..., jsonb_build_object('email', ARRAY[OLD.email, NEW.email]));

🤔 의사결정 기준

상황 추천
단순 변경 추적 Trigger + audit_log
시점별 row 복원 History 테이블
다른 시스템에 stream CDC (Debezium)
Compliance 강함 WORM 별 시스템 (S3 Object Lock)
앱 코드 복잡 Trigger (한 곳)
Multi-tenant tenant_id 같이 저장

안티패턴

  • App-level only 인데 raw SQL 우회: trigger 가 안전망.
  • PII 그대로 audit log: GDPR — 마스킹 또는 별 보안 영역.
  • audit_log 인덱스 없음: row_id / changed_at 검색 느림.
  • audit 테이블 update/delete 허용: WORM 깨짐. 권한 분리.
  • 변경 내용만 — who 없음: 추적 무의미.
  • JSON 거대: 1MB+ 필드는 reference 만.
  • Retention 정책 없음: 무한히 자라남.

🤖 LLM 활용 힌트

  • Trigger 가 자동, app-level 은 빠뜨릴 수 있음.
  • who = SET LOCAL 또는 ORM hook.
  • old/new JSONB 가 가장 단순 강력.

🔗 관련 문서