256 lines
11 KiB
Bash
256 lines
11 KiB
Bash
#!/bin/bash
|
||
|
||
# 颜色定义
|
||
RED='\033[0;31m'
|
||
GREEN='\033[0;32m'
|
||
YELLOW='\033[1;33m'
|
||
BLUE='\033[0;34m'
|
||
NC='\033[0m'
|
||
|
||
PROJECT_ROOT="/www/wwwroot/chat.ay.lc" # <--- 确认这里的路径是否正确
|
||
export COMPOSER_ALLOW_SUPERUSER=1
|
||
SUPERVISORCTL_BIN=""
|
||
PHP_BIN=""
|
||
|
||
for candidate in /usr/bin/supervisorctl /usr/local/bin/supervisorctl /www/server/panel/pyenv/bin/supervisorctl; do
|
||
if [ -x "$candidate" ]; then
|
||
SUPERVISORCTL_BIN="$candidate"
|
||
break
|
||
fi
|
||
done
|
||
|
||
for candidate in /www/server/php/84/bin/php /usr/bin/php /usr/local/bin/php; do
|
||
if [ -x "$candidate" ]; then
|
||
PHP_BIN="$candidate"
|
||
break
|
||
fi
|
||
done
|
||
|
||
echo -e "${BLUE}========================================${NC}"
|
||
echo -e "${BLUE} 🚀 Laravel 稳健更新脚本 (带严格检查) ${NC}"
|
||
echo -e "${BLUE}========================================${NC}"
|
||
|
||
cd "$PROJECT_ROOT" || { echo -e "${RED}❌ 无法进入项目目录:$PROJECT_ROOT${NC}"; exit 1; }
|
||
|
||
if [ -z "$PHP_BIN" ]; then
|
||
echo -e "${RED}❌ 未找到可用 PHP 可执行文件,请确认服务器已安装 PHP 8.4。${NC}"
|
||
exit 1
|
||
fi
|
||
|
||
PHP_VERSION=$("$PHP_BIN" -r 'echo PHP_VERSION;' 2>/dev/null)
|
||
echo -e "${BLUE}使用 PHP:$PHP_BIN (版本:${PHP_VERSION:-unknown})${NC}"
|
||
|
||
frontend_build_required() {
|
||
local file="$1"
|
||
|
||
case "$file" in
|
||
resources/js/*|resources/css/*|resources/views/*.blade.php|resources/views/*/*.blade.php|resources/views/*/*/*.blade.php|resources/views/*/*/*/*.blade.php)
|
||
return 0
|
||
;;
|
||
package.json|package-lock.json|vite.config.*|tailwind.config.*|postcss.config.*)
|
||
return 0
|
||
;;
|
||
esac
|
||
|
||
return 1
|
||
}
|
||
|
||
# 1. Git Pull(先重置 lock 文件,避免服务器环境差异导致冲突)
|
||
echo -e "${YELLOW}[1/8] 拉取代码...${NC}"
|
||
BEFORE_REV=$(git rev-parse HEAD 2>/dev/null || echo "")
|
||
git checkout -- composer.lock package-lock.json 2>/dev/null || true
|
||
git fetch origin && git pull origin master
|
||
if [ $? -ne 0 ]; then echo -e "${RED}❌ Git 失败${NC}"; exit 1; fi
|
||
AFTER_REV=$(git rev-parse HEAD 2>/dev/null || echo "")
|
||
|
||
if [ -n "$BEFORE_REV" ] && [ -n "$AFTER_REV" ] && [ "$BEFORE_REV" = "$AFTER_REV" ]; then
|
||
echo -e "${GREEN}✅ 当前代码已是最新版本,本次没有可升级内容,跳过后续部署步骤。${NC}"
|
||
exit 0
|
||
fi
|
||
|
||
FRONTEND_BUILD_NEEDED=0
|
||
FRONTEND_CHANGED_FILES=""
|
||
MIGRATION_NEEDED=0
|
||
NEW_MIGRATION_FILES=""
|
||
if [ -z "$BEFORE_REV" ] || [ -z "$AFTER_REV" ]; then
|
||
FRONTEND_BUILD_NEEDED=1
|
||
FRONTEND_CHANGED_FILES="无法识别更新前后版本,保险起见执行构建"
|
||
MIGRATION_NEEDED=1
|
||
NEW_MIGRATION_FILES="无法识别更新前后版本,保险起见执行迁移检查"
|
||
elif [ "$BEFORE_REV" != "$AFTER_REV" ]; then
|
||
while IFS= read -r changed_file; do
|
||
if frontend_build_required "$changed_file"; then
|
||
FRONTEND_BUILD_NEEDED=1
|
||
FRONTEND_CHANGED_FILES="${FRONTEND_CHANGED_FILES}${changed_file}"$'\n'
|
||
fi
|
||
done < <(git diff --name-only "$BEFORE_REV" "$AFTER_REV")
|
||
|
||
while IFS=$'\t' read -r change_status changed_file extra_path; do
|
||
if [ "$change_status" = "A" ]; then
|
||
case "$changed_file" in
|
||
database/migrations/*.php)
|
||
MIGRATION_NEEDED=1
|
||
NEW_MIGRATION_FILES="${NEW_MIGRATION_FILES}${changed_file}"$'\n'
|
||
;;
|
||
esac
|
||
fi
|
||
done < <(git diff --name-status "$BEFORE_REV" "$AFTER_REV")
|
||
fi
|
||
|
||
# 2. Composer Install (关键检查点)
|
||
echo -e "${YELLOW}[2/8] 安装依赖 (Composer)...${NC}"
|
||
composer install --no-dev --optimize-autoloader --classmap-authoritative --no-interaction
|
||
COMPOSER_EXIT_CODE=$?
|
||
|
||
if [ $COMPOSER_EXIT_CODE -ne 0 ]; then
|
||
echo -e "${RED}========================================${NC}"
|
||
echo -e "${RED} ❌ 致命错误:Composer 安装失败! ${NC}"
|
||
echo -e "${RED} vendor/autoload.php 未生成,网站将不可用。 ${NC}"
|
||
echo -e "${RED} 请检查上方的错误日志 (网络?内存?PHP版本?) ${NC}"
|
||
echo -e "${RED}========================================${NC}"
|
||
exit 1
|
||
fi
|
||
|
||
# 检查文件是否真的存在
|
||
if [ ! -f "vendor/autoload.php" ]; then
|
||
echo -e "${RED}❌ 奇怪:Composer 显示成功,但 vendor/autoload.php 不存在!${NC}"
|
||
exit 1
|
||
fi
|
||
echo -e "${GREEN}✅ 依赖安装成功且文件存在。${NC}"
|
||
|
||
# 关键 Composer polyfill 文件检查:若 vendor 目录不完整,Laravel 接口会返回 PHP Warning HTML,前端会误判为 JSON 解析失败
|
||
if [ ! -f "vendor/symfony/polyfill-mbstring/bootstrap.php" ]; then
|
||
echo -e "${RED}========================================${NC}"
|
||
echo -e "${RED} ❌ 致命错误:Composer vendor 不完整! ${NC}"
|
||
echo -e "${RED} 缺少 vendor/symfony/polyfill-mbstring/bootstrap.php。 ${NC}"
|
||
echo -e "${RED} 请删除线上 vendor 后重新执行 composer install。 ${NC}"
|
||
echo -e "${RED}========================================${NC}"
|
||
exit 1
|
||
fi
|
||
|
||
# 重新生成生产环境 autoload,避免 autoload_files.php 指向已缺失的旧文件
|
||
composer dump-autoload --no-dev --optimize --classmap-authoritative --no-interaction
|
||
if [ $? -ne 0 ]; then echo -e "${RED}❌ Composer autoload 重建失败${NC}"; exit 1; fi
|
||
|
||
# 用当前 PHP 直接加载 Laravel autoload,提前暴露 vendor 缺文件 / 权限 / autoload 缓存问题
|
||
"$PHP_BIN" -d opcache.enable_cli=0 -r "require __DIR__.'/vendor/autoload.php'; echo 'Composer autoload OK'.PHP_EOL;"
|
||
if [ $? -ne 0 ]; then
|
||
echo -e "${RED}========================================${NC}"
|
||
echo -e "${RED} ❌ 致命错误:PHP 无法加载 vendor/autoload.php! ${NC}"
|
||
echo -e "${RED} 请检查 vendor 是否完整、文件权限是否正确、Composer 是否使用了正确 PHP 版本。 ${NC}"
|
||
echo -e "${RED}========================================${NC}"
|
||
exit 1
|
||
fi
|
||
|
||
# 3. 前端构建
|
||
if [ "$FRONTEND_BUILD_NEEDED" -eq 1 ]; then
|
||
echo -e "${YELLOW}[3/8] 检测到前端构建输入变更,开始前端构建 (npm run build)...${NC}"
|
||
if [ -n "$FRONTEND_CHANGED_FILES" ]; then
|
||
echo -e "${BLUE}触发构建的文件:${NC}"
|
||
printf '%s\n' "$FRONTEND_CHANGED_FILES"
|
||
fi
|
||
npm run build
|
||
if [ $? -ne 0 ]; then echo -e "${RED}❌ npm run build 失败${NC}"; exit 1; fi
|
||
echo -e "${GREEN}✅ 前端资源构建完成。${NC}"
|
||
else
|
||
echo -e "${GREEN}[3/8] 未检测到前端构建输入变更,跳过 npm run build。${NC}"
|
||
fi
|
||
|
||
|
||
# 4. 数据库迁移
|
||
if [ "$MIGRATION_NEEDED" -eq 1 ]; then
|
||
echo -e "${YELLOW}[4/8] 检测到新增迁移文件,执行数据库迁移...${NC}"
|
||
if [ -n "$NEW_MIGRATION_FILES" ]; then
|
||
echo -e "${BLUE}新增迁移文件:${NC}"
|
||
printf '%s\n' "$NEW_MIGRATION_FILES"
|
||
fi
|
||
"$PHP_BIN" artisan migrate --force
|
||
if [ $? -ne 0 ]; then echo -e "${RED}❌ 数据库迁移失败${NC}"; exit 1; fi
|
||
else
|
||
echo -e "${GREEN}[4/8] 未检测到新增迁移文件,跳过数据库迁移。${NC}"
|
||
fi
|
||
|
||
# 5. 优化
|
||
echo -e "${YELLOW}[5/8] 生产环境优化...${NC}"
|
||
# 注意:optimize 命令内部已经包含了 config:cache, route:cache, event:cache,此处无须多余处理
|
||
"$PHP_BIN" artisan optimize:clear && "$PHP_BIN" artisan optimize && "$PHP_BIN" artisan view:cache
|
||
|
||
# 6. 重启 Horizon / 队列进程
|
||
echo -e "${YELLOW}[6/8] 重启 Horizon...${NC}"
|
||
"$PHP_BIN" artisan horizon:terminate >/dev/null 2>&1 || true
|
||
if [ -n "$SUPERVISORCTL_BIN" ]; then
|
||
"$SUPERVISORCTL_BIN" restart horizon >/dev/null 2>&1 || "$SUPERVISORCTL_BIN" restart horizon:* >/dev/null 2>&1 || true
|
||
fi
|
||
|
||
# 7. 重启 Reverb,确保长驻 WebSocket 进程加载最新代码和配置
|
||
echo -e "${YELLOW}[7/8] 重启 Reverb...${NC}"
|
||
REVERB_RESTARTED=0
|
||
REVERB_CAN_AUTO_RESTART=0
|
||
SUPERVISOR_REVERB_TARGET=""
|
||
|
||
# 先探测是否存在可自动拉起 Reverb 的进程管理器,避免在纯手工启动场景下执行 reverb:restart 后把聊天室停掉。
|
||
if [ -n "$SUPERVISORCTL_BIN" ]; then
|
||
if "$SUPERVISORCTL_BIN" status reverb >/dev/null 2>&1; then
|
||
SUPERVISOR_REVERB_TARGET="reverb"
|
||
REVERB_CAN_AUTO_RESTART=1
|
||
elif "$SUPERVISORCTL_BIN" status reverb:* >/dev/null 2>&1; then
|
||
SUPERVISOR_REVERB_TARGET="reverb:*"
|
||
REVERB_CAN_AUTO_RESTART=1
|
||
fi
|
||
fi
|
||
|
||
# Laravel 官方文档说明:reverb:restart 适合在 Supervisor 等进程管理器托管下使用。
|
||
if [ "$REVERB_CAN_AUTO_RESTART" -eq 1 ]; then
|
||
"$PHP_BIN" artisan reverb:restart >/dev/null 2>&1
|
||
if [ $? -eq 0 ]; then
|
||
REVERB_RESTARTED=1
|
||
echo -e "${GREEN}✅ 已执行 php artisan reverb:restart。${NC}"
|
||
fi
|
||
|
||
# 若线上通过 Supervisor 托管 reverb:start,再补一次显式 restart,尽量确保进程被拉起。
|
||
if [ -n "$SUPERVISOR_REVERB_TARGET" ]; then
|
||
"$SUPERVISORCTL_BIN" restart "$SUPERVISOR_REVERB_TARGET" >/dev/null 2>&1 || true
|
||
REVERB_RESTARTED=1
|
||
echo -e "${GREEN}✅ 已尝试重启 Supervisor 托管的 Reverb 进程。${NC}"
|
||
fi
|
||
else
|
||
echo -e "${YELLOW}⚠️ 未检测到 Reverb 进程管理器,已跳过 reverb:restart,避免把手工启动的聊天室长连接服务停掉。${NC}"
|
||
echo -e "${YELLOW}⚠️ 若当前服务器是手工执行 php artisan reverb:start,请在部署完成后手动重启该进程。${NC}"
|
||
fi
|
||
|
||
# 8. 重载 PHP-FPM / opcache,避免 Web 进程继续使用旧的 autoload 缓存
|
||
echo -e "${YELLOW}[8/8] 重载 PHP-FPM...${NC}"
|
||
PHP_FPM_RELOADED=0
|
||
if command -v systemctl >/dev/null 2>&1; then
|
||
for svc in php-fpm php-fpm-84 php-fpm-83 php-fpm-82 php84-php-fpm php83-php-fpm php82-php-fpm; do
|
||
if systemctl list-unit-files "$svc.service" >/dev/null 2>&1; then
|
||
systemctl reload "$svc" >/dev/null 2>&1 || systemctl restart "$svc" >/dev/null 2>&1 || true
|
||
PHP_FPM_RELOADED=1
|
||
fi
|
||
done
|
||
fi
|
||
for init_script in /etc/init.d/php-fpm /etc/init.d/php-fpm-*; do
|
||
if [ -x "$init_script" ]; then
|
||
"$init_script" reload >/dev/null 2>&1 || "$init_script" restart >/dev/null 2>&1 || true
|
||
PHP_FPM_RELOADED=1
|
||
fi
|
||
done
|
||
if [ "$PHP_FPM_RELOADED" -eq 1 ]; then
|
||
echo -e "${GREEN}✅ PHP-FPM 已尝试重载。${NC}"
|
||
else
|
||
echo -e "${YELLOW}⚠️ 未识别到 PHP-FPM 服务,请在面板中手动重启当前站点使用的 PHP。${NC}"
|
||
fi
|
||
|
||
# 权限修正(针对宝塔或 Nginx + FPM 环境)
|
||
echo -e "${YELLOW}[后处理] 修复权限...${NC}"
|
||
# 将所有文件所属权变更为 Web 运行用户(如 www),防止 root 权限导致框架日志或缓存写入失败
|
||
chown -R www:www .
|
||
# 默认读写执行权限
|
||
chmod -R 755 .
|
||
# 针对 Laravel 必须具备写入权限的核心目录给予 775 权限
|
||
chmod -R 775 storage bootstrap/cache
|
||
|
||
echo -e "${GREEN}========================================${NC}"
|
||
echo -e "${GREEN} 🎉 更新成功!网站已恢复。 ${NC}"
|
||
echo -e "${GREEN}========================================${NC}"
|