diff --git a/scripts/deploy-nas.sh b/scripts/deploy-nas.sh index af70c9f..e0a23a0 100644 --- a/scripts/deploy-nas.sh +++ b/scripts/deploy-nas.sh @@ -1,6 +1,9 @@ #!/bin/bash set -euo pipefail +# ── 서비스 목록 (한 곳에서만 관리) ── +SERVICES="backend travel-proxy deployer stock-lab music-lab blog-lab realestate-lab agent-office nginx scripts" + # 1. 자동 감지: Docker 컨테이너 내부인가? if [ -d "/repo" ] && [ -d "/runtime" ]; then echo "Detected Docker Container environment." @@ -16,7 +19,7 @@ else # 환경변수가 없으면 현재 디렉토리를 SRC로 SRC="${REPO_PATH:-$(pwd)}" DST="${RUNTIME_PATH:-}" - + if [ -z "$DST" ]; then echo "Error: RUNTIME_PATH is not set. Please create .env file with RUNTIME_PATH defined." exit 1 @@ -28,11 +31,19 @@ echo "Target: $DST" cd "$SRC" -# 레포에서 운영으로 반영할 항목들만 복사/동기화 (필요한 것만 적기) -# backend, travel-proxy, deployer, nginx, scripts, docker-compose.yml, .env 등 +# 레포에서 운영으로 반영할 항목들만 복사/동기화 RSYNC_OPTS="-rl --delete --no-owner --no-group --exclude .git --exclude __pycache__ --exclude *.pyc --exclude data/" -for dir in backend travel-proxy deployer stock-lab music-lab blog-lab realestate-lab agent-office nginx scripts; do +# 파일 권한 설정값 +DEPLOY_USER="bgg8988" +DEPLOY_GROUP="users" +DEPLOY_MODE="755" + +for dir in $SERVICES; do + # rsync 전 대상 디렉토리 권한 확보 (Docker root 소유 파일 대응) + if [ -d "$DST/$dir" ]; then + chmod -R u+rwX "$DST/$dir/" 2>/dev/null || true + fi rsync $RSYNC_OPTS "$SRC/$dir/" "$DST/$dir/" || { rc=$? if [ $rc -ne 23 ]; then @@ -52,13 +63,8 @@ rsync -rl --no-owner --no-group "$SRC/docker-compose.yml" "$DST/docker-compose.y } # 파일 권한 설정 — bgg8988:users 755 -# 호스트(bgg8988)에서는 본인 소유 파일만 변경 가능, deployer 컨테이너(root)에서는 전부 가능 -DEPLOY_USER="bgg8988" -DEPLOY_GROUP="users" -DEPLOY_MODE="755" - echo "Setting ownership ${DEPLOY_USER}:${DEPLOY_GROUP} and mode ${DEPLOY_MODE}..." -for dir in backend travel-proxy deployer stock-lab music-lab blog-lab realestate-lab agent-office nginx scripts; do +for dir in $SERVICES; do chown -R "${DEPLOY_USER}:${DEPLOY_GROUP}" "$DST/$dir/" 2>/dev/null || true chmod -R "$DEPLOY_MODE" "$DST/$dir/" 2>/dev/null || true done diff --git a/scripts/deploy.sh b/scripts/deploy.sh index 2da32ce..beb0e99 100644 --- a/scripts/deploy.sh +++ b/scripts/deploy.sh @@ -5,6 +5,16 @@ set -euo pipefail exec 200>/tmp/deploy.lock flock -n 200 || { echo "Deploy already running, skipping"; exit 0; } +# ── 서비스 목록 (한 곳에서만 관리) ── +# docker compose 서비스명 (deployer 제외 — 자기 자신을 재빌드하면 스크립트 중단) +BUILD_TARGETS="backend travel-proxy stock-lab music-lab blog-lab realestate-lab agent-office frontend" +# 컨테이너 이름 (고아 정리용) +CONTAINER_NAMES="lotto-backend stock-lab music-lab blog-lab realestate-lab agent-office travel-proxy lotto-frontend" +# 헬스체크 대상 +HEALTH_ENDPOINTS="backend stock-lab travel-proxy music-lab blog-lab realestate-lab agent-office" +# data 디렉토리 +DATA_DIRS="music stock blog realestate agent-office" + # 1. 자동 감지: Docker 컨테이너 내부인가? if [ -d "/repo" ] && [ -d "/runtime" ]; then echo "Detected Docker Container environment." @@ -55,19 +65,20 @@ done bash "$SRC/scripts/deploy-nas.sh" # ── data 디렉토리 보장 (볼륨 마운트 실패 방지) ── -mkdir -p "$DST/data" "$DST/data/music" "$DST/data/stock" "$DST/data/blog" "$DST/data/realestate" "$DST/data/agent-office" +mkdir -p "$DST/data" +for d in $DATA_DIRS; do + mkdir -p "$DST/data/$d" +done -# ── 서비스 재빌드 (deployer 제외 — 자기 자신을 재빌드하면 스크립트 중단됨) ── +# ── 서비스 재빌드 (deployer 제외) ── cd "$DST" -BUILD_TARGETS="backend travel-proxy stock-lab music-lab blog-lab realestate-lab agent-office frontend" - -# 1) compose가 관리하는 컨테이너 정리 (deployer 제외) +# 1) compose가 관리하는 컨테이너 정리 docker compose stop $BUILD_TARGETS 2>/dev/null || true docker compose rm -f $BUILD_TARGETS 2>/dev/null || true # 2) Synology NAS 고아 컨테이너(해시 prefix 포함) 추가 정리 -for cname in lotto-backend stock-lab music-lab blog-lab realestate-lab agent-office travel-proxy lotto-frontend; do +for cname in $CONTAINER_NAMES; do docker rm -f "$cname" 2>/dev/null || true done @@ -80,10 +91,9 @@ echo "Waiting for services to start..." sleep 5 HEALTH_OK=true -# 컨테이너 내부에서는 서비스명 + 내부포트(8000)로 접근 -for endpoint in "http://backend:8000/health" "http://stock-lab:8000/health" "http://travel-proxy:8000/health" "http://music-lab:8000/health" "http://blog-lab:8000/health" "http://realestate-lab:8000/health" "http://agent-office:8000/health"; do - if ! curl -sf --max-time 10 --retry 2 --retry-delay 3 "$endpoint" > /dev/null 2>&1; then - echo "HEALTH_FAIL: $endpoint" +for svc in $HEALTH_ENDPOINTS; do + if ! curl -sf --max-time 10 --retry 2 --retry-delay 3 "http://$svc:8000/health" > /dev/null 2>&1; then + echo "HEALTH_FAIL: http://$svc:8000/health" HEALTH_OK=false fi done diff --git a/scripts/healthcheck.sh b/scripts/healthcheck.sh index 0e1f2c0..c60e95c 100644 --- a/scripts/healthcheck.sh +++ b/scripts/healthcheck.sh @@ -2,7 +2,6 @@ set -euo pipefail # NAS 내부 헬스체크용 (localhost 사용) -# 포트: backend(18000), travel-proxy(19000), frontend(8080) # Colors GREEN='\033[0;32m' @@ -16,41 +15,54 @@ echo "======================================" check_url() { local name="$1" local url="$2" - + # HTTP 상태 코드만 가져옴 (타임아웃 5초) status=$(curl -o /dev/null -s -w "%{http_code}" --max-time 5 "$url" || echo "FAIL") - + if [[ "$status" == "200" ]]; then echo -e "[${GREEN}OK${NC}] $name ($url) -> $status" else echo -e "[${RED}XX${NC}] $name ($url) -> $status" - # 하나라도 실패하면 exit 1 (CI/CD용) - # exit 1 fi } echo "" -echo "--- 1. Backend Service ---" +echo "--- 1. Backend (Lotto) ---" check_url "Backend Health" "http://localhost:18000/health" check_url "Lotto Latest" "http://localhost:18000/api/lotto/latest" check_url "Stats API" "http://localhost:18000/api/lotto/stats" echo "" -echo "--- 2. Travel Proxy Service ---" -# Travel Proxy는 Main.py에서 루트(/) 엔드포인트가 없을 수 있어서 regions 체크 -check_url "Travel Regions" "http://localhost:19000/api/travel/regions" - -echo "" -echo "--- 3. Stock Lab Service ---" +echo "--- 2. Stock Lab ---" check_url "Stock Health" "http://localhost:18500/health" check_url "Stock News" "http://localhost:18500/api/stock/news" check_url "Stock Indices" "http://localhost:18500/api/stock/indices" echo "" -echo "--- 4. Frontend (Nginx) ---" -# 외부 포트 8080으로 접속 테스트 +echo "--- 3. Music Lab ---" +check_url "Music Health" "http://localhost:18600/health" +check_url "Music Providers" "http://localhost:18600/api/music/providers" + +echo "" +echo "--- 4. Blog Lab ---" +check_url "Blog Health" "http://localhost:18700/health" + +echo "" +echo "--- 5. Realestate Lab ---" +check_url "Realestate Health" "http://localhost:18800/health" + +echo "" +echo "--- 6. Agent Office ---" +check_url "Agent Office Health" "http://localhost:18900/health" +check_url "Agent Office Agents" "http://localhost:18900/api/agent-office/agents" + +echo "" +echo "--- 7. Travel Proxy ---" +check_url "Travel Regions" "http://localhost:19000/api/travel/regions" + +echo "" +echo "--- 8. Frontend (Nginx) ---" check_url "Frontend Home" "http://localhost:8080" -# Nginx가 Backend로 잘 프록시하는지 체크 (실제 존재하는 api 호출) check_url "Nginx->Backend Proxy" "http://localhost:8080/api/lotto/latest" echo ""