Shell指令碼控制docker容器啟動順序
阿新 • • 發佈:2021-03-05
## 1.遇到的問題
在分散式專案部署的過程中,經常要求伺服器重啟之後,應用(包括資料庫)能夠自動恢復使用.雖然使用`docker update --restart=always containerid`能夠讓容器自動隨docker啟動,但是並不能保證是在資料庫啟動之後啟動,如果資料庫未啟動,那麼將導致應用啟動失敗;網上還有一種解決方法是通過docker-compose容器編排來控制啟動順序,這個博主研究的比較少.
## 2.解決思路
使用Shell指令碼來控制,思路大致如下
1. 探測資料庫埠來檢驗資料庫是否啟動成功.
2. 資料庫啟動成功後,探測配置中心及服務註冊中心的埠來檢驗其是否啟動成功.
3. 當資料庫及配置中心都啟動之後,再啟動其他微服務應用.
## 3.埠探測
埠探測使用的命令是
`nc -w 1 host port >${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} >${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} >${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
```
## 5.Shell輸入輸出重定向
寫這個指令碼的時候,也讓博主對Shell輸入輸出重定向更加熟悉
一般情況下,每個 Unix/Linux 命令執行時都會開啟三個檔案:
+ 標準輸入檔案(stdin):stdin的檔案描述符為0,Unix程式預設從stdin讀取資料。
+ 標準輸出檔案(stdout):stdout 的檔案描述符為1,Unix程式預設向stdout輸出資料。
+ 標準錯誤檔案(stderr):stderr的檔案描述符為2,Unix程式會向stderr流中寫入錯誤資訊。
命令 | 說明
---------|----------
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),可以這樣寫:
```sh
command > file 2>&1
或者
command >> file 2>&1
```
### /dev/null檔案
**/dev/null**是一個特殊的檔案,寫入到它的內容都會被丟棄;如果嘗試從該檔案讀取內容,那麼什麼也讀不到。但是 /dev/null 檔案非常有用,將命令的輸出重定向到它,會起到**禁止輸出**的效果
`command > /dev/null 2>&1` 可以遮蔽stdout和stderr
## 參考
[菜鳥教程-Shell](https://www.runoob.com/linux/linux-shell-io-redirectio