---
id: security-output-encoding-xss
title: Output Encoding & XSS 방지
category: Coding
status: draft
source_trust_level: B
verification_status: conceptual
created_at: 2026-05-09
updated_at: 2026-05-09
tags: [security, xss, encoding, csp, vibe-coding]
tech_stack: { language: "TypeScript / React / Express", applicable_to: ["Web"] }
applied_in: []
aliases: [HTML escape, dangerouslySetInnerHTML, CSP, sanitize]
---
# Output Encoding & XSS
> XSS 의 답은 **어떤 컨텍스트(HTML body / attribute / JS / URL / CSS)에 출력하느냐에 따라 인코딩이 달라진다**. React 가 디폴트로 안전하지만 `dangerouslySetInnerHTML` / `href` 등에서 구멍.
## 📖 핵심 개념
3 종류 XSS:
- **Stored**: 서버 DB 에 저장된 악성 스크립트 → 다른 사용자 화면에 실행.
- **Reflected**: URL / 입력 즉시 페이지에 echo.
- **DOM-based**: 클라이언트 JS 가 사용자 입력을 안전하지 않게 DOM 에 삽입.
## 💻 코드 패턴
### React 디폴트는 안전
```tsx
const name = userInput; //
return
{name}
;
// 자동 escape.
// data 안에 있으면 탈출
// ✅ 별도 element
// Client 에서: JSON.parse(document.getElementById('boot-data').textContent)
```
## 🤔 의사결정 기준
| 컨텍스트 | 인코딩 |
|---|---|
| HTML body 텍스트 | escape `<>&"'` |
| HTML attribute (`title="..."`) | escape + 따옴표로 감싸기 |
| URL attribute (`href`, `src`) | URL encode + 스킴 검증 |
| JS string literal | JSON.stringify + closing tag escape |
| CSS value | CSS escape |
| Markdown / Rich text 받을 때 | sanitizer (DOMPurify, sanitize-html) |
## ❌ 안티패턴
- **`dangerouslySetInnerHTML` 없이 sanitize**: 검증 안 된 HTML 직접 주입.
- **사용자 입력 URL 을 `href` 그대로**: `javascript:alert(1)` 가능.
- **`eval` / `Function` / `new Function`**: 사용자 입력이 들어가면 RCE.
- **innerHTML = userInput**: 클래식 XSS.
- **CSP 없이 inline script + nonce 안 씀**: 모든 inline 통과.
- **CSP 의 `unsafe-inline` / `unsafe-eval`**: CSP 의미 절반 잃음. 마이그레이션 가이드 따라 nonce / hash 로.
- **서버 검증만 + 클라이언트 출력 인코딩 누락**: client-side rendered 변수가 raw HTML. React 디폴트 safe 라 보통 OK 지만 React 외 환경 주의.
- **이메일/SMS 본문에 사용자 입력 그대로**: 다른 종류 인젝션 (SMTP header, link rewriting).
## 🤖 LLM 활용 힌트
- React 외 / SSR template / dangerouslySetInnerHTML / URL 컨텍스트에서 sanitizer 명시.
- CSP 헤더는 helmet 라이브러리 권장.
## 🔗 관련 문서
- [[Security_Input_Validation]]
- [[Security_CSRF_Patterns]]