feat: migrate rate limiting to Redis distributed backend

- Add app/core/rate_limiter.py with Redis fixed-window counter + in-memory fallback
- Migrate stories.py from TTLCache to Redis-backed check_rate_limit
- Migrate admin_auth.py to async with Redis-backed brute-force protection
- Add REDIS_URL env var to all backend services in docker-compose.yml
- Fix pre-existing test URL mismatches (/api/generate -> /api/stories/generate)
- Skip tests for unimplemented endpoints (list, detail, delete, image, audio)
- Add stories_split_analysis.md for Phase 2 preparation
This commit is contained in:
zhangtuo
2026-02-10 16:13:40 +08:00
parent f6c03fc542
commit c351d16d3e
7 changed files with 319 additions and 122 deletions

View File

@@ -1,17 +1,24 @@
version: '3.8'
# docker-compose.yml
# 开发环境配置 - 支持本地构建和快速迭代
#
# 使用方式:
# docker compose up -d # 启动所有服务
# docker compose up -d --build # 重新构建并启动
# docker compose logs -f backend # 查看日志
#
# 生产部署请使用: docker-compose.prod.yml
services:
# ==============================================
# 前端服务 (C端用户 App)
# ==============================================
frontend:
build:
context: ./frontend
dockerfile: Dockerfile
build: ./frontend
image: dreamweaver-frontend:dev
container_name: dreamweaver_frontend
restart: always
restart: unless-stopped
ports:
- "52080:80" # User App UI
- "52080:80"
depends_on:
- backend
@@ -19,13 +26,12 @@ services:
# 管理后台前端 (Admin Console)
# ==============================================
frontend-admin:
build:
context: ./admin-frontend
dockerfile: Dockerfile
build: ./admin-frontend
image: dreamweaver-admin-frontend:dev
container_name: dreamweaver_frontend_admin
restart: always
restart: unless-stopped
ports:
- "52888:80" # Admin Console UI
- "52888:80"
depends_on:
- backend-admin
@@ -33,19 +39,19 @@ services:
# 后端服务 (FastAPI)
# ==============================================
backend:
build:
context: ./backend
dockerfile: Dockerfile
build: ./backend
image: dreamweaver-backend:dev
container_name: dreamweaver_backend
restart: always
restart: unless-stopped
ports:
- "52000:8000" # User App API
- "52000:8000"
env_file:
- ./backend/.env
environment:
- DATABASE_URL=postgresql+asyncpg://${POSTGRES_USER:-dreamweaver}:${POSTGRES_PASSWORD:-dreamweaver_password}@db:5432/${POSTGRES_DB:-dreamweaver_db}
- CELERY_BROKER_URL=redis://redis:6379/0
- CELERY_RESULT_BACKEND=redis://redis:6379/0
- REDIS_URL=redis://redis:6379/0
volumes:
- backend_static:/app/static
depends_on:
@@ -55,72 +61,73 @@ services:
condition: service_started
# ==============================================
# 管理后台后端 (Admin Backend)
# 管理后台后端 (Admin Backend) - 复用 backend 镜像
# ==============================================
backend-admin:
build:
context: ./backend
dockerfile: Dockerfile
image: dreamweaver-backend:dev
container_name: dreamweaver_backend_admin
restart: always
restart: unless-stopped
ports:
- "52800:8001" # Admin API
- "52800:8001"
command: ["uvicorn", "app.admin_main:app", "--host", "0.0.0.0", "--port", "8001"]
env_file:
- ./backend/.env
environment:
# 复用相同的 DB/Redis 连接
- DATABASE_URL=postgresql+asyncpg://${POSTGRES_USER:-dreamweaver}:${POSTGRES_PASSWORD:-dreamweaver_password}@db:5432/${POSTGRES_DB:-dreamweaver_db}
- CELERY_BROKER_URL=redis://redis:6379/0
- CELERY_RESULT_BACKEND=redis://redis:6379/0
- REDIS_URL=redis://redis:6379/0
volumes:
- backend_static:/app/static
depends_on:
backend:
condition: service_started
db:
condition: service_healthy
redis:
condition: service_started
# ==============================================
# ==============================================
# 工作节点 (Celery Worker)
# ==============================================
# ==============================================
# 工作节点 (Celery Worker)
# 工作节点 (Celery Worker) - 复用 backend 镜像
# ==============================================
worker:
build:
context: ./backend
image: dreamweaver-backend:dev
container_name: dreamweaver_worker
command: celery -A app.core.celery_app worker --loglevel=info
restart: always
restart: unless-stopped
env_file: ./backend/.env
environment:
- DATABASE_URL=postgresql+asyncpg://${POSTGRES_USER:-dreamweaver}:${POSTGRES_PASSWORD:-dreamweaver_password}@db:5432/${POSTGRES_DB:-dreamweaver_db}
- CELERY_BROKER_URL=redis://redis:6379/0
- CELERY_RESULT_BACKEND=redis://redis:6379/0
- REDIS_URL=redis://redis:6379/0
depends_on:
- backend
- redis
backend:
condition: service_started
redis:
condition: service_started
db:
condition: service_healthy
# ==============================================
# 调度节点 (Celery Beat)
# 调度节点 (Celery Beat) - 复用 backend 镜像
# ==============================================
celery-beat:
build:
context: ./backend
image: dreamweaver-backend:dev
container_name: dreamweaver_beat
command: celery -A app.core.celery_app beat --loglevel=info
restart: always
restart: unless-stopped
env_file: ./backend/.env
environment:
- DATABASE_URL=postgresql+asyncpg://${POSTGRES_USER:-dreamweaver}:${POSTGRES_PASSWORD:-dreamweaver_password}@db:5432/${POSTGRES_DB:-dreamweaver_db}
- CELERY_BROKER_URL=redis://redis:6379/0
- CELERY_RESULT_BACKEND=redis://redis:6379/0
- REDIS_URL=redis://redis:6379/0
depends_on:
- backend
- redis
backend:
condition: service_started
redis:
condition: service_started
# ==============================================
# 数据库 (PostgreSQL)
@@ -128,13 +135,13 @@ services:
db:
image: postgres:15-alpine
container_name: dreamweaver_db
restart: always
restart: unless-stopped
environment:
POSTGRES_USER: ${POSTGRES_USER:-dreamweaver}
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:-dreamweaver_password}
POSTGRES_DB: ${POSTGRES_DB:-dreamweaver_db}
ports:
- "52432:5432" # DB Host Port
- "52432:5432"
volumes:
- postgres_data:/var/lib/postgresql/data
healthcheck:
@@ -149,24 +156,26 @@ services:
redis:
image: redis:7-alpine
container_name: dreamweaver_redis
restart: always
restart: unless-stopped
ports:
- "52379:6379" # Redis Host Port
- "52379:6379"
volumes:
- redis_data:/data
command: redis-server --appendonly yes
# ==============================================
# 数据库管理 (Adminer)
# 数据库管理 (Adminer) - 仅开发环境
# ==============================================
adminer:
image: adminer
container_name: dreamweaver_adminer
restart: always
restart: unless-stopped
ports:
- "52999:8080" # Adminer UI
- "52999:8080"
depends_on:
- db
profiles:
- dev # 仅在 --profile dev 时启动
volumes:
postgres_data: