start.sh 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. #!/bin/bash
  2. # ========================================
  3. # 拾音器异响检测系统启动脚本
  4. # ========================================
  5. #
  6. # 使用方法:
  7. # ./start.sh # 前台运行
  8. # ./start.sh -d # 后台运行
  9. # ./start.sh --daemon # 后台运行
  10. # ./start.sh stop # 停止服务
  11. # ./start.sh restart # 重启服务
  12. # ./start.sh status # 查看状态
  13. #
  14. # 日志文件:
  15. # 前台运行:直接输出到控制台
  16. # 后台运行:logs/system.log(RotatingFileHandler 自动轮转)
  17. #
  18. # 切换到脚本所在目录
  19. cd "$(dirname "$0")"
  20. # PID文件路径
  21. PID_FILE="logs/pid.txt"
  22. # ========================================
  23. # 函数:激活conda环境
  24. # ========================================
  25. activate_conda() {
  26. if command -v conda &> /dev/null; then
  27. # 激活 conda 环境
  28. source $(conda info --base)/etc/profile.d/conda.sh
  29. conda activate pump_asd
  30. echo "已激活 conda 环境: pump_asd"
  31. fi
  32. }
  33. # ========================================
  34. # 函数:检查进程是否运行
  35. # ========================================
  36. is_running() {
  37. if [ -f "$PID_FILE" ]; then
  38. PID=$(cat "$PID_FILE")
  39. # 检查进程是否存在
  40. if ps -p "$PID" > /dev/null 2>&1; then
  41. return 0 # 运行中
  42. fi
  43. fi
  44. return 1 # 未运行
  45. }
  46. # ========================================
  47. # 函数:获取当前PID
  48. # ========================================
  49. get_pid() {
  50. if [ -f "$PID_FILE" ]; then
  51. cat "$PID_FILE"
  52. else
  53. echo ""
  54. fi
  55. }
  56. # ========================================
  57. # 函数:启动服务
  58. # ========================================
  59. start_service() {
  60. # 检查是否已经运行
  61. if is_running; then
  62. echo "服务已在运行中, PID: $(get_pid)"
  63. echo "如需重启,请使用: ./start.sh restart"
  64. return 1
  65. fi
  66. # 激活conda环境
  67. activate_conda
  68. # 检查必要文件
  69. if [ ! -f "run_pickup_monitor.py" ]; then
  70. echo "错误: run_pickup_monitor.py 不存在"
  71. exit 1
  72. fi
  73. if [ ! -f "config/pickup_config.db" ]; then
  74. echo "错误: config/pickup_config.db 不存在"
  75. echo "请先运行迁移脚本: python tool/migrate_yaml_to_db.py"
  76. exit 1
  77. fi
  78. # 创建日志目录
  79. mkdir -p logs
  80. # 启动服务
  81. echo "后台运行模式..."
  82. # stdout/stderr 丢弃,所有日志由 RotatingFileHandler 写入 logs/system.log
  83. nohup python run_pickup_monitor.py > /dev/null 2>&1 &
  84. PID=$!
  85. echo $PID > "$PID_FILE"
  86. # 等待1秒检查是否正常启动
  87. sleep 1
  88. if ps -p "$PID" > /dev/null 2>&1; then
  89. echo "服务启动成功, PID: $PID"
  90. echo "日志文件: logs/system.log"
  91. echo ""
  92. echo "查看日志: tail -f logs/system.log"
  93. echo "停止服务: ./start.sh stop"
  94. echo "重启服务: ./start.sh restart"
  95. else
  96. echo "服务启动失败,请检查日志: logs/system.log"
  97. rm -f "$PID_FILE"
  98. return 1
  99. fi
  100. }
  101. # ========================================
  102. # 函数:停止服务
  103. # ========================================
  104. stop_service() {
  105. if ! is_running; then
  106. echo "服务未运行"
  107. rm -f "$PID_FILE"
  108. return 0
  109. fi
  110. PID=$(get_pid)
  111. echo "正在停止服务, PID: $PID"
  112. # 发送 SIGTERM 信号,优雅停止
  113. kill "$PID" 2>/dev/null
  114. # 等待进程结束(最多等待10秒)
  115. WAIT_COUNT=0
  116. while ps -p "$PID" > /dev/null 2>&1; do
  117. if [ $WAIT_COUNT -ge 10 ]; then
  118. echo "进程未响应,强制终止..."
  119. kill -9 "$PID" 2>/dev/null
  120. break
  121. fi
  122. sleep 1
  123. WAIT_COUNT=$((WAIT_COUNT + 1))
  124. echo "等待进程结束... ($WAIT_COUNT/10)"
  125. done
  126. rm -f "$PID_FILE"
  127. echo "服务已停止"
  128. }
  129. # ========================================
  130. # 函数:重启服务
  131. # ========================================
  132. restart_service() {
  133. echo "=========================================="
  134. echo "重启拾音器异响检测服务"
  135. echo "=========================================="
  136. stop_service
  137. echo ""
  138. sleep 2 # 等待2秒确保资源完全释放
  139. start_service
  140. }
  141. # ========================================
  142. # 函数:查看服务状态
  143. # ========================================
  144. show_status() {
  145. echo "=========================================="
  146. echo "拾音器异响检测服务状态"
  147. echo "=========================================="
  148. if is_running; then
  149. PID=$(get_pid)
  150. echo "状态: 运行中"
  151. echo "PID: $PID"
  152. echo ""
  153. # 显示进程信息
  154. echo "进程详情:"
  155. ps -p "$PID" -o pid,ppid,user,%cpu,%mem,etime,command | head -2
  156. echo ""
  157. # 显示最近日志
  158. echo "最近10行日志:"
  159. echo "------------------------------------------"
  160. tail -10 logs/system.log 2>/dev/null || echo "(无日志)"
  161. else
  162. echo "状态: 未运行"
  163. if [ -f "$PID_FILE" ]; then
  164. echo "注意: PID文件存在但进程已停止,可能是异常退出"
  165. rm -f "$PID_FILE"
  166. fi
  167. fi
  168. }
  169. # ========================================
  170. # 函数:前台运行
  171. # ========================================
  172. run_foreground() {
  173. # 检查是否已经运行
  174. if is_running; then
  175. echo "服务已在后台运行中, PID: $(get_pid)"
  176. echo "请先停止: ./start.sh stop"
  177. return 1
  178. fi
  179. # 激活conda环境
  180. activate_conda
  181. # 检查必要文件
  182. if [ ! -f "run_pickup_monitor.py" ]; then
  183. echo "错误: run_pickup_monitor.py 不存在"
  184. exit 1
  185. fi
  186. if [ ! -f "config/pickup_config.db" ]; then
  187. echo "错误: config/pickup_config.db 不存在"
  188. echo "请先运行迁移脚本: python tool/migrate_yaml_to_db.py"
  189. exit 1
  190. fi
  191. # 创建日志目录
  192. mkdir -p logs
  193. echo "前台运行模式..."
  194. python run_pickup_monitor.py
  195. }
  196. # ========================================
  197. # 函数:显示帮助
  198. # ========================================
  199. show_help() {
  200. echo "拾音器异响检测系统 - 启动脚本"
  201. echo ""
  202. echo "用法: ./start.sh [命令]"
  203. echo ""
  204. echo "命令:"
  205. echo " (无参数) 前台运行"
  206. echo " -d, --daemon 后台运行"
  207. echo " start 后台启动服务"
  208. echo " stop 停止服务"
  209. echo " restart 重启服务"
  210. echo " status 查看服务状态"
  211. echo " help 显示帮助信息"
  212. echo ""
  213. echo "示例:"
  214. echo " ./start.sh -d # 后台启动"
  215. echo " ./start.sh restart # 重启服务"
  216. echo " ./start.sh status # 查看状态"
  217. }
  218. # ========================================
  219. # 主逻辑
  220. # ========================================
  221. case "$1" in
  222. stop)
  223. stop_service
  224. ;;
  225. restart)
  226. restart_service
  227. ;;
  228. status)
  229. show_status
  230. ;;
  231. start|-d|--daemon)
  232. start_service
  233. ;;
  234. help|--help|-h)
  235. show_help
  236. ;;
  237. "")
  238. run_foreground
  239. ;;
  240. *)
  241. echo "未知命令: $1"
  242. echo ""
  243. show_help
  244. exit 1
  245. ;;
  246. esac