Shell腳本控制docker容器啓動順序

1.遇到的問題

在分佈式項目部署的過程當中,常常要求服務器重啓以後,應用(包括數據庫)可以自動恢復使用.雖然使用docker update --restart=always containerid可以讓容器自動隨docker啓動,可是並不能保證是在數據庫啓動以後啓動,若是數據庫未啓動,那麼將致使應用啓動失敗;網上還有一種解決方法是經過docker-compose容器編排來控制啓動順序,這個博主研究的比較少.html

2.解決思路

使用Shell腳原本控制,思路大體以下linux

  1. 探測數據庫端口來檢驗數據庫是否啓動成功.
  2. 數據庫啓動成功後,探測配置中心及服務註冊中心的端口來檢驗其是否啓動成功.
  3. 當數據庫及配置中心都啓動以後,再啓動其餘微服務應用.

3.端口探測

端口探測使用的命令是docker

nc -w 1 host port </dev/null && echo "200"shell

host:目標主機的ip數據庫

port:服務監聽的端口bash

若是服務啓動了 這條命令會返回 200,未啓動則返回空.服務器

4.Shell腳本

直接貼代碼了,使用的配置中心是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

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),能夠這樣寫:

command > file 2>&1
或者
command >> file 2>&1

/dev/null文件

/dev/null是一個特殊的文件,寫入到它的內容都會被丟棄;若是嘗試從該文件讀取內容,那麼什麼也讀不到。可是 /dev/null 文件很是有用,將命令的輸出重定向到它,會起到禁止輸出的效果

command > /dev/null 2>&1 能夠屏蔽stdout和stderr

參考

菜鳥教程-Shell

相關文章
相關標籤/搜索