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>
268 lines
13 KiB
Markdown
268 lines
13 KiB
Markdown
# 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](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)
|
||
|
||
```bash
|
||
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. **prereq** — `sqlplus`, `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 으로 정책/뷰/유저 상태 점검
|
||
|
||
세부 단계만 따로 돌리려면:
|
||
|
||
```bash
|
||
./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](docs/05-dds-variant.md))
|
||
|
||
```bash
|
||
./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 레시피
|
||
|
||
```sql
|
||
-- 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](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](docs/03-detailed-guide.md) — 권한 매핑 모델, 정책
|
||
함수의 동적 SQL, Secure Context 사용 이유, 운영 시 주의점 등 전체 해설.
|
||
* [docs/05-dds-variant.md](docs/05-dds-variant.md) — Oracle AI Database 26ai
|
||
의 신기능 **Deep Data Security** 로 같은 데모를 재구현 (선택). 선언형 SQL
|
||
Data Grant 가 VPD 의 PL/SQL 정책 함수를 어떻게 한 줄로 대체하는지 비교.
|
||
|
||
---
|
||
|
||
## 라이선스
|
||
|
||
MIT.
|