Email 관련 업데이트

This commit is contained in:
0914eagle 2025-08-19 18:13:35 +09:00
parent dc37907546
commit 80613a02bf
3 changed files with 376 additions and 34 deletions

View File

@ -95,42 +95,43 @@
## 3. skill-email DB 연결 (희재)
### 3.1 DB 연결 설정
- [ ] psycopg2 의존성 추가
- [ ] .env 파일 수정
- [ ] TOKEN_PROVIDER=database
- [ ] POSTGRES_CONNECTION_STRING 설정
- [ ] 데이터베이스 연결 테스트 스크립트 작성
- [ ] 연결 확인
- [x] psycopg2 의존성 추가
- [x] .env 파일 수정
- [x] TOKEN_PROVIDER=database
- [x] POSTGRES_CONNECTION_STRING 설정
- [x] 데이터베이스 연결 테스트 스크립트 작성
- [x] 연결 확인
### 3.2 DBCredentialsProvider 구현
- [ ] services/db_credentials_provider.py 생성
- [ ] __init__ 메서드 구현
- [ ] DB 연결 초기화
- [ ] 연결 풀 설정
- [ ] get_credentials 메서드 구현
- [ ] SQL 쿼리 작성
- [ ] 토큰 조회 로직
- [ ] JSON 파싱
- [ ] Credentials 객체 생성
- [ ] refresh_token 메서드 구현
- [ ] 토큰 갱신 로직
- [ ] DB 업데이트
- [ ] 에러 처리 추가
- [ ] 토큰 없음 처리
- [ ] DB 연결 실패 처리
- [ ] 파싱 에러 처리
- [x] services/db_credentials_provider.py 생성
- [x] __init__ 메서드 구현
- [x] DB 연결 초기화
- [x] 연결 풀 설정
- [x] get_credentials 메서드 구현
- [x] SQL 쿼리 작성
- [x] 토큰 조회 로직
- [x] JSON 파싱
- [x] Credentials 객체 생성
- [x] oauth_config에서 client_id/secret 가져오기
- [x] save_credentials 메서드 구현 (토큰 업데이트용)
- [x] 토큰 갱신 로직
- [x] DB 업데이트
- [x] 에러 처리 추가
- [x] 토큰 없음 처리
- [x] DB 연결 실패 처리
- [x] 파싱 에러 처리
### 3.3 기존 코드 수정
- [ ] main.py 수정
- [ ] FileCredentialsProvider import 제거
- [ ] DBCredentialsProvider import 추가
- [ ] Provider 초기화 코드 변경
- [ ] 환경변수 체크 로직 수정
- [ ] 테스트 코드 작성
- [ ] 로컬 테스트
- [x] main.py 수정
- [x] FileCredentialsProvider 유지 (조건부 사용)
- [x] DBCredentialsProvider import 추가
- [x] Provider 초기화 코드 변경 (TOKEN_PROVIDER 환경변수 기반)
- [x] 환경변수 체크 로직 수정
- [x] 테스트 코드 작성
- [x] 로컬 테스트
### 3.4 배포 준비
- [ ] requirements.txt 업데이트
- [x] requirements.txt 업데이트
- [ ] Dockerfile 확인
- [ ] 빌드 테스트
- [ ] 커밋 및 푸시

View File

@ -282,14 +282,15 @@ CREATE TABLE gmail_audit_logs (
## 6. 구현 단계
### Phase 1: 백엔드 기초
- [ ] 필요 테이블 생성
- [x] 필요 테이블 생성
- robeing_stats 테이블 (레벨 관리)
- gmail_audit_logs 테이블 (감사 로그)
- gmail_tokens에 is_equipped, equipped_to 컬럼 추가
- [ ] skill-email DB 연결 코드 작성
- [x] skill-email DB 연결 코드 작성 (희재)
- FileCredentialsProvider → DBCredentialsProvider 전환
- PostgreSQL 연결 설정 추가 (192.168.219.45)
- [ ] robeing-monitor 서비스 구축
- PostgreSQL 연결 설정 추가 (localhost:5433 SSH 터널)
- oauth_config 컬럼에서 client_id/secret 가져오기 구현
- [x] robeing-monitor 서비스 구축
- 기존 robeing-state-service 확장
- 포트 9024로 변경
- 아이템 관리 API 추가

View File

@ -0,0 +1,340 @@
# skill-email DB 연결 구현 완료 보고서
## 작업일: 2025-08-19
## 작업자: 희재
## 상태: 완료
---
## 1. 작업 개요
### 목적
skill-email 서비스가 Gmail OAuth 토큰을 파일이 아닌 PostgreSQL 데이터베이스에서 직접 조회하도록 변경
### 배경
- 기존: 파일 기반 토큰 관리 (FileCredentialsProvider)
- 변경: PostgreSQL gmail_tokens 테이블 기반 관리 (DBCredentialsProvider)
- 이유: 중앙집중식 토큰 관리, 보안 강화, 다중 서비스 통합
---
## 2. 구현 내용
### 2.1 의존성 추가
**파일**: `/home/heejae/skill-email/requirements.txt`
```python
psycopg2-binary>=2.9.9 # PostgreSQL 연결용
```
### 2.2 환경 설정
**파일**: `/home/heejae/skill-email/.env`
```bash
# Skill Email Configuration
PORT=8501
# Token Provider Configuration
TOKEN_PROVIDER=database
# PostgreSQL Connection (SSH tunnel)
POSTGRES_CONNECTION_STRING=postgresql://robeings:robeings@localhost:5433/auth_db
# Service Name
SERVICE_NAME=skill-email
# Log Level
LOG_LEVEL=INFO
```
### 2.3 DBCredentialsProvider 구현
**파일**: `/home/heejae/skill-email/services/db_credentials_provider.py`
#### 주요 기능
1. **연결 풀 관리**
- SimpleConnectionPool 사용 (최소 1개, 최대 5개 연결)
- 연결 재사용으로 성능 최적화
2. **get_credentials() 메서드**
```python
def get_credentials(self, user_id: str) -> Optional[Credentials]:
# gmail_tokens 테이블에서 조회
# is_equipped=true인 토큰만 조회
# token_data + oauth_config 컬럼 모두 활용
```
- gmail_tokens 테이블 조회
- token_data 컬럼: access_token, refresh_token
- oauth_config 컬럼: client_id, client_secret, token_uri
- Google Credentials 객체로 변환
3. **save_credentials() 메서드**
```python
def save_credentials(self, user_id: str, creds: Credentials) -> bool:
# 갱신된 토큰을 DB에 저장
# access_token 필드로 저장 (DB 구조 호환)
```
4. **check_token_exists() 메서드**
```python
def check_token_exists(self, user_id: str) -> bool:
# 사용자의 장착된 토큰 존재 여부 확인
```
#### 특별 처리 사항
- **토큰 필드 매핑**: DB의 `access_token` ↔ Credentials의 `token`
- **oauth_config 파싱**: JSON 형식의 client credentials 추출
- **is_equipped 필터**: 장착된 토큰만 조회
### 2.4 main.py 수정
**파일**: `/home/heejae/skill-email/main.py`
#### 변경 내용
```python
# 전역 credentials provider
credentials_provider = None
def get_gmail_service() -> GmailService:
global credentials_provider
if credentials_provider is None:
if TOKEN_PROVIDER == "database":
if not POSTGRES_CONNECTION_STRING:
logger.error("POSTGRES_CONNECTION_STRING not set")
raise ValueError("Database connection string required")
logger.info("Using database credentials provider")
credentials_provider = DBCredentialsProvider(POSTGRES_CONNECTION_STRING)
else:
logger.info("Using file credentials provider")
credentials_provider = FileCredentialsProvider(TOKEN_BASE)
return GmailService(credentials_provider)
```
- TOKEN_PROVIDER 환경변수로 File/DB 모드 선택
- 의존성 주입 패턴 사용 (Depends)
- 기존 FileCredentialsProvider 호환성 유지
---
## 3. 데이터베이스 구조
### gmail_tokens 테이블
```sql
CREATE TABLE gmail_tokens (
id SERIAL PRIMARY KEY,
user_id VARCHAR(100),
robeing_id VARCHAR(50),
token_data JSONB, -- access_token, refresh_token
oauth_config JSONB, -- client_id, client_secret, token_uri
scopes TEXT[],
metadata JSONB,
expiry TIMESTAMP,
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
is_equipped BOOLEAN DEFAULT FALSE,
equipped_to VARCHAR(50)
);
```
### 데이터 예시
```json
// token_data
{
"token_type": "Bearer",
"access_token": "ya29.a0AS3H6NxLyI1Ss1VJ-_WUkpR...",
"refresh_token": "1//0eZrT..."
}
// oauth_config
{
"client_id": "1044056803209-0h8n5kcl22rvl740mdgpejp58v27mlro.apps.googleusercontent.com",
"client_secret": "GOCSPX-...",
"token_uri": "https://oauth2.googleapis.com/token"
}
```
---
## 4. 테스트 결과
### 4.1 연결 테스트
```bash
cd /home/heejae/skill-email && python3 test_db.py
```
- ✅ PostgreSQL 연결 성공
- ✅ gmail_tokens 테이블 접근 가능
- ✅ 3개 토큰 확인 (heejae, test, unknown)
### 4.2 Provider 테스트
```bash
cd /home/heejae/skill-email && python3 test_integration.py
```
- ✅ DBCredentialsProvider 생성 성공
- ✅ heejae 토큰 로드 성공
- Access Token: ya29.a0AS3H6NxLyI1Ss1VJ-_WUkpR...
- Refresh Token: 있음
- Client ID: 1044056803209-0h8n5kcl22rvl740...
- Client Secret: 있음
- Token URI: https://oauth2.googleapis.com/token
- Scopes: ['https://www.googleapis.com/auth/gmail.send']
### 4.3 통합 테스트 스크립트
**파일**: `/home/heejae/skill-email/test_integration.py`
- DBCredentialsProvider 테스트
- GmailService 통합 테스트
- 토큰 존재 여부 확인
---
## 5. SSH 터널 설정
### 연결 구성
```bash
# SSH 터널 생성 (51123 서버 → 51124 서버)
sshpass -p "19800508" ssh -f -N -L 5433:localhost:5432 admin@124.55.18.179 -p 51123
```
- 로컬 포트 5433 → 원격 PostgreSQL 5432
- 51123 서버 (124.55.18.179)의 PostgreSQL 접근
- auth_db 데이터베이스 사용
---
## 6. 문제 해결
### 6.1 토큰 필드 불일치
- **문제**: DB는 `access_token`, Credentials는 `token` 사용
- **해결**:
```python
token=token_data.get('access_token') or token_data.get('token')
```
### 6.2 is_equipped 필터
- **문제**: 모든 토큰이 is_equipped=false 상태
- **해결**:
```sql
UPDATE gmail_tokens SET is_equipped=true WHERE user_id='heejae'
```
### 6.3 oauth_config 누락
- **문제**: client_id, client_secret이 없어 API 호출 불가
- **해결**: oauth_config 컬럼에서 추가로 조회
```python
cur.execute("""
SELECT token_data, oauth_config
FROM gmail_tokens
WHERE user_id = %s AND is_equipped = true
""")
```
### 6.4 인코딩 오류
- **문제**: .env 파일 UTF-8 디코딩 오류
- **해결**: .env 파일 재작성
---
## 7. 코드 구조
```
/home/heejae/skill-email/
├── .env # 환경변수 설정
├── requirements.txt # 의존성 목록
├── main.py # FastAPI 앱 (수정됨)
├── services/
│ ├── gmail_service.py # 기존 Gmail 서비스
│ ├── db_credentials_provider.py # 새로 추가된 DB Provider
│ └── ...
├── test_db.py # DB 연결 테스트
├── test_integration.py # 통합 테스트
└── ...
```
---
## 8. 보안 고려사항
1. **토큰 암호화**: DB에 평문 저장 (추후 암호화 고려)
2. **연결 풀**: 최대 5개 연결로 제한
3. **에러 처리**: 토큰 정보 로그에 노출 방지
4. **SSH 터널**: PostgreSQL 직접 노출 방지
---
## 9. 성능 최적화
1. **연결 풀링**: SimpleConnectionPool 사용
2. **JSON 파싱**: 한 번만 파싱하여 재사용
3. **로깅 레벨**: INFO로 설정 (프로덕션에서는 WARNING 권장)
---
## 10. 남은 작업
### 즉시 필요
- [ ] 51124 서버 배포
- [ ] Docker 이미지 빌드 및 테스트
- [ ] rb8001과 통합 테스트
### 추후 개선
- [ ] 토큰 암호화 저장
- [ ] 토큰 자동 갱신 로직
- [ ] 연결 풀 모니터링
- [ ] 캐싱 레이어 추가
---
## 11. 호환성
### 기존 시스템과 호환
- ✅ FileCredentialsProvider 유지 (TOKEN_PROVIDER=file)
- ✅ 기존 API 인터페이스 변경 없음
- ✅ 동일한 GmailService 사용
### 새로운 기능
- ✅ 중앙집중식 토큰 관리
- ✅ 다중 robeing 지원 (equipped_to 필드)
- ✅ 감사 로그 준비 (gmail_audit_logs 테이블과 연동 가능)
---
## 12. 검증 체크리스트
- [x] PostgreSQL 연결 성공
- [x] gmail_tokens 테이블 조회
- [x] 토큰 데이터 파싱
- [x] Credentials 객체 생성
- [x] oauth_config 통합
- [x] 에러 처리 동작
- [x] 연결 풀 정상 작동
- [x] 환경변수 기반 Provider 선택
- [ ] 실제 Gmail API 호출
- [ ] 토큰 갱신 플로우
---
## 13. 참고 명령어
```bash
# DB 연결 테스트
cd /home/heejae/skill-email && python3 test_db.py
# Provider 테스트
cd /home/heejae/skill-email && python3 test_integration.py
# SSH 터널 재연결
sshpass -p "19800508" ssh -f -N -L 5433:localhost:5432 admin@124.55.18.179 -p 51123
# 토큰 상태 확인
python3 -c "
import psycopg2
conn = psycopg2.connect('postgresql://robeings:robeings@localhost:5433/auth_db')
cur = conn.cursor()
cur.execute('SELECT user_id, is_equipped FROM gmail_tokens')
for row in cur.fetchall():
print(f'{row[0]}: equipped={row[1]}')
"
```
---
**작업 완료: 2025-08-19**
**다음 단계: rb8001 통합 (4번 작업)**