deploy.sh 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. #!/bin/bash
  2. set -e
  3. DATE=$(date +%Y%m%d%H%M)
  4. # 基础路径
  5. BASE_PATH=/work/projects/yudao-server
  6. # 编译后 jar 的地址。部署时,Jenkins 会上传 jar 包到该目录下
  7. SOURCE_PATH=$BASE_PATH/build
  8. # 服务名称。同时约定部署服务的 jar 包名字也为它。
  9. SERVER_NAME=yudao-server
  10. # 环境
  11. PROFILES_ACTIVE=development
  12. # 健康检查 URL
  13. HEALTH_CHECK_URL=http://127.0.0.1:48080/actuator/health/
  14. # heapError 存放路径
  15. HEAP_ERROR_PATH=$BASE_PATH/heapError
  16. # JVM 参数
  17. JAVA_OPS="-Xms512m -Xmx512m -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=$HEAP_ERROR_PATH"
  18. # SkyWalking Agent 配置
  19. #export SW_AGENT_NAME=$SERVER_NAME
  20. #export SW_AGENT_COLLECTOR_BACKEND_SERVICES=192.168.0.84:11800
  21. #export SW_GRPC_LOG_SERVER_HOST=192.168.0.84
  22. #export SW_AGENT_TRACE_IGNORE_PATH="Redisson/PING,/actuator/**,/admin/**"
  23. #export JAVA_AGENT=-javaagent:/work/skywalking/apache-skywalking-apm-bin/agent/skywalking-agent.jar
  24. # 备份
  25. function backup() {
  26. # 如果不存在,则无需备份
  27. if [ ! -f "$BASE_PATH/$SERVER_NAME.jar" ]; then
  28. echo "[backup] $BASE_PATH/$SERVER_NAME.jar 不存在,跳过备份"
  29. # 如果存在,则备份到 backup 目录下,使用时间作为后缀
  30. else
  31. echo "[backup] 开始备份 $SERVER_NAME ..."
  32. cp $BASE_PATH/$SERVER_NAME.jar $BASE_PATH/backup/$SERVER_NAME-$DATE.jar
  33. echo "[backup] 备份 $SERVER_NAME 完成"
  34. fi
  35. }
  36. # 最新构建代码 移动到项目环境
  37. function transfer() {
  38. echo "[transfer] 开始转移 $SERVER_NAME.jar"
  39. # 删除原 jar 包
  40. if [ ! -f "$BASE_PATH/$SERVER_NAME.jar" ]; then
  41. echo "[transfer] $BASE_PATH/$SERVER_NAME.jar 不存在,跳过删除"
  42. else
  43. echo "[transfer] 移除 $BASE_PATH/$SERVER_NAME.jar 完成"
  44. rm $BASE_PATH/$SERVER_NAME.jar
  45. fi
  46. # 复制新 jar 包
  47. echo "[transfer] 从 $SOURCE_PATH 中获取 $SERVER_NAME.jar 并迁移至 $BASE_PATH ...."
  48. cp $SOURCE_PATH/$SERVER_NAME.jar $BASE_PATH
  49. echo "[transfer] 转移 $SERVER_NAME.jar 完成"
  50. }
  51. # 停止:优雅关闭之前已经启动的服务
  52. function stop() {
  53. echo "[stop] 开始停止 $BASE_PATH/$SERVER_NAME"
  54. PID=$(ps -ef | grep $BASE_PATH/$SERVER_NAME | grep -v "grep" | awk '{print $2}')
  55. # 如果 Java 服务启动中,则进行关闭
  56. if [ -n "$PID" ]; then
  57. # 正常关闭
  58. echo "[stop] $BASE_PATH/$SERVER_NAME 运行中,开始 kill [$PID]"
  59. kill -15 $PID
  60. # 等待最大 120 秒,直到关闭完成。
  61. for ((i = 0; i < 120; i++))
  62. do
  63. sleep 1
  64. PID=$(ps -ef | grep $BASE_PATH/$SERVER_NAME | grep -v "grep" | awk '{print $2}')
  65. if [ -n "$PID" ]; then
  66. echo -e ".\c"
  67. else
  68. echo "[stop] 停止 $BASE_PATH/$SERVER_NAME 成功"
  69. break
  70. fi
  71. done
  72. # 如果正常关闭失败,那么进行强制 kill -9 进行关闭
  73. if [ -n "$PID" ]; then
  74. echo "[stop] $BASE_PATH/$SERVER_NAME 失败,强制 kill -9 $PID"
  75. kill -9 $PID
  76. fi
  77. # 如果 Java 服务未启动,则无需关闭
  78. else
  79. echo "[stop] $BASE_PATH/$SERVER_NAME 未启动,无需停止"
  80. fi
  81. }
  82. # 启动:启动后端项目
  83. function start() {
  84. # 开启启动前,打印启动参数
  85. echo "[start] 开始启动 $BASE_PATH/$SERVER_NAME"
  86. echo "[start] JAVA_OPS: $JAVA_OPS"
  87. echo "[start] JAVA_AGENT: $JAVA_AGENT"
  88. echo "[start] PROFILES: $PROFILES_ACTIVE"
  89. # 开始启动
  90. BUILD_ID=dontKillMe nohup java -server $JAVA_OPS $JAVA_AGENT -jar $BASE_PATH/$SERVER_NAME.jar --spring.profiles.active=$PROFILES_ACTIVE &
  91. echo "[start] 启动 $BASE_PATH/$SERVER_NAME 完成"
  92. }
  93. # 健康检查:自动判断后端项目是否正常启动
  94. function healthCheck() {
  95. # 如果配置健康检查,则进行健康检查
  96. if [ -n "$HEALTH_CHECK_URL" ]; then
  97. # 健康检查最大 120 秒,直到健康检查通过
  98. echo "[healthCheck] 开始通过 $HEALTH_CHECK_URL 地址,进行健康检查";
  99. for ((i = 0; i < 120; i++))
  100. do
  101. # 请求健康检查地址,只获取状态码。
  102. result=`curl -I -m 10 -o /dev/null -s -w %{http_code} $HEALTH_CHECK_URL || echo "000"`
  103. # 如果状态码为 200,则说明健康检查通过
  104. if [ "$result" == "200" ]; then
  105. echo "[healthCheck] 健康检查通过";
  106. break
  107. # 如果状态码非 200,则说明未通过。sleep 1 秒后,继续重试
  108. else
  109. echo -e ".\c"
  110. sleep 1
  111. fi
  112. done
  113. # 健康检查未通过,则异常退出 shell 脚本,不继续部署。
  114. if [ ! "$result" == "200" ]; then
  115. echo "[healthCheck] 健康检查不通过,可能部署失败。查看日志,自行判断是否启动成功";
  116. tail -n 10 nohup.out
  117. exit 1;
  118. # 健康检查通过,打印最后 10 行日志,可能部署的人想看下日志。
  119. else
  120. tail -n 10 nohup.out
  121. fi
  122. # 如果未配置健康检查,则 sleep 120 秒,人工看日志是否部署成功。
  123. else
  124. echo "[healthCheck] HEALTH_CHECK_URL 未配置,开始 sleep 120 秒";
  125. sleep 120
  126. echo "[healthCheck] sleep 120 秒完成,查看日志,自行判断是否启动成功";
  127. tail -n 50 nohup.out
  128. fi
  129. }
  130. # 部署
  131. function deploy() {
  132. cd $BASE_PATH
  133. # 备份原 jar
  134. backup
  135. # 停止 Java 服务
  136. stop
  137. # 部署新 jar
  138. transfer
  139. # 启动 Java 服务
  140. start
  141. # 健康检查
  142. healthCheck
  143. }
  144. deploy