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:
@@ -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
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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 ""
|
||||||
|
|||||||
Reference in New Issue
Block a user