build_release.sh 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. #!/bin/bash
  2. # 遇到错误立即退出
  3. set -e
  4. echo "========================================================"
  5. echo " ShuDao SafeAI 生产环境一键部署脚本 (Python)"
  6. echo "========================================================"
  7. echo ""
  8. # 设置路径变量
  9. ROOT_DIR=$(pwd)
  10. PARENT_DIR=$(dirname "$ROOT_DIR")
  11. FRONTEND_DIR="$PARENT_DIR/shudao-main/shudao-vue-frontend"
  12. BACKEND_DIR="$ROOT_DIR"
  13. DEPLOY_DIR="/opt/www/shudao-chat-py"
  14. SERVICE_NAME="shudao-chat-py"
  15. SERVICE_PORT=22000
  16. PYTHON_BIN="python3"
  17. VENV_DIR="$DEPLOY_DIR/venv"
  18. # 1. 前端构建
  19. echo "[1/9] 正在构建前端项目 (Vue)..."
  20. cd "$FRONTEND_DIR"
  21. npm run build
  22. if [ $? -ne 0 ]; then
  23. echo "[ERROR] 前端构建失败!请检查错误信息。"
  24. exit 1
  25. fi
  26. echo "[SUCCESS] 前端构建完成。"
  27. echo ""
  28. # 2. 清理后端旧资源
  29. echo "[2/9] 清理后端旧资源..."
  30. cd "$BACKEND_DIR"
  31. rm -rf "$BACKEND_DIR/assets" 2>/dev/null || true
  32. rm -rf "$BACKEND_DIR/views" 2>/dev/null || true
  33. echo "[SUCCESS] 清理完成。"
  34. echo ""
  35. # 3. 复制新资源
  36. echo "[3/9] 整合前端资源到后端..."
  37. mkdir -p "$BACKEND_DIR/assets"
  38. mkdir -p "$BACKEND_DIR/views"
  39. cp -r "$FRONTEND_DIR/dist/assets/"* "$BACKEND_DIR/assets/"
  40. cp "$FRONTEND_DIR/dist/index.html" "$BACKEND_DIR/views/index.html"
  41. echo "[SUCCESS] 资源整合完成。"
  42. echo ""
  43. # 4. 检查并复制生产环境配置
  44. echo "[4/9] 配置生产环境..."
  45. if [ -f "$BACKEND_DIR/config.yaml" ]; then
  46. echo "[INFO] 使用现有 config.yaml"
  47. else
  48. if [ -f "$BACKEND_DIR/config.example.yaml" ]; then
  49. cp "$BACKEND_DIR/config.example.yaml" "$BACKEND_DIR/config.yaml"
  50. echo "[WARNING] 已从 config.example.yaml 创建 config.yaml,请手动配置!"
  51. else
  52. echo "[ERROR] 未找到配置文件模板!"
  53. exit 1
  54. fi
  55. fi
  56. # 检查 .env 文件
  57. if [ ! -f "$BACKEND_DIR/.env" ]; then
  58. echo "[WARNING] 未找到 .env 文件,请确保环境变量已正确配置!"
  59. echo "[INFO] 可以从 .env.example 复制并修改"
  60. fi
  61. echo "[SUCCESS] 配置检查完成。"
  62. echo ""
  63. # 5. 停止旧服务
  64. echo "[5/9] 停止旧服务..."
  65. pkill -f "uvicorn main:app" 2>/dev/null || true
  66. pkill -f "$SERVICE_NAME" 2>/dev/null || true
  67. sleep 2
  68. # 确认服务已停止
  69. if pgrep -f "uvicorn main:app" > /dev/null 2>&1; then
  70. echo "[WARNING] 服务未完全停止,强制终止..."
  71. pkill -9 -f "uvicorn main:app" 2>/dev/null || true
  72. sleep 1
  73. fi
  74. echo "[SUCCESS] 旧服务已停止。"
  75. echo ""
  76. # 6. 部署到目标目录
  77. echo "[6/9] 部署到 $DEPLOY_DIR..."
  78. mkdir -p "$DEPLOY_DIR"
  79. rsync -av --exclude='__pycache__' \
  80. --exclude='*.pyc' \
  81. --exclude='venv' \
  82. --exclude='.git' \
  83. --exclude='logs/*.log' \
  84. --exclude='.env' \
  85. "$BACKEND_DIR/" "$DEPLOY_DIR/"
  86. # 如果 .env 存在,复制过去
  87. if [ -f "$BACKEND_DIR/.env" ]; then
  88. cp "$BACKEND_DIR/.env" "$DEPLOY_DIR/"
  89. fi
  90. echo "[SUCCESS] 部署完成。"
  91. echo ""
  92. # 7. 创建/更新虚拟环境
  93. echo "[7/9] 配置 Python 虚拟环境..."
  94. cd "$DEPLOY_DIR"
  95. if [ ! -d "$VENV_DIR" ]; then
  96. echo "[INFO] 创建新的虚拟环境..."
  97. $PYTHON_BIN -m venv "$VENV_DIR"
  98. fi
  99. # 激活虚拟环境并安装依赖
  100. source "$VENV_DIR/bin/activate"
  101. pip install --upgrade pip -i https://pypi.tuna.tsinghua.edu.cn/simple
  102. pip install -r requirements.txt -i https://pypi.tuna.tsinghua.edu.cn/simple
  103. if [ $? -ne 0 ]; then
  104. echo "[ERROR] 依赖安装失败!"
  105. exit 1
  106. fi
  107. echo "[SUCCESS] Python 环境配置完成。"
  108. echo ""
  109. # 8. 创建日志目录
  110. echo "[8/9] 准备运行环境..."
  111. mkdir -p "$DEPLOY_DIR/logs"
  112. mkdir -p "$DEPLOY_DIR/static"
  113. chmod +x "$DEPLOY_DIR/main.py" 2>/dev/null || true
  114. echo "[SUCCESS] 运行环境准备完成。"
  115. echo ""
  116. # 9. 启动新服务
  117. echo "[9/9] 启动新服务..."
  118. cd "$DEPLOY_DIR"
  119. # 使用 nohup 启动服务(生产环境建议使用 systemd 或 supervisor)
  120. nohup "$VENV_DIR/bin/python" -m uvicorn main:app \
  121. --host 0.0.0.0 \
  122. --port $SERVICE_PORT \
  123. --workers 4 \
  124. > logs/app.log 2>&1 &
  125. sleep 3
  126. # 检查服务状态
  127. if pgrep -f "uvicorn main:app" > /dev/null 2>&1; then
  128. echo "[SUCCESS] 服务已启动!"
  129. echo ""
  130. echo "========================================================"
  131. echo " 生产环境部署完成!"
  132. echo "========================================================"
  133. echo ""
  134. echo "服务状态: 运行中"
  135. echo "服务端口: $SERVICE_PORT"
  136. echo "部署目录: $DEPLOY_DIR"
  137. echo "日志文件: $DEPLOY_DIR/logs/app.log"
  138. echo "虚拟环境: $VENV_DIR"
  139. echo ""
  140. echo "查看日志: tail -f $DEPLOY_DIR/logs/app.log"
  141. echo "查看进程: ps aux | grep uvicorn"
  142. echo "停止服务: pkill -f 'uvicorn main:app'"
  143. echo ""
  144. echo "[建议] 生产环境请使用 systemd 管理服务"
  145. echo " 参考: $DEPLOY_DIR/deploy/shudao-chat-py.service"
  146. echo ""
  147. else
  148. echo "[ERROR] 服务启动失败!"
  149. echo "请检查日志:"
  150. tail -30 "$DEPLOY_DIR/logs/app.log"
  151. exit 1
  152. fi