mirror of
https://github.com/uprightbass360/AzerothCore-RealmMaster.git
synced 2026-01-13 00:58:34 +00:00
199 lines
6.0 KiB
Bash
Executable File
199 lines
6.0 KiB
Bash
Executable File
#!/bin/bash
|
||
# Project: ac-compose
|
||
set -e
|
||
|
||
# Simple profile-aware deploy + health check for profiles-verify/docker-compose.yml
|
||
|
||
BLUE='\033[0;34m'; GREEN='\033[0;32m'; YELLOW='\033[1;33m'; RED='\033[0;31m'; NC='\033[0m'
|
||
info(){ echo -e "${BLUE}ℹ️ $*${NC}"; }
|
||
ok(){ echo -e "${GREEN}✅ $*${NC}"; }
|
||
warn(){ echo -e "${YELLOW}⚠️ $*${NC}"; }
|
||
err(){ echo -e "${RED}❌ $*${NC}"; }
|
||
|
||
PROJECT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")/.." && pwd)"
|
||
COMPOSE_FILE="$PROJECT_DIR/docker-compose.yml"
|
||
ENV_FILE=""
|
||
PROFILES=(db services-standard client-data modules tools)
|
||
SKIP_DEPLOY=false
|
||
QUICK=false
|
||
|
||
usage(){
|
||
cat <<EOF
|
||
Usage: $0 [--profiles p1,p2,...] [--env-file path] [--skip-deploy] [--quick]
|
||
Default profiles: db,services-standard,client-data,modules,tools
|
||
Examples:
|
||
$0 --profiles db,services-standard,client-data --env-file ./services.env
|
||
$0 --profiles db,services-playerbots,client-data-bots,modules,tools
|
||
EOF
|
||
}
|
||
|
||
while [[ $# -gt 0 ]]; do
|
||
case "$1" in
|
||
--profiles) IFS=',' read -r -a PROFILES <<< "$2"; shift 2;;
|
||
--env-file) ENV_FILE="$2"; shift 2;;
|
||
--skip-deploy) SKIP_DEPLOY=true; shift;;
|
||
--quick) QUICK=true; shift;;
|
||
-h|--help) usage; exit 0;;
|
||
*) err "Unknown arg: $1"; usage; exit 1;;
|
||
esac
|
||
done
|
||
|
||
resolve_project_name(){
|
||
local env_path
|
||
if [ -n "$ENV_FILE" ]; then
|
||
env_path="$ENV_FILE"
|
||
else
|
||
env_path="$(dirname "$COMPOSE_FILE")/.env"
|
||
fi
|
||
local raw_name=""
|
||
if [ -f "$env_path" ]; then
|
||
raw_name="$(grep -E '^COMPOSE_PROJECT_NAME=' "$env_path" | tail -n1 | cut -d'=' -f2-)"
|
||
fi
|
||
if [ -z "$raw_name" ]; then
|
||
raw_name="acore-compose"
|
||
fi
|
||
local sanitized
|
||
sanitized="$(echo "$raw_name" | tr '[:upper:]' '[:lower:]')"
|
||
sanitized="${sanitized// /-}"
|
||
sanitized="$(echo "$sanitized" | tr -cd 'a-z0-9_-')"
|
||
if [[ -z "$sanitized" ]]; then
|
||
sanitized="acore-compose"
|
||
elif [[ ! "$sanitized" =~ ^[a-z0-9] ]]; then
|
||
sanitized="ac${sanitized}"
|
||
fi
|
||
echo "$sanitized"
|
||
}
|
||
|
||
run_compose(){
|
||
local compose_args=()
|
||
local project_name
|
||
project_name="$(resolve_project_name)"
|
||
compose_args+=(--project-name "$project_name")
|
||
if [ -n "$ENV_FILE" ]; then
|
||
compose_args+=(--env-file "$ENV_FILE")
|
||
fi
|
||
compose_args+=(-f "$COMPOSE_FILE")
|
||
if [ "$(read_env_value MYSQL_EXPOSE_PORT "0")" = "1" ]; then
|
||
local extra_file
|
||
extra_file="$(dirname "$COMPOSE_FILE")/docker-compose.mysql-expose.yml"
|
||
if [ -f "$extra_file" ]; then
|
||
compose_args+=(-f "$extra_file")
|
||
else
|
||
warn "MYSQL_EXPOSE_PORT=1 but ${extra_file} missing; skipping port exposure override."
|
||
fi
|
||
fi
|
||
docker compose "${compose_args[@]}" "$@"
|
||
}
|
||
|
||
env_file_path(){
|
||
if [ -n "$ENV_FILE" ]; then
|
||
echo "$ENV_FILE"
|
||
else
|
||
echo "$(dirname "$COMPOSE_FILE")/.env"
|
||
fi
|
||
}
|
||
|
||
read_env_value(){
|
||
local key="$1" default="${2:-}"
|
||
local env_path value
|
||
env_path="$(env_file_path)"
|
||
if [ -f "$env_path" ]; then
|
||
value="$(grep -E "^${key}=" "$env_path" | tail -n1 | cut -d'=' -f2- | tr -d '\r')"
|
||
fi
|
||
if [ -z "$value" ]; then
|
||
value="$default"
|
||
fi
|
||
echo "$value"
|
||
}
|
||
|
||
handle_auto_rebuild(){
|
||
local storage_path
|
||
storage_path="$(read_env_value STORAGE_PATH_LOCAL "./local-storage")"
|
||
if [[ "$storage_path" != /* ]]; then
|
||
# Remove leading ./ if present
|
||
storage_path="${storage_path#./}"
|
||
storage_path="$(dirname "$COMPOSE_FILE")/$storage_path"
|
||
fi
|
||
local sentinel="$storage_path/modules/.requires_rebuild"
|
||
[ -f "$sentinel" ] || return 0
|
||
|
||
info "Module rebuild required (detected $(realpath "$sentinel" 2>/dev/null || echo "$sentinel"))."
|
||
local auto_rebuild
|
||
auto_rebuild="$(read_env_value AUTO_REBUILD_ON_DEPLOY "0")"
|
||
if [ "$auto_rebuild" != "1" ]; then
|
||
warn "Run ./scripts/rebuild-with-modules.sh after preparing your source tree."
|
||
return 0
|
||
fi
|
||
|
||
local rebuild_source
|
||
rebuild_source="$(read_env_value MODULES_REBUILD_SOURCE_PATH "")"
|
||
info "AUTO_REBUILD_ON_DEPLOY=1; invoking ./scripts/rebuild-with-modules.sh."
|
||
local cmd=(./scripts/rebuild-with-modules.sh --yes)
|
||
if [ -n "$rebuild_source" ]; then
|
||
cmd+=(--source "$rebuild_source")
|
||
fi
|
||
if "${cmd[@]}"; then
|
||
info "Module rebuild completed."
|
||
else
|
||
warn "Automatic rebuild failed; run ./scripts/rebuild-with-modules.sh manually."
|
||
fi
|
||
}
|
||
|
||
check_health(){
|
||
local name="$1"
|
||
local status=$(docker inspect --format='{{.State.Health.Status}}' "$name" 2>/dev/null || echo "no-health-check")
|
||
if [ "$status" = "healthy" ]; then ok "$name: healthy"; return 0; fi
|
||
if docker ps --format '{{.Names}}' | grep -q "^${name}$"; then ok "$name: running"; return 0; fi
|
||
err "$name: not running"; return 1
|
||
}
|
||
|
||
wait_log(){
|
||
local name="$1"; local needle="$2"; local attempts="${3:-360}"; local interval=5
|
||
info "Waiting for $name log: '$needle' ... (timeout: $((attempts*interval))s)"
|
||
for i in $(seq 1 "$attempts"); do
|
||
if docker logs "$name" 2>/dev/null | grep -q "$needle"; then ok "$name ready"; return 0; fi
|
||
sleep "$interval"
|
||
done
|
||
warn "$name did not report '$needle'"
|
||
return 1
|
||
}
|
||
|
||
deploy(){
|
||
info "Deploying profiles: ${PROFILES[*]}"
|
||
local args=()
|
||
for p in "${PROFILES[@]}"; do args+=(--profile "$p"); done
|
||
run_compose "${args[@]}" up -d
|
||
}
|
||
|
||
health_checks(){
|
||
info "Checking container health"
|
||
local failures=0
|
||
check_health ac-mysql || ((failures++))
|
||
check_health ac-authserver || ((failures++))
|
||
check_health ac-worldserver || ((failures++))
|
||
if [ "$QUICK" = false ]; then
|
||
info "Port checks"
|
||
for port in 64306 3784 8215 7778 8081 4201; do
|
||
if timeout 3 bash -c "</dev/tcp/127.0.0.1/$port" 2>/dev/null; then ok "port $port: open"; else warn "port $port: closed"; fi
|
||
done
|
||
fi
|
||
if [ $failures -eq 0 ]; then ok "All core services healthy"; else err "$failures service checks failed"; return 1; fi
|
||
}
|
||
|
||
main(){
|
||
if [ "$SKIP_DEPLOY" = false ]; then
|
||
deploy
|
||
# Wait for client-data completion if profile active
|
||
if printf '%s\n' "${PROFILES[@]}" | grep -q '^client-data$\|^client-data-bots$'; then
|
||
wait_log ac-client-data "Game data setup complete" || true
|
||
fi
|
||
# Give worldserver time to boot
|
||
sleep 10
|
||
fi
|
||
health_checks
|
||
handle_auto_rebuild
|
||
info "Endpoints: MySQL:64306, Auth:3784, World:8215, SOAP:7778, phpMyAdmin:8081, Keira3:4201"
|
||
}
|
||
|
||
main "$@"
|