fix(deploy): 서비스 목록 변수화 + rsync 전 권한 확보 + healthcheck 전서비스 추가

- deploy.sh / deploy-nas.sh: 서비스 목록을 변수로 통합하여 누락 방지
- deploy-nas.sh: rsync 전 chmod u+rwX로 Docker root 소유 파일 권한 확보
- healthcheck.sh: music-lab, blog-lab, realestate-lab, agent-office 추가

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-11 14:27:36 +09:00
parent 678440a2bd
commit cce84de8be
3 changed files with 63 additions and 35 deletions

View File

@@ -1,6 +1,9 @@
#!/bin/bash #!/bin/bash
set -euo pipefail set -euo pipefail
# ── 서비스 목록 (한 곳에서만 관리) ──
SERVICES="backend travel-proxy deployer stock-lab music-lab blog-lab realestate-lab agent-office nginx scripts"
# 1. 자동 감지: Docker 컨테이너 내부인가? # 1. 자동 감지: Docker 컨테이너 내부인가?
if [ -d "/repo" ] && [ -d "/runtime" ]; then if [ -d "/repo" ] && [ -d "/runtime" ]; then
echo "Detected Docker Container environment." echo "Detected Docker Container environment."
@@ -28,11 +31,19 @@ echo "Target: $DST"
cd "$SRC" 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/" 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/" || { rsync $RSYNC_OPTS "$SRC/$dir/" "$DST/$dir/" || {
rc=$? rc=$?
if [ $rc -ne 23 ]; then 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: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}..." 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 chown -R "${DEPLOY_USER}:${DEPLOY_GROUP}" "$DST/$dir/" 2>/dev/null || true
chmod -R "$DEPLOY_MODE" "$DST/$dir/" 2>/dev/null || true chmod -R "$DEPLOY_MODE" "$DST/$dir/" 2>/dev/null || true
done done

View File

@@ -5,6 +5,16 @@ set -euo pipefail
exec 200>/tmp/deploy.lock exec 200>/tmp/deploy.lock
flock -n 200 || { echo "Deploy already running, skipping"; exit 0; } 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 컨테이너 내부인가? # 1. 자동 감지: Docker 컨테이너 내부인가?
if [ -d "/repo" ] && [ -d "/runtime" ]; then if [ -d "/repo" ] && [ -d "/runtime" ]; then
echo "Detected Docker Container environment." echo "Detected Docker Container environment."
@@ -55,19 +65,20 @@ done
bash "$SRC/scripts/deploy-nas.sh" bash "$SRC/scripts/deploy-nas.sh"
# ── data 디렉토리 보장 (볼륨 마운트 실패 방지) ── # ── 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" cd "$DST"
BUILD_TARGETS="backend travel-proxy stock-lab music-lab blog-lab realestate-lab agent-office frontend" # 1) compose가 관리하는 컨테이너 정리
# 1) compose가 관리하는 컨테이너 정리 (deployer 제외)
docker compose stop $BUILD_TARGETS 2>/dev/null || true docker compose stop $BUILD_TARGETS 2>/dev/null || true
docker compose rm -f $BUILD_TARGETS 2>/dev/null || true docker compose rm -f $BUILD_TARGETS 2>/dev/null || true
# 2) Synology NAS 고아 컨테이너(해시 prefix 포함) 추가 정리 # 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 docker rm -f "$cname" 2>/dev/null || true
done done
@@ -80,10 +91,9 @@ echo "Waiting for services to start..."
sleep 5 sleep 5
HEALTH_OK=true HEALTH_OK=true
# 컨테이너 내부에서는 서비스명 + 내부포트(8000)로 접근 for svc in $HEALTH_ENDPOINTS; do
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 "http://$svc:8000/health" > /dev/null 2>&1; then
if ! curl -sf --max-time 10 --retry 2 --retry-delay 3 "$endpoint" > /dev/null 2>&1; then echo "HEALTH_FAIL: http://$svc:8000/health"
echo "HEALTH_FAIL: $endpoint"
HEALTH_OK=false HEALTH_OK=false
fi fi
done done

View File

@@ -2,7 +2,6 @@
set -euo pipefail set -euo pipefail
# NAS 내부 헬스체크용 (localhost 사용) # NAS 내부 헬스체크용 (localhost 사용)
# 포트: backend(18000), travel-proxy(19000), frontend(8080)
# Colors # Colors
GREEN='\033[0;32m' GREEN='\033[0;32m'
@@ -24,33 +23,46 @@ check_url() {
echo -e "[${GREEN}OK${NC}] $name ($url) -> $status" echo -e "[${GREEN}OK${NC}] $name ($url) -> $status"
else else
echo -e "[${RED}XX${NC}] $name ($url) -> $status" echo -e "[${RED}XX${NC}] $name ($url) -> $status"
# 하나라도 실패하면 exit 1 (CI/CD용)
# exit 1
fi fi
} }
echo "" echo ""
echo "--- 1. Backend Service ---" echo "--- 1. Backend (Lotto) ---"
check_url "Backend Health" "http://localhost:18000/health" check_url "Backend Health" "http://localhost:18000/health"
check_url "Lotto Latest" "http://localhost:18000/api/lotto/latest" check_url "Lotto Latest" "http://localhost:18000/api/lotto/latest"
check_url "Stats API" "http://localhost:18000/api/lotto/stats" check_url "Stats API" "http://localhost:18000/api/lotto/stats"
echo "" echo ""
echo "--- 2. Travel Proxy Service ---" echo "--- 2. Stock Lab ---"
# Travel Proxy는 Main.py에서 루트(/) 엔드포인트가 없을 수 있어서 regions 체크
check_url "Travel Regions" "http://localhost:19000/api/travel/regions"
echo ""
echo "--- 3. Stock Lab Service ---"
check_url "Stock Health" "http://localhost:18500/health" check_url "Stock Health" "http://localhost:18500/health"
check_url "Stock News" "http://localhost:18500/api/stock/news" check_url "Stock News" "http://localhost:18500/api/stock/news"
check_url "Stock Indices" "http://localhost:18500/api/stock/indices" check_url "Stock Indices" "http://localhost:18500/api/stock/indices"
echo "" echo ""
echo "--- 4. Frontend (Nginx) ---" echo "--- 3. Music Lab ---"
# 외부 포트 8080으로 접속 테스트 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" check_url "Frontend Home" "http://localhost:8080"
# Nginx가 Backend로 잘 프록시하는지 체크 (실제 존재하는 api 호출)
check_url "Nginx->Backend Proxy" "http://localhost:8080/api/lotto/latest" check_url "Nginx->Backend Proxy" "http://localhost:8080/api/lotto/latest"
echo "" echo ""