기존 Stock/Music 에이전트 패턴을 따라 2개 신규 에이전트 도입. - Blog 에이전트 (10:00 매일): 트렌드 키워드 1개 자동 선택 → blog-lab 파이프라인 전체 (research→generate→market→review) 자동 실행 → 평가 점수와 본문 요약을 텔레그램 승인 요청으로 푸시 → 승인 시 published 전환, 거절 시 작업 종료 - Realestate 에이전트 (09:15 매일): realestate-lab 수집 트리거 → 신규 매칭 상위 5건 + 대시보드를 텔레그램 리포트 → 조회한 매칭은 자동 읽음 처리 - service_proxy: blog-lab/realestate-lab REST 호출 래퍼 추가 - agents 레지스트리 + DB 시드 + 스케줄러 3개 잡 등록 - docker-compose: agent-office에 BLOG_LAB_URL/REALESTATE_LAB_URL 주입 - README: 에이전트 구성 표 + 명령어 + 스케줄러 잡 정리 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
206 lines
6.4 KiB
YAML
206 lines
6.4 KiB
YAML
name: webpage
|
|
|
|
services:
|
|
backend:
|
|
build:
|
|
context: ./backend
|
|
args:
|
|
APP_VERSION: ${APP_VERSION:-dev}
|
|
container_name: lotto-backend
|
|
restart: unless-stopped
|
|
ports:
|
|
- "18000:8000"
|
|
environment:
|
|
- TZ=${TZ:-Asia/Seoul}
|
|
- LOTTO_ALL_URL=${LOTTO_ALL_URL:-https://smok95.github.io/lotto/results/all.json}
|
|
- LOTTO_LATEST_URL=${LOTTO_LATEST_URL:-https://smok95.github.io/lotto/results/latest.json}
|
|
volumes:
|
|
- ${RUNTIME_PATH}/data:/app/data
|
|
healthcheck:
|
|
test: ["CMD", "python", "-c", "import urllib.request; urllib.request.urlopen('http://localhost:8000/health')"]
|
|
interval: 30s
|
|
timeout: 5s
|
|
retries: 3
|
|
|
|
stock-lab:
|
|
build:
|
|
context: ./stock-lab
|
|
args:
|
|
APP_VERSION: ${APP_VERSION:-dev}
|
|
container_name: stock-lab
|
|
restart: unless-stopped
|
|
ports:
|
|
- "18500:8000"
|
|
environment:
|
|
- TZ=${TZ:-Asia/Seoul}
|
|
- WINDOWS_AI_SERVER_URL=${WINDOWS_AI_SERVER_URL:-http://192.168.0.5:8000}
|
|
- GEMINI_API_KEY=${GEMINI_API_KEY:-}
|
|
- GEMINI_MODEL=${GEMINI_MODEL:-gemini-1.5-flash}
|
|
- ADMIN_API_KEY=${ADMIN_API_KEY:-}
|
|
- ANTHROPIC_API_KEY=${ANTHROPIC_API_KEY:-}
|
|
- ANTHROPIC_MODEL=${ANTHROPIC_MODEL:-claude-haiku-4-5-20251001}
|
|
- LLM_PROVIDER=${LLM_PROVIDER:-claude}
|
|
- OLLAMA_URL=${OLLAMA_URL:-http://192.168.45.59:11435}
|
|
- OLLAMA_MODEL=${OLLAMA_MODEL:-qwen3:14b}
|
|
- CORS_ALLOW_ORIGINS=${CORS_ALLOW_ORIGINS:-http://localhost:3007,http://localhost:8080}
|
|
volumes:
|
|
- ${RUNTIME_PATH}/data/stock:/app/data
|
|
healthcheck:
|
|
test: ["CMD", "python", "-c", "import urllib.request; urllib.request.urlopen('http://localhost:8000/health')"]
|
|
interval: 30s
|
|
timeout: 5s
|
|
retries: 3
|
|
|
|
music-lab:
|
|
build:
|
|
context: ./music-lab
|
|
container_name: music-lab
|
|
restart: unless-stopped
|
|
ports:
|
|
- "18600:8000"
|
|
environment:
|
|
- TZ=${TZ:-Asia/Seoul}
|
|
- MUSIC_AI_SERVER_URL=${MUSIC_AI_SERVER_URL:-}
|
|
- SUNO_API_KEY=${SUNO_API_KEY:-}
|
|
- MUSIC_MEDIA_BASE=${MUSIC_MEDIA_BASE:-/media/music}
|
|
- CORS_ALLOW_ORIGINS=${CORS_ALLOW_ORIGINS:-http://localhost:3007,http://localhost:8080}
|
|
volumes:
|
|
- ${RUNTIME_PATH}/data/music:/app/data
|
|
healthcheck:
|
|
test: ["CMD", "python", "-c", "import urllib.request; urllib.request.urlopen('http://localhost:8000/health')"]
|
|
interval: 30s
|
|
timeout: 5s
|
|
retries: 3
|
|
|
|
blog-lab:
|
|
build:
|
|
context: ./blog-lab
|
|
container_name: blog-lab
|
|
restart: unless-stopped
|
|
ports:
|
|
- "18700:8000"
|
|
environment:
|
|
- TZ=${TZ:-Asia/Seoul}
|
|
- ANTHROPIC_API_KEY=${ANTHROPIC_API_KEY:-}
|
|
- NAVER_CLIENT_ID=${NAVER_CLIENT_ID:-}
|
|
- NAVER_CLIENT_SECRET=${NAVER_CLIENT_SECRET:-}
|
|
- CORS_ALLOW_ORIGINS=${CORS_ALLOW_ORIGINS:-http://localhost:3007,http://localhost:8080}
|
|
volumes:
|
|
- ${RUNTIME_PATH}/data/blog:/app/data
|
|
healthcheck:
|
|
test: ["CMD", "python", "-c", "import urllib.request; urllib.request.urlopen('http://localhost:8000/health')"]
|
|
interval: 30s
|
|
timeout: 5s
|
|
retries: 3
|
|
|
|
realestate-lab:
|
|
build:
|
|
context: ./realestate-lab
|
|
container_name: realestate-lab
|
|
restart: unless-stopped
|
|
ports:
|
|
- "18800:8000"
|
|
environment:
|
|
- TZ=${TZ:-Asia/Seoul}
|
|
- DATA_GO_KR_API_KEY=${DATA_GO_KR_API_KEY:-}
|
|
- CORS_ALLOW_ORIGINS=${CORS_ALLOW_ORIGINS:-http://localhost:3007,http://localhost:8080}
|
|
volumes:
|
|
- ${RUNTIME_PATH}/data/realestate:/app/data
|
|
healthcheck:
|
|
test: ["CMD", "python", "-c", "import urllib.request; urllib.request.urlopen('http://localhost:8000/health')"]
|
|
interval: 30s
|
|
timeout: 5s
|
|
retries: 3
|
|
|
|
agent-office:
|
|
build:
|
|
context: ./agent-office
|
|
container_name: agent-office
|
|
restart: unless-stopped
|
|
ports:
|
|
- "18900:8000"
|
|
environment:
|
|
- TZ=${TZ:-Asia/Seoul}
|
|
- CORS_ALLOW_ORIGINS=${CORS_ALLOW_ORIGINS:-http://localhost:3007,http://localhost:8080}
|
|
- STOCK_LAB_URL=http://stock-lab:8000
|
|
- MUSIC_LAB_URL=http://music-lab:8000
|
|
- BLOG_LAB_URL=http://blog-lab:8000
|
|
- REALESTATE_LAB_URL=http://realestate-lab:8000
|
|
- TELEGRAM_BOT_TOKEN=${TELEGRAM_BOT_TOKEN:-}
|
|
- TELEGRAM_CHAT_ID=${TELEGRAM_CHAT_ID:-}
|
|
- TELEGRAM_WEBHOOK_URL=${TELEGRAM_WEBHOOK_URL:-}
|
|
volumes:
|
|
- ${RUNTIME_PATH:-.}/data/agent-office:/app/data
|
|
depends_on:
|
|
- stock-lab
|
|
- music-lab
|
|
- blog-lab
|
|
- realestate-lab
|
|
healthcheck:
|
|
test: ["CMD", "python", "-c", "import urllib.request; urllib.request.urlopen('http://localhost:8000/health')"]
|
|
interval: 30s
|
|
timeout: 5s
|
|
retries: 3
|
|
|
|
travel-proxy:
|
|
build: ./travel-proxy
|
|
container_name: travel-proxy
|
|
restart: unless-stopped
|
|
user: "${PUID}:${PGID}"
|
|
ports:
|
|
- "19000:8000"
|
|
environment:
|
|
- TZ=${TZ:-Asia/Seoul}
|
|
- TRAVEL_ROOT=${TRAVEL_ROOT:-/data/travel}
|
|
- TRAVEL_THUMB_ROOT=${TRAVEL_THUMB_ROOT:-/data/thumbs}
|
|
- TRAVEL_MEDIA_BASE=${TRAVEL_MEDIA_BASE:-/media/travel}
|
|
- TRAVEL_CACHE_TTL=${TRAVEL_CACHE_TTL:-300}
|
|
- CORS_ALLOW_ORIGINS=${CORS_ALLOW_ORIGINS:-http://localhost:3007,http://localhost:8080}
|
|
volumes:
|
|
- ${PHOTO_PATH}:/data/travel:ro
|
|
- ${RUNTIME_PATH}/travel-thumbs:/data/thumbs:rw
|
|
healthcheck:
|
|
test: ["CMD", "python", "-c", "import urllib.request; urllib.request.urlopen('http://localhost:8000/health')"]
|
|
interval: 30s
|
|
timeout: 5s
|
|
retries: 3
|
|
|
|
frontend:
|
|
image: nginx:alpine
|
|
container_name: lotto-frontend
|
|
restart: unless-stopped
|
|
depends_on:
|
|
- music-lab
|
|
- blog-lab
|
|
- realestate-lab
|
|
ports:
|
|
- "8080:80"
|
|
volumes:
|
|
- ${FRONTEND_PATH}:/usr/share/nginx/html:ro
|
|
- ${RUNTIME_PATH}/nginx/default.conf:/etc/nginx/conf.d/default.conf:ro
|
|
- ${PHOTO_PATH}:/data/travel:ro
|
|
- ${RUNTIME_PATH}/travel-thumbs:/data/thumbs:ro
|
|
- ${RUNTIME_PATH}/data/music:/data/music:ro
|
|
extra_hosts:
|
|
- "host.docker.internal:host-gateway"
|
|
healthcheck:
|
|
test: ["CMD", "wget", "-q", "--spider", "http://localhost:80/"]
|
|
interval: 30s
|
|
timeout: 5s
|
|
retries: 3
|
|
|
|
deployer:
|
|
build: ./deployer
|
|
container_name: webpage-deployer
|
|
restart: unless-stopped
|
|
ports:
|
|
- "127.0.0.1:19010:9000"
|
|
environment:
|
|
- TZ=${TZ:-Asia/Seoul}
|
|
- WEBHOOK_SECRET=${WEBHOOK_SECRET}
|
|
volumes:
|
|
- ${REPO_PATH}:/repo:rw
|
|
- ${RUNTIME_PATH}:/runtime:rw
|
|
- ${RUNTIME_PATH}/scripts:/scripts:ro
|
|
- /var/run/docker.sock:/var/run/docker.sock
|