#!/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 | xargs -r rm -rf # ── 소스 → 운영 반영 ── bash "$SRC/scripts/deploy-nas.sh" # ── data 디렉토리 보장 (볼륨 마운트 실패 방지) ── mkdir -p "$DST/data" "$DST/data/music" "$DST/data/stock" "$DST/data/blog" "$DST/data/realestate" # ── 변경된 서비스만 재빌드 (deployer 제외 — 자기 자신을 재빌드하면 스크립트 중단됨) ── cd "$DST" # Synology NAS에서 고아 컨테이너(해시 prefix 포함)가 이름 충돌을 일으키므로 # 패턴 매칭으로 관련 컨테이너를 모두 제거 후 재생성 SERVICES="lotto-backend stock-lab music-lab blog-lab realestate-lab travel-proxy lotto-frontend" for svc in $SERVICES; do ids=$(docker ps -a --filter "name=$svc" -q 2>/dev/null) if [ -n "$ids" ]; then echo "Removing containers matching '$svc': $ids" docker rm -f $ids 2>/dev/null || true fi done docker compose up -d --build backend travel-proxy stock-lab music-lab blog-lab realestate-lab frontend 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"; 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