Files
vpd-permission-poc/README.md
devmrko b80d11953b Add DDS setup quick-reference section to README
Existing docs/05-dds-variant.md covered DDS in depth but readers had to
dig through it to find the actual setup recipe. New README section
distills it to: prereqs, 4-step DDL recipe, the 4 most common gotchas
(including the ORA-01917 END USER + regular ROLE trap and the VPD/DDS
view collision), and a short list of common variants.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-05-26 16:36:13 +09:00

13 KiB
Raw Permalink Blame History

VPD Permission POC — ADB 중심 행 단위 접근 제어

Oracle Autonomous Database (26ai) 의 VPD (Virtual Private Database)Data Redaction 을 이용해, ADB 한 곳에서 여러 원본 DB (AWS RDS for PostgreSQL, MySQL) 의 데이터를 유저 → 그룹 → 권한 매핑 테이블 만으로 행/열 단위 통제하는 End-to-End 데모입니다.

핵심: 원본 DB 는 단순한 데이터 소스로 두고, 접근 권한은 ADB 의 VPD 정책 함수와 Secure Application Context 로 통합 관리합니다. 애플리케이션 코드 변경 없이 로그인하는 DB 유저에 따라 자동으로 행이 필터링됩니다.


무엇을 통제하는가

이 POC 가 다루는 통제 축은 세 가지입니다.

1. 행 단위 접근 제어 (Row-Level Access Control)

네 명의 엔드유저가 같은 SQL — 예: SELECT * FROM v_customers_pg — 를 실행해도, 세션 정체성에 매핑된 권한에 따라 서로 다른 행 집합 을 돌려받습니다.

  • 권한 회수는 GRANT SELECT 해제가 아니라 permission 테이블의 행 삭제로 일어납니다. GRANT 는 4 명 모두 동일하게 받아 두고, 차이를 만드는 것은 매핑 데이터입니다.
  • 매핑이 아예 없는 유저는 자동으로 빈 결과 (VPD: 1=0) 또는 객체 자체 미노출 (DDS: ORA-00942) — default deny.
  • region IN ('APAC') 같은 행 필터는 정책 함수 또는 데이터 그랜트의 WHERE 절로 ADB 한 곳에서 선언됩니다. 원본 DB (PostgreSQL, MySQL) 는 이 사실을 모릅니다.

2. 컬럼 마스킹 (Column-Level Redaction)

행 자체는 보여주되 특정 컬럼만 가리는 통제입니다. 예: 영업팀은 고객 행은 다 보지만 email 컬럼은 항상 NULL 로 반환.

  • VPD 경로: 별도의 DBMS_REDACT 정책 (sql/adb/06a_redaction.sql).
  • DDS 경로: 데이터 그랜트 안에 (ALL COLUMNS EXCEPT email) 한 줄로 흡수.

3. 두 가지 구현 — VPD 와 DDS

동일한 4 유저 × 2 소스 매트릭스를 두 방식으로 각각 구현했습니다. 결과(행/마스킹)는 같지만, 정책을 어떻게 표현하느냐 가 다릅니다.

항목 VPD (기존) DDS (26ai 신규)
정책 표현 PL/SQL 정책 함수 + Application Context 선언형 SQL (CREATE DATA GRANT)
컬럼 마스킹 별도 DBMS_REDACT 정책 객체 그랜트 안에 ALL COLUMNS EXCEPT
권한 없는 접근 0 rows 반환 (객체는 보임) 객체 자체 미노출 (ORA-00942)
신원 매핑 DB 유저 → LOGON 트리거 → 세션 컨텍스트 END USER → DATA ROLE → DATA GRANT
운영 코드량 정책 함수 + 컨텍스트 패키지 + 로그온 트리거 데이터 그랜트 DDL 만

애플리케이션 코드는 두 경우 모두 권한 로직을 전혀 알 필요가 없습니다. 누가 어떤 행/컬럼을 볼 수 있는지는 ADB 한 곳에서 결정되고, 원본 DB 와 애플리케이션은 그 결정의 결과만 받습니다.

DDS 변형의 능력 (멀티테넌트, OAuth2 federated identity, MAC 모드, 운영 단위 그랜트 등) 과 한계는 docs/05-dds-variant.md 참고.


구성요소

계층 객체 역할
원격 public.customers (PG), ecommerce_poc.customers (MySQL) 원본 데이터
ADB - 연결 RDS_POSTGRES_LINK, RDS_LINK DBMS_CLOUD_ADMIN.CREATE_DATABASE_LINK 로 만든 heterogeneous DB Link
ADB - 매핑 app_customer, app_user, app_group, user_group, db_source, permission 누가 어느 소스의 어느 region 을 볼 수 있는지
ADB - 컨텍스트 vpd_ctx (Secure Application Context) + ctx_pkg 로그인 시 권한을 세션 컨텍스트로 로딩
ADB - 뷰 v_customers_pg, v_customers_my DB Link 위의 통합 뷰. 이 뷰에만 VPD/Redaction 정책이 붙음
ADB - 정책 CUSTOMERS_PG_POLICY, CUSTOMERS_MY_POLICY, PII_REDACT_PG/MY VPD 행 필터 + 이메일 마스킹
ADB - 엔드유저 vpduser_my, vpduser_pg, vpduser_both, vpduser_none 최소 권한. 뷰만 SELECT 가능. LOGON 트리거로 컨텍스트 자동 로딩

빠른 시작 (One-Click)

git clone https://github.com/<you>/vpd-permission-poc.git
cd vpd-permission-poc

# 1) 환경값 채우기
cp .env.example .env
$EDITOR .env

# 2) 끝
./run.sh

./run.sh 가 다음을 순서대로 실행합니다:

  1. prereqsqlplus, psql, mysql 존재 + .env 변수 검증
  2. source — 원격 PG/MySQL 에 customers 테이블 + seed (멱등)
  3. adb — ADB 측 cleanup → DB Link → 권한 테이블/seed → context/view/policy → 엔드유저
  4. tests — 4 명 (vpduser_my/vpduser_pg/vpduser_both/vpduser_none) 로 접속해 행 필터/마스킹/default-deny 검증
  5. audit — ADMIN 으로 정책/뷰/유저 상태 점검

세부 단계만 따로 돌리려면:

./run.sh prereq      # 환경만 검증
./run.sh source      # 원격 DB 만 준비
./run.sh adb         # ADB 만 준비
./run.sh tests       # 엔드유저 테스트만
./run.sh audit       # ADMIN audit 만
./run.sh teardown    # ADB 측 객체 + DB Link/credential 정리

(선택) Oracle 26ai Deep Data Security 변형도 같은 시나리오를 declarative SQL 로 재현합니다. VPD 데모와 공존하며 별도 ddsuser_/v_dds_ 객체로 깔립니다. (docs/05-dds-variant.md)

./run.sh dds            # dds-setup + dds-tests
./run.sh dds-setup      # 13_dds_variant.sql 적용
./run.sh dds-tests      # ddsuser_* 4명 매트릭스 검증
./run.sh dds-teardown   # DDS 객체만 정리 (VPD 데모는 보존)

사전 준비

항목 필요 사항
ADB Autonomous Database (Always Free 도 가능). Wallet 다운로드 + Instant Client 에 풀어둠
PostgreSQL 호스트/포트 도달 가능. vpdpoc (혹은 원하는 이름) DB 가 미리 생성되어 있어야 함
MySQL 호스트/포트 도달 가능. ecommerce_poc (혹은 원하는 이름) DB 가 미리 생성되어 있어야 함
클라이언트 도구 sqlplus (Instant Client), psql, mysql

.env.example 의 각 항목 의미는 같은 파일 안 주석 참고.


데모 시나리오 — 2×2 source access matrix

sql/adb/03_seed.sql 의 매핑 (4 유저, 4 케이스):

DB 유저 그룹 PG 뷰 MySQL 뷰 VPD 결과
vpduser_my MY_ONLY ✗ 0 rows ✓ ALL PG=1=0 / MY=NULL
vpduser_pg PG_ONLY ✓ ALL ✗ 0 rows PG=NULL / MY=1=0
vpduser_both BOTH_SOURCES ✓ ALL ✓ ALL PG=NULL / MY=NULL
vpduser_none (그룹 없음) ✗ 0 rows ✗ 0 rows PG=1=0 / MY=1=0 (default deny)

핵심 포인트:

  • 네 유저 모두 양쪽 뷰에 SELECT GRANT 는 동일하게 주어집니다. 차이를 만드는 건 GRANT 가 아니라 permission 테이블의 행. 즉 권한 회수 = GRANT 해제가 아니라 permission row 삭제.
  • vpduser_none 처럼 매핑이 아예 없는 유저는 자동으로 fail-closed (1=0 predicate) — deny by default.
  • 누구든 원본 테이블 직접 접근 시도 (@RDS_POSTGRES_LINK 등) → 권한 없음.

sql/adb/08_tests_user_my.sql 가 우회 시도 5개 (원격 직접 SELECT, 컨텍스트 스푸핑, DBMS_RLS 변경, 매핑 테이블 SELECT) 를 시도하고 모두 ORA-xxxxx 로 실패하는 것을 보여줍니다. 09/10/11 은 각 유저의 expected 행 수를 가볍게 확인합니다.

Row-level region 필터링 (예: vpduser_both 가 PG 는 APAC 만) 도 정책 함수에 그대로 살아있습니다. 03_seed.sql 하단의 주석 처리된 UPDATE permission ... allowed_regions='APAC' 한 줄이면 활성화됩니다.


DDS 설정 핵심

DDS 변형을 직접 셋업할 때 알아야 하는 최소 단위. sql/adb/13_dds_variant.sql 가 이 순서로 돌아가며, 행/컬럼 통제가 모두 선언형 DDL 한 줄 로 끝납니다.

사전 조건

  • Oracle AI Database 23.26.2 이상 (ADB 는 기본 충족)
  • COMPATIBLE 파라미터 ≥ 20.0
  • ADMIN 권한: CREATE END USER, CREATE DATA ROLE, GRANT DATA ROLE, CREATE DATA GRANT

4 스텝 DDL 레시피

-- 1) END USER — 스키마 없음, 로그온 전용
CREATE END USER "alice" IDENTIFIED BY "<pw>";

-- 2) CREATE SESSION 캐리어. END USER 는 regular ROLE 을 직접 받지 못함
--    (ORA-01917) 이므로 반드시 DATA ROLE 으로 한 번 감싼다.
CREATE ROLE      dds_db_role;
GRANT CREATE SESSION TO dds_db_role;

CREATE DATA ROLE analyst_role;
GRANT dds_db_role TO analyst_role;

-- 3) END USER ↔ DATA ROLE 매핑
GRANT DATA ROLE analyst_role TO "alice";

-- 4) 데이터 그랜트 — 행 필터(WHERE) + 컬럼 마스킹(ALL COLUMNS EXCEPT) 이 한 줄에 결합
CREATE DATA GRANT admin.alice_apac_grant
  AS SELECT (ALL COLUMNS EXCEPT email)
  ON   admin.v_customers
  WHERE region = 'APAC'
  TO   analyst_role;

반드시 알아야 할 함정

증상 원인 대응
ORA-01917 user or role does not exist END USER 에게 regular ROLE 을 직접 GRANT 위처럼 DATA ROLE 으로 한 번 감싸기
권한 있는 유저인데 0 rows 같은 뷰에 VPD policy 가 살아있음 (VPD 의 1=0 가 DDS 보다 먼저 적용) DDS 전용 뷰 (예: v_dds_*) 를 따로 만들거나 SET USE DATA GRANTS ONLY 적용
권한 없는 유저가 0 rows 가 아니라 ORA-00942 DDS 정상 동작 — 객체 자체 미노출 그대로 둠. VPD 의 "0 rows" 보다 강한 enumeration 방어
데이터 그랜트가 안 먹는 것 같음 dba_data_grants 미확인 SELECT * FROM dba_data_grants; / dba_data_roles / dba_end_users 로 상태 점검

자주 쓰는 변주

  • 컬럼만 마스킹: AS SELECT (ALL COLUMNS EXCEPT ssn) ON ... — 별도 redaction 정책 불필요
  • MAC 모드 (해당 객체는 DATA GRANT 로만 접근): ALTER TABLE admin.customers SET USE DATA GRANTS ONLY ENABLED;
  • 운영(operation) 단위: AS INSERT ON ..., AS UPDATE (status) ON ... WHERE owner_id = ORA_END_USER_CONTEXT.username TO ...
  • Federated identity: CREATE DATA ROLE ... MAPPED TO 'AZURE_ROLE=Sales'; — OAuth2 토큰 클레임으로 자동 매핑

자세한 옵션과 운영 가이드는 docs/05-dds-variant.md 참고.


디렉토리

.
├── run.sh                       # 원클릭 엔트리포인트
├── .env.example
├── scripts/lib/common.sh        # log/ok/warn/die + env 검증 헬퍼
├── sql/
│   ├── source/
│   │   ├── postgres_setup.sql   # 원격 PG: customers + 12 rows
│   │   └── mysql_setup.sql      # 원격 MySQL: customers + 12 rows
│   └── adb/
│       ├── 00_cleanup.sql       # 멱등 teardown
│       ├── 01_dblinks.sql       # DB Link + credential 생성
│       ├── 02_perm_tables.sql   # 6개 권한 매핑 테이블
│       ├── 03_seed.sql          # 데모 매핑 데이터
│       ├── 04_secure_ctx.sql    # ctx_pkg + vpd_ctx (Secure App Context)
│       ├── 05_views.sql         # DB Link 통합 뷰
│       ├── 06_policy.sql        # VPD 정책 + 정책 함수
│       ├── 06a_redaction.sql    # Data Redaction (이메일 마스킹)
│       ├── 07_end_users.sql        # 4 유저 + LOGON 트리거
│       ├── 08_tests_user_my.sql    # MY only — 우회 시도 5종 포함
│       ├── 09_tests_user_pg.sql    # PG only
│       ├── 10_tests_user_both.sql  # both
│       ├── 11_tests_user_none.sql  # default deny (fail-closed) 검증
│       ├── 12_tests_admin_audit.sql
│       ├── 13_dds_variant.sql      # (선택) 같은 4-user 매트릭스를 26ai Deep Data Security 로 재구현
│       ├── 14_tests_dds_user.sql   # (선택) ddsuser_* 공용 매트릭스 검증 + bypass 시도
│       └── 15_dds_cleanup.sql      # (선택) DDS 객체 멱등 정리
└── docs/
    ├── 03-detailed-guide.md     # 한국어 상세 설명 (아키텍처, 정책 로직, 운영 고려사항)
    └── 05-dds-variant.md        # (선택) VPD ↔ DDS 1:1 매핑 + 26ai 변형 사용법

더 깊이 알고 싶으면

  • docs/03-detailed-guide.md — 권한 매핑 모델, 정책 함수의 동적 SQL, Secure Context 사용 이유, 운영 시 주의점 등 전체 해설.
  • docs/05-dds-variant.md — Oracle AI Database 26ai 의 신기능 Deep Data Security 로 같은 데모를 재구현 (선택). 선언형 SQL Data Grant 가 VPD 의 PL/SQL 정책 함수를 어떻게 한 줄로 대체하는지 비교.

라이선스

MIT.