在分佈式項目部署的過程當中,常常要求服務器重啓以後,應用(包括數據庫)可以自動恢復使用.雖然使用docker update --restart=always containerid
可以讓容器自動隨docker啓動,可是並不能保證是在數據庫啓動以後啓動,若是數據庫未啓動,那麼將致使應用啓動失敗;網上還有一種解決方法是經過docker-compose容器編排來控制啓動順序,這個博主研究的比較少.html
使用Shell腳原本控制,思路大體以下linux
端口探測使用的命令是docker
nc -w 1 host port </dev/null && echo "200"
shell
host:目標主機的ip數據庫
port:服務監聽的端口bash
若是服務啓動了 這條命令會返回 200,未啓動則返回空.服務器
直接貼代碼了,使用的配置中心是nacos運維
#!/bin/bash #chkconfig: 2345 80 90 #description:autoStartMaintenanceService.sh # #前提: #1.docker必須能開機自啓 #2.docker可以正常啓動運維服務 #3.此腳本須運行微服務所在的機器上 # ##須要修改的配置-----開始 ##數據庫所在的機器IP DATABASE_HOST=192.169.1.52 ##數據庫監聽的端口 DATABASE_PORT=3306 ##微服務所在機器IP LOCAL_HOST=192.169.1.46 ##微服務訪問端口 Maintenance_Port=8180 ##NACOS所在機器的ip NACOS_HOST=192.169.1.82 ##NACOS的監聽端口 NACOS_PORT=8848 ##微服務容器名稱(NAMES列) Maintenance_Container_Name="umc-maintenance" ##該腳本生成的日誌路徑 Log_Path=/home/test/log ##須要修改的配置-----結束 ## ##循環延時時間(s)秒 LOOP_TIME=5 at_time="" at_date="" getAtTime() { at_time="$(date +%Y-%m-%d-%H:%M:%S) --- " at_date=$(date +%Y-%m-%d) } autoStartWebService() { ##若是日誌路徑不存在則建立 if [ ! -d "$Log_Path" ]; then mkdir -p $Log_Path fi while true; do ##判斷數據庫是否啓動 req_message=$(nc -w 1 ${DATABASE_HOST} ${DATABASE_PORT} </dev/null && echo "200") if [ -n "$req_message" ]; then getAtTime echo "$at_time Database is running" >>${Log_Path}/"$at_date"_autoStartMaintenanceService.log waitNacosStarting else getAtTime echo "$at_time Database is not running and please wait for Database starting" >>${Log_Path}/"$at_date"_autoStartMaintenanceService.log sleep $LOOP_TIME fi done } ##判斷Nacos是否啓動 waitNacosStarting() { req_message=$(nc -w 1 ${NACOS_HOST} ${NACOS_PORT} </dev/null && echo "200") if test $((req_message)) -eq 200; then getAtTime echo "$at_time Nacos is running" >>${Log_Path}/"$at_date"_autoStartMaintenanceService.log startMaintenanceService sleep $LOOP_TIME else getAtTime echo "$at_time Nacos is not running and please wait for nacos starting" >>${Log_Path}/"$at_date"_autoStartMaintenanceService.log sleep $LOOP_TIME fi } ##啓動微服務 startMaintenanceService() { req_message=$(nc -w 1 ${LOCAL_HOST} ${Maintenance_Port} </dev/null && echo "200") if test $((req_message)) -eq 200; then getAtTime echo "$at_time Maintenance service is running" >>${Log_Path}/"$at_date"_autoStartMaintenanceService.log else container_id=$(docker ps -a | grep $Maintenance_Container_Name | grep -v grep | awk '{print $1}') getAtTime echo "$at_time Maintenance service container id is ${container_id}" >>${Log_Path}/"$at_date"_autoStartMaintenanceService.log docker start ${container_id} fi } autoStartWebService
寫這個腳本的時候,也讓博主對Shell輸入輸出重定向更加熟悉分佈式
通常狀況下,每一個 Unix/Linux 命令運行時都會打開三個文件:微服務
命令 | 說明 |
---|---|
command > file | 將輸出重定向到 file且會覆蓋file |
command < file | 將輸入重定向到 file |
command >> file | 將輸出以追加的方式重定向到file |
command 2> file | 將錯誤輸出到file且會覆蓋file |
command 2>> file | 將錯誤以追加的方式重定向到file |
<< tag | 將開始標記 tag 和結束標記 tag 之間的內容做爲輸入 |
若是但願將 stdout 和 stderr 合併後重定向到 file(即將正確信息和錯誤信息都輸出到file),能夠這樣寫:
command > file 2>&1 或者 command >> file 2>&1
/dev/null是一個特殊的文件,寫入到它的內容都會被丟棄;若是嘗試從該文件讀取內容,那麼什麼也讀不到。可是 /dev/null 文件很是有用,將命令的輸出重定向到它,會起到禁止輸出的效果
command > /dev/null 2>&1
能夠屏蔽stdout和stderr