Console Output
+ set -e
+ mkdir -p /zoumh/java/zmh/backend/packages /zoumh/java/zmh/backend/logs /zoumh/java/zmh/backend/bin /zoumh/java/zmh/backend/docs /zoumh/java/zmh/backend/nginx
+ cp -f ruoyi-gateway/target/ruoyi-gateway.jar /zoumh/java/zmh/backend/packages/
+ cp -f ruoyi-auth/target/ruoyi-auth.jar /zoumh/java/zmh/backend/packages/
+ cp -f ruoyi-modules/ruoyi-system/target/ruoyi-modules-system.jar /zoumh/java/zmh/backend/packages/
+ cp -f ruoyi-modules/ruoyi-file/target/ruoyi-modules-file.jar /zoumh/java/zmh/backend/packages/
+ printf %s #!/usr/bin/env bash
set -euo pipefail

PACKAGE_DIR="${PACKAGE_DIR:-/zoumh/java/zmh/backend/packages}"
LOG_DIR="${LOG_DIR:-/zoumh/java/zmh/backend/logs}"
JAVA_IMAGE="${JAVA_IMAGE:-eclipse-temurin:21-jre}"
NACOS_ADDR="${NACOS_ADDR:-156.225.28.110:8848}"
NACOS_USERNAME="${NACOS_USERNAME:-nacos}"
NACOS_PASSWORD="${NACOS_PASSWORD:-zoumh}"
REDIS_HOST="${REDIS_HOST:-172.21.0.1}"
REDIS_PORT="${REDIS_PORT:-6379}"
REDIS_PASSWORD="${REDIS_PASSWORD:-zoumh}"
REDIS_DATABASE="${REDIS_DATABASE:-0}"
TZ_NAME="${TZ_NAME:-Asia/Shanghai}"
JAVA_TMPDIR="${JAVA_TMPDIR:-/tmp/zoumh-java}"
DEFAULT_JAVA_OPTS="${DEFAULT_JAVA_OPTS:--Dfile.encoding=UTF-8 -Djava.security.egd=file:/dev/./urandom -Djava.io.tmpdir=/tmp/zoumh-java -XX:+UseG1GC -XX:+UseStringDeduplication -XX:+ExitOnOutOfMemoryError -XX:MaxMetaspaceSize=128m -XX:ReservedCodeCacheSize=64m -XX:MaxDirectMemorySize=80m}"
JAVA_OPTS_AUTH="${JAVA_OPTS_AUTH:--Xms128m -Xmx320m}"
JAVA_OPTS_SYSTEM="${JAVA_OPTS_SYSTEM:--Xms96m -Xmx320m}"
JAVA_OPTS_FILE="${JAVA_OPTS_FILE:--Xms64m -Xmx160m}"
JAVA_OPTS_GATEWAY="${JAVA_OPTS_GATEWAY:--Xms96m -Xmx320m -XX:MaxDirectMemorySize=128m}"
DOCKER_MEMORY_AUTH="${DOCKER_MEMORY_AUTH:-512m}"
DOCKER_MEMORY_SYSTEM="${DOCKER_MEMORY_SYSTEM:-512m}"
DOCKER_MEMORY_FILE="${DOCKER_MEMORY_FILE:-288m}"
DOCKER_MEMORY_GATEWAY="${DOCKER_MEMORY_GATEWAY:-512m}"
DOCKER_MEMORY_RESERVATION_AUTH="${DOCKER_MEMORY_RESERVATION_AUTH:-256m}"
DOCKER_MEMORY_RESERVATION_SYSTEM="${DOCKER_MEMORY_RESERVATION_SYSTEM:-224m}"
DOCKER_MEMORY_RESERVATION_FILE="${DOCKER_MEMORY_RESERVATION_FILE:-128m}"
DOCKER_MEMORY_RESERVATION_GATEWAY="${DOCKER_MEMORY_RESERVATION_GATEWAY:-224m}"
DOCKER_PIDS_LIMIT="${DOCKER_PIDS_LIMIT:-256}"
MINIO_IMAGE="${MINIO_IMAGE:-minio/minio:latest}"
MINIO_CONTAINER_NAME="${MINIO_CONTAINER_NAME:-minio}"
MINIO_ROOT_USER="${MINIO_ROOT_USER:-minio}"
MINIO_ROOT_PASSWORD="${MINIO_ROOT_PASSWORD:-zmh0012078070}"
MINIO_DATA_DIR="${MINIO_DATA_DIR:-/zoumh/data/minio}"
MINIO_CONSOLE_PORT="${MINIO_CONSOLE_PORT:-9001}"
MINIO_API_PORT="${MINIO_API_PORT:-9000}"
MINIO_NETWORK="${MINIO_NETWORK:-docker-compose_backend}"
PUBLIC_NGINX_CONF_SOURCE="${PUBLIC_NGINX_CONF_SOURCE:-/zoumh/java/zmh/backend/nginx/nginx.conf}"
PUBLIC_NGINX_CONF_TARGET="${PUBLIC_NGINX_CONF_TARGET-/zoumh/data/nginx/conf/nginx.conf}"

mkdir -p "${PACKAGE_DIR}" "${LOG_DIR}"

ensure_docker_shell_env() {
  local docker_cmd docker_bin docker_dir env_file helper_file bashrc_snippet
  docker_cmd="$(command -v docker || true)"
  if [[ -z "${docker_cmd}" ]]; then
    echo "docker command not found in deploy environment" >&2
    exit 1
  fi

  docker_bin="$(readlink -f "${docker_cmd}" 2>/dev/null || true)"
  if [[ -z "${docker_bin}" || ! -x "${docker_bin}" ]]; then
    if [[ -x /usr/bin/docker ]]; then
      docker_bin="/usr/bin/docker"
    else
      docker_bin="${docker_cmd}"
    fi
  fi

  docker_dir="$(dirname "${docker_bin}")"
  env_file="/etc/profile.d/zoumh-docker.sh"
  helper_file="/zoumh/sh/docker.sh"
  bashrc_snippet="# >>> zoumh docker env >>>"

  mkdir -p /etc/profile.d /zoumh/sh
  if [[ -L /usr/local/bin/docker && ! -e /usr/local/bin/docker ]]; then
    rm -f /usr/local/bin/docker
  fi
  ln -sf "${docker_bin}" /usr/local/bin/docker || true
  ln -sf "${docker_bin}" /usr/bin/docker || true

  clean_shell_hook() {
    local shell_file="$1"
    local temp_file
    [[ -f "${shell_file}" ]] || touch "${shell_file}"
    temp_file="$(mktemp)"
    awk '
      BEGIN {
        skip_block = 0
        skip_legacy_fi = 0
      }
      /^# >>> zoumh docker env >>>$/ {
        skip_block = 1
        next
      }
      /^# <<< zoumh docker env <<</ {
        skip_block = 0
        next
      }
      skip_block {
        next
      }
      skip_legacy_fi && /^fi$/ {
        skip_legacy_fi = 0
        next
      }
      /^# zoumh docker env$/ {
        skip_legacy_fi = 1
        next
      }
      /\/etc\/profile\.d\/zoumh-docker\.sh/ {
        next
      }
      /\/zoumh\/sh\/docker\.sh/ {
        next
      }
      {
        skip_legacy_fi = 0
        print
      }
    ' "${shell_file}" > "${temp_file}"
    cat "${temp_file}" > "${shell_file}"
    rm -f "${temp_file}"
  }

  cat > "${env_file}" <<EOF
export DOCKER_HOME='${docker_dir}'
case ":\$PATH:" in
  *:"${docker_dir}":*) ;;
  *) export PATH="${docker_dir}:\$PATH" ;;
esac
EOF
  chmod 644 "${env_file}"

  clean_shell_hook /etc/bashrc
  cat >> /etc/bashrc <<EOF

${bashrc_snippet}
if [ -f /etc/profile.d/zoumh-docker.sh ]; then
  . /etc/profile.d/zoumh-docker.sh
fi
# <<< zoumh docker env <<<
EOF

  mkdir -p /root
  for shell_file in /root/.bashrc /root/.bash_profile; do
    clean_shell_hook "${shell_file}"
    cat >> "${shell_file}" <<EOF

${bashrc_snippet}
if [ -f /etc/profile.d/zoumh-docker.sh ]; then
  . /etc/profile.d/zoumh-docker.sh
fi
# <<< zoumh docker env <<<
EOF
  done

  cat > "${helper_file}" <<EOF
#!/usr/bin/env bash
set -e
export DOCKER_HOME='${docker_dir}'
case ":\$PATH:" in
  *:"${docker_dir}":*) ;;
  *) export PATH="${docker_dir}:\$PATH" ;;
esac

if [[ \$# -eq 0 ]]; then
  exec "${docker_bin}" --version
fi

exec "${docker_bin}" "\$@"
EOF
  chmod +x "${helper_file}"
}

ensure_docker_shell_env

cleanup_removed_module_menu_data() {
  local sql
  sql="$(cat <<'EOF'
DELETE rm
FROM sys_role_menu rm
INNER JOIN sys_menu m ON m.menu_id = rm.menu_id
WHERE m.perms LIKE 'hotel:monitor:%'
   OR m.path = 'hotel'
   OR m.path = 'monitor'
   OR m.component = 'hotel/monitor/index'
   OR m.route_name IN ('Hotel', 'HotelMonitor');

DELETE FROM sys_menu
WHERE perms LIKE 'hotel:monitor:%'
   OR path = 'hotel'
   OR path = 'monitor'
   OR component = 'hotel/monitor/index'
   OR route_name IN ('Hotel', 'HotelMonitor');
EOF
)"

  if docker ps --format '{{.Names}}' | grep -qx 'mysql8'; then
    docker exec -i mysql8 mysql -uroot -pzoumh zoumh -e "${sql}" >/dev/null
    echo "removed stale zoumh-hotel-monitor menu data"
  else
    echo "skip menu cleanup: mysql8 container not running"
  fi
}

ensure_minio() {
  mkdir -p "${MINIO_DATA_DIR}"
  docker rm -f "${MINIO_CONTAINER_NAME}" >/dev/null 2>&1 || true
  docker run -d \
    --name "${MINIO_CONTAINER_NAME}" \
    --restart unless-stopped \
    --network "${MINIO_NETWORK}" \
    -p "${MINIO_API_PORT}:9000" \
    -p "${MINIO_CONSOLE_PORT}:9001" \
    --memory="384m" \
    --memory-reservation="128m" \
    --pids-limit="128" \
    --log-opt max-size=20m \
    --log-opt max-file=3 \
    -e TZ="${TZ_NAME}" \
    -e MINIO_ROOT_USER="${MINIO_ROOT_USER}" \
    -e MINIO_ROOT_PASSWORD="${MINIO_ROOT_PASSWORD}" \
    -v "${MINIO_DATA_DIR}:/data" \
    "${MINIO_IMAGE}" server /data --console-address ":9001"
  echo "started ${MINIO_CONTAINER_NAME}"
}

sync_public_nginx_conf() {
  if [[ ! -f "${PUBLIC_NGINX_CONF_SOURCE}" ]]; then
    echo "skip public nginx sync: source not found ${PUBLIC_NGINX_CONF_SOURCE}"
    return 0
  fi

  if [[ -z "${PUBLIC_NGINX_CONF_TARGET}" ]]; then
    echo "skip public nginx sync: target is empty"
    return 0
  fi

  mkdir -p "$(dirname "${PUBLIC_NGINX_CONF_TARGET}")"

  local backup_file
  backup_file="${PUBLIC_NGINX_CONF_TARGET}.bak.$(date +%Y%m%d%H%M%S)"
  if [[ -f "${PUBLIC_NGINX_CONF_TARGET}" ]]; then
    cp -f "${PUBLIC_NGINX_CONF_TARGET}" "${backup_file}"
  fi

  cp -f "${PUBLIC_NGINX_CONF_SOURCE}" "${PUBLIC_NGINX_CONF_TARGET}"

  if docker ps --format '{{.Names}}' | grep -qx 'nginx'; then
    if docker exec nginx nginx -t >/dev/null 2>&1; then
      docker exec nginx nginx -s reload >/dev/null 2>&1 || docker restart nginx >/dev/null
      echo "public nginx config synced and reloaded"
    else
      echo "public nginx config test failed, rolling back" >&2
      if [[ -f "${backup_file}" ]]; then
        cp -f "${backup_file}" "${PUBLIC_NGINX_CONF_TARGET}"
        docker exec nginx nginx -t >/dev/null 2>&1 && docker exec nginx nginx -s reload >/dev/null 2>&1 || docker restart nginx >/dev/null 2>&1 || true
      fi
      return 1
    fi
  else
    echo "public nginx config synced; nginx container not running, skip reload"
  fi
}

docker rm -f "ruoyi-monitor" >/dev/null 2>&1 || true
docker rm -f "ruoyi-job" >/dev/null 2>&1 || true
docker rm -f "ruoyi-gen" >/dev/null 2>&1 || true
docker rm -f "zoumh-tools" >/dev/null 2>&1 || true
docker rm -f "zoumh-hotel-monitor" >/dev/null 2>&1 || true
cleanup_removed_module_menu_data
sync_public_nginx_conf

run_java_service() {
  local name="$1"
  local jar_name="$2"
  local java_opts="$3"
  local docker_memory="$4"
  local docker_memory_reservation="$5"
  shift 5

  if [[ ! -f "${PACKAGE_DIR}/${jar_name}" ]]; then
    echo "skip ${name}: ${jar_name} not found"
    return 0
  fi

  docker rm -f "${name}" >/dev/null 2>&1 || true

  docker run -d \
    --name "${name}" \
    --restart unless-stopped \
    --network host \
    --memory="${docker_memory}" \
    --memory-reservation="${docker_memory_reservation}" \
    --pids-limit="${DOCKER_PIDS_LIMIT}" \
    --log-opt max-size=20m \
    --log-opt max-file=3 \
    -e TZ="${TZ_NAME}" \
    -e "JAVA_OPTS=${DEFAULT_JAVA_OPTS} ${java_opts}" \
    "$@" \
    -v "${PACKAGE_DIR}:/app" \
    -v "${LOG_DIR}:/logs" \
    -v "${JAVA_TMPDIR}:${JAVA_TMPDIR}" \
    "${JAVA_IMAGE}" \
    sh -lc "mkdir -p '${JAVA_TMPDIR}' && exec java \$JAVA_OPTS -jar /app/${jar_name} > /logs/${name}.log 2>&1"

  echo "started ${name}"
}

run_java_service \
  "ruoyi-auth" \
  "ruoyi-auth.jar" \
  "${JAVA_OPTS_AUTH}" \
  "${DOCKER_MEMORY_AUTH}" \
  "${DOCKER_MEMORY_RESERVATION_AUTH}" \
  -e "SPRING_CLOUD_NACOS_SERVER_ADDR=${NACOS_ADDR}" \
  -e "SPRING_CLOUD_NACOS_USERNAME=${NACOS_USERNAME}" \
  -e "SPRING_CLOUD_NACOS_PASSWORD=${NACOS_PASSWORD}"

run_java_service \
  "ruoyi-system" \
  "ruoyi-modules-system.jar" \
  "${JAVA_OPTS_SYSTEM}" \
  "${DOCKER_MEMORY_SYSTEM}" \
  "${DOCKER_MEMORY_RESERVATION_SYSTEM}" \
  -e "SPRING_CLOUD_NACOS_SERVER_ADDR=${NACOS_ADDR}" \
  -e "SPRING_CLOUD_NACOS_USERNAME=${NACOS_USERNAME}" \
  -e "SPRING_CLOUD_NACOS_PASSWORD=${NACOS_PASSWORD}" \
  -e "SPRING_CLOUD_NACOS_DISCOVERY_SERVICE=ruoyi-system"

run_java_service \
  "ruoyi-file" \
  "ruoyi-modules-file.jar" \
  "${JAVA_OPTS_FILE}" \
  "${DOCKER_MEMORY_FILE}" \
  "${DOCKER_MEMORY_RESERVATION_FILE}" \
  -e "SPRING_CLOUD_NACOS_SERVER_ADDR=${NACOS_ADDR}" \
  -e "SPRING_CLOUD_NACOS_USERNAME=${NACOS_USERNAME}" \
  -e "SPRING_CLOUD_NACOS_PASSWORD=${NACOS_PASSWORD}"

run_java_service \
  "ruoyi-gateway" \
  "ruoyi-gateway.jar" \
  "${JAVA_OPTS_GATEWAY}" \
  "${DOCKER_MEMORY_GATEWAY}" \
  "${DOCKER_MEMORY_RESERVATION_GATEWAY}" \
  -e "SPRING_CLOUD_NACOS_SERVER_ADDR=${NACOS_ADDR}" \
  -e "SPRING_CLOUD_NACOS_USERNAME=${NACOS_USERNAME}" \
  -e "SPRING_CLOUD_NACOS_PASSWORD=${NACOS_PASSWORD}" \
  -e "SECURITY_CAPTCHA_ENABLED=false" \
  -e "SPRING_DATA_REDIS_HOST=${REDIS_HOST}" \
  -e "SPRING_DATA_REDIS_PORT=${REDIS_PORT}" \
  -e "SPRING_DATA_REDIS_PASSWORD=${REDIS_PASSWORD}" \
  -e "SPRING_DATA_REDIS_DATABASE=${REDIS_DATABASE}"

if [[ -n "${POST_DEPLOY_CMD:-}" ]]; then
  sh -lc "${POST_DEPLOY_CMD}"
fi

docker run --rm "${JAVA_IMAGE}" java -version 2>&1 | head -n 1 || true

login_probe='{"username":"admin","password":"admin123"}'
sleep 20
probe_response="$(curl -ksS -H 'Content-Type: application/json' -d "${login_probe}" https://zoumh.com/prod-api/auth/login || true)"
gateway_probe="$(curl -sS -H 'Content-Type: application/json' -d "${login_probe}" http://127.0.0.1:8080/auth/login || true)"
echo "login_probe_response=${probe_response}"
echo "gateway_probe_response=${gateway_probe}"
if printf '%s\n%s' "${probe_response}" "${gateway_probe}" | grep -Eq '"code":500|502 Bad Gateway'; then
  echo "--- gateway.log tail ---"
  tail -n 80 "${LOG_DIR}/ruoyi-gateway.log" 2>/dev/null || true
  echo "--- auth.log tail ---"
  tail -n 80 "${LOG_DIR}/ruoyi-auth.log" 2>/dev/null || true
  echo "--- system.log tail ---"
  tail -n 80 "${LOG_DIR}/ruoyi-system.log" 2>/dev/null || true
fi

docker stats --no-stream --format '{{.Name}}\t{{.MemUsage}}' | grep -E 'ruoyi-(auth|system|file|gateway)' || true
docker ps --format 'table {{.Names}}\t{{.Status}}' | grep -E 'ruoyi-(auth|system|file|gateway)' || true
echo "--- ruoyi-file.log tail ---"
tail -n 120 "${LOG_DIR}/ruoyi-file.log" 2>/dev/null || true
echo "--- restarting containers ---"
docker ps -a --filter status=restarting --format 'table {{.Names}}\t{{.Status}}' || true

+ base64 -d
+ cp -f docs/server-ops-guide.md /zoumh/java/zmh/backend/docs/server-ops-guide.md
+ cp -f docs/server-ops-guide.md /zoumh/java/zmh/README-ops.md
+ cp -f docker/nginx/conf/nginx.conf /zoumh/java/zmh/backend/nginx/nginx.conf
+ chmod +x /zoumh/java/zmh/backend/bin/deploy-backend-host.sh
+ PACKAGE_DIR=/zoumh/java/zmh/backend/packages LOG_DIR=/zoumh/java/zmh/backend/logs PUBLIC_NGINX_CONF_SOURCE=/zoumh/java/zmh/backend/nginx/nginx.conf PUBLIC_NGINX_CONF_TARGET= POST_DEPLOY_CMD=set -e
docker rm -f seata >/dev/null 2>&1 || true
docker run -d --name seata --restart unless-stopped --network docker-compose_backend alpine:3.20 sh -lc "while true; do sleep 3600; done"
docker rm -f ncm2mp3 >/dev/null 2>&1 || true
mkdir -p /zoumh/java/zmh/backend/ncm2mp3
curl -L --fail --retry 3 -o /zoumh/java/zmh/backend/ncm2mp3/ncm2mp3-1.0.0.jar https://raw.githubusercontent.com/zoumh001207/zoumh/a98cf6bce7db76c6a8d7c788c03438562b53ef5f/tools/ncm2mp3/ncm2mp3-1.0.0.jar
docker run -d --name ncm2mp3 --restart unless-stopped --network docker-compose_backend --memory=512m --memory-reservation=192m --pids-limit=128 --log-opt max-size=20m --log-opt max-file=3 -e TZ=Asia/Shanghai -e SERVER_PORT=1207 -v /zoumh/java/zmh/backend/ncm2mp3:/app eclipse-temurin:21-jre sh -lc 'exec java -jar /app/ncm2mp3-1.0.0.jar'
sleep 6
cp -f /zoumh/java/zmh/backend/nginx/nginx.conf /zoumh/data/nginx/conf/nginx.conf
docker exec nginx getent hosts seata
docker exec nginx getent hosts ncm2mp3
docker exec nginx nginx -t
docker exec nginx nginx -s reload /zoumh/java/zmh/backend/bin/deploy-backend-host.sh
ln: '/usr/bin/docker' and '/usr/bin/docker' are the same file
mysql: [Warning] Using a password on the command line interface can be insecure.
removed stale zoumh-hotel-monitor menu data
skip public nginx sync: target is empty
a0f95d982a7ea9a6075b1a0c286a4197e7f43012cf15dca8dbda3105c8c1f78f
started ruoyi-auth
86dec857b37d833da2c87a06259bd5fada79db96346049fac7b194df743a6b6f
started ruoyi-system
09df1c40b55632c73be30957be6ec39976baf1b8ca198387ec88dc2a483fce47
started ruoyi-file
ee0b44e3a698e3df40a6475c8819fe9b2dc9863bd11cb147499b88c214dfc70e
started ruoyi-gateway
Unable to find image 'alpine:3.20' locally
3.20: Pulling from library/alpine
76eb174b37c3: Pulling fs layer
90cca658fc45: Download complete
7208ffea29d1: Download complete
76eb174b37c3: Download complete
76eb174b37c3: Pull complete
Digest: sha256:a4f4213abb84c497377b8544c81b3564f313746700372ec4fe84653e4fb03805
Status: Downloaded newer image for alpine:3.20
8aa650c71cc3b312e236e802cd092d15564134fe844719f81220245f890d4576
% Total % Received % Xferd Average Speed Time Time Time Current
Dload Upload Total Spent Left Speed
0 0 0 0 0 0 0 0 --:--:-- --:--:-- --:--:-- 0
0 0 0 0 0 0 0 0 --:--:-- --:--:-- --:--:-- 0
23 21.9M 23 5352k 0 0 4764k 0 0:00:04 0:00:01 0:00:03 4761k
49 21.9M 49 10.7M 0 0 5055k 0 0:00:04 0:00:02 0:00:02 5054k
75 21.9M 75 16.5M 0 0 5438k 0 0:00:04 0:00:03 0:00:01 5438k
100 21.9M 100 21.9M 0 0 5530k 0 0:00:04 0:00:04 --:--:-- 5530k
14cc3cca0659dbd86f4f1fffe06725cfcf2071a0f3c1154ee00cd1d07278f233
172.21.0.11 seata seata
172.21.0.10 ncm2mp3 ncm2mp3
nginx: [emerg] cannot load certificate "/etc/letsencrypt/live/zoumh.com/fullchain.pem": BIO_new_file() failed (SSL: error:80000002:system library::No such file or directory:calling fopen(/etc/letsencrypt/live/zoumh.com/fullchain.pem, r) error:10000080:BIO routines::no such file)
nginx: configuration file /etc/nginx/nginx.conf test failed