- deploy.sh의 BUILD_TARGETS, 고아 컨테이너 정리, 헬스체크, data 디렉토리에 agent-office 추가 - .releases 오래된 백업 삭제 시 chmod u+rwX로 권한 확보 후 삭제 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
97 lines
3.1 KiB
Bash
97 lines
3.1 KiB
Bash
#!/bin/bash
|
|
set -euo pipefail
|
|
|
|
# ── 동시 배포 방지 (flock) ──
|
|
exec 200>/tmp/deploy.lock
|
|
flock -n 200 || { echo "Deploy already running, skipping"; exit 0; }
|
|
|
|
# 1. 자동 감지: Docker 컨테이너 내부인가?
|
|
if [ -d "/repo" ] && [ -d "/runtime" ]; then
|
|
echo "Detected Docker Container environment."
|
|
SRC="/repo"
|
|
DST="/runtime"
|
|
else
|
|
# 2. Host 환경: .env 로드 시도
|
|
if [ -f ".env" ]; then
|
|
echo "Loading .env file..."
|
|
set -a; source .env; set +a
|
|
fi
|
|
|
|
SRC="${REPO_PATH:-$(pwd)}"
|
|
DST="${RUNTIME_PATH:-/volume1/docker/webpage}"
|
|
|
|
if [ -z "$DST" ]; then
|
|
echo "Error: RUNTIME_PATH is not set."
|
|
exit 1
|
|
fi
|
|
fi
|
|
|
|
echo "Source: $SRC"
|
|
echo "Target: $DST"
|
|
|
|
git config --global --add safe.directory "$SRC"
|
|
|
|
cd "$SRC"
|
|
git fetch --all --prune
|
|
git pull --ff-only
|
|
|
|
# ── 릴리즈 백업 (롤백용) ──
|
|
TAG="$(date +%Y%m%d-%H%M%S)"
|
|
BACKUP_DIR="$DST/.releases/$TAG"
|
|
mkdir -p "$BACKUP_DIR.tmp"
|
|
rsync -a --delete \
|
|
--exclude ".releases" \
|
|
--exclude "data" \
|
|
"$DST/" "$BACKUP_DIR.tmp/"
|
|
mv "$BACKUP_DIR.tmp" "$BACKUP_DIR"
|
|
|
|
# 오래된 릴리즈 정리 (최근 5개만 유지, 권한 문제 우회)
|
|
ls -dt "$DST/.releases"/*/ 2>/dev/null | tail -n +6 | while read -r old_dir; do
|
|
chmod -R u+rwX "$old_dir" 2>/dev/null || true
|
|
rm -rf "$old_dir" 2>/dev/null || true
|
|
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"
|
|
|
|
# ── 서비스 재빌드 (deployer 제외 — 자기 자신을 재빌드하면 스크립트 중단됨) ──
|
|
cd "$DST"
|
|
|
|
BUILD_TARGETS="backend travel-proxy stock-lab music-lab blog-lab realestate-lab agent-office frontend"
|
|
|
|
# 1) compose가 관리하는 컨테이너 정리 (deployer 제외)
|
|
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
|
|
docker rm -f "$cname" 2>/dev/null || true
|
|
done
|
|
|
|
# 3) 재빌드 및 시작
|
|
docker compose up -d --build $BUILD_TARGETS
|
|
docker exec lotto-frontend nginx -s reload 2>/dev/null || true
|
|
|
|
# ── 배포 후 헬스체크 ──
|
|
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"
|
|
HEALTH_OK=false
|
|
fi
|
|
done
|
|
|
|
if [ "$HEALTH_OK" = false ]; then
|
|
echo "DEPLOY_FAIL: Some services failed health check. Backup available: $TAG"
|
|
exit 1
|
|
else
|
|
echo "DEPLOY_OK $TAG"
|
|
fi
|