您好,登錄后才能下訂單哦!
本文小編為大家詳細介紹“docker容器中怎么捕獲信號”,內容詳細,步驟清晰,細節處理妥當,希望這篇“docker容器中怎么捕獲信號”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學習新知識吧。
信號(linux)
信號是一種進程間通信的形式。一個信號就是內核發送給進程的一個消息,告訴進程發生了某種事件。當一個信號被發送給一個進程后,進程會立即中斷當前的執行流并開始執行信號的處理程序。如果沒有為這個信號指定處理程序,就執行默認的處理程序。
進程需要為自己感興趣的信號注冊處理程序,比如為了能讓程序優雅的退出(接到退出的請求后能夠對資源進行清理)一般程序都會處理 sigterm 信號。與 sigterm 信號不同,sigkill 信號會粗暴的結束一個進程。因此我們的應用應該實現這樣的目錄:捕獲并處理 sigterm 信號,從而優雅的退出程序。如果我們失敗了,用戶就只能通過 sigkill 信號這一終極手段了。除了 sigterm 和 sigkill ,還有像 sigusr1 這樣的專門支持用戶自定義行為的信號。下面的代碼簡單的說明在 nodejs 中如何為一個信號注冊處理程序:
process.on('sigterm', function() { console.log('shutting down...'); });
關于信號的更多信息,筆者在《》一文中有所提及,這里不再贅述。
容器中的信號
docker 的 stop 和 kill 命令都是用來向容器發送信號的。注意,只有容器中的 1 號進程能夠收到信號,這一點非常關鍵!
stop 命令會首先發送 sigterm 信號,并等待應用優雅的結束。如果發現應用沒有結束(用戶可以指定等待的時間),就再發送一個 sigkill 信號強行結束程序。
kill 命令默認發送的是 sigkill 信號,當然你可以通過 -s 選項指定任何信號。
下面我們通過一個 nodejs 應用演示信號在容器中的工作過程。創建 app.js 文件,內容如下:
'use strict'; var http = require('http'); var server = http.createserver(function (req, res) { res.writehead(200, {'content-type': 'text/plain'}); res.end('hello world\n'); }).listen(3000, '0.0.0.0'); console.log('server started'); var signals = { 'sigint': 2, 'sigterm': 15 }; function shutdown(signal, value) { server.close(function () { console.log('server stopped by ' + signal); process.exit(128 + value); }); } object.keys(signals).foreach(function (signal) { process.on(signal, function () { shutdown(signal, signals[signal]); }); });
這個應用是一個 http 服務器,監聽端口 3000,為 sigint 和 sigterm 信號注冊了處理程序。接下來我們將介紹以不同的方式在容器中運行程序時信號的處理情況。
應用程序作為容器中的 1 號進程
創建 dockerfile 文件,把上面的應用打包到鏡像中:
from iojs:onbuild copy ./app.js ./app.js copy ./package.json ./package.json expose 3000 entrypoint ["node", "app"]
請注意 entrypoint 指令的寫法,這種寫法會讓 node 在容器中以 1 號進程的身份運行。
接下來創建鏡像:
$ docker build --no-cache -t signal-app -f dockerfile .
然后啟動容器運行應用程序:
$ docker run -it --rm -p 3000:3000 --name="my-app" signal-app
此時 node 應用在容器中的進程號為 1:
現在我們讓程序退出,執行命令:
$ docker container kill --signal="sigterm" my-app
此時應用會以我們期望的方式退出:
應用程序不是容器中的 1 號進程
創建一個啟動應用程序的腳本文件 app1.sh,內容如下:
#!/usr/bin/env bash node app
然后創建 dockerfile1 文件,內容如下:
from iojs:onbuild copy ./app.js ./app.js copy ./app1.sh ./app1.sh copy ./package.json ./package.json run chmod +x ./app1.sh expose 3000 entrypoint ["./app1.sh"]
接下來創建鏡像:
$ docker build --no-cache -t signal-app1 -f dockerfile1 .
然后啟動容器運行應用程序:
$ docker run -it --rm -p 3000:3000 --name="my-app1" signal-app1
此時 node 應用在容器中的進程號不再是 1:
現在給 my-app1 發送 sigterm 信號試試,已經無法退出程序了!在這個場景中,應用程序由 bash 腳本啟動,bash 作為容器中的 1 號進程收到了 sigterm 信號,但是它沒有做出任何的響應動作。
我們可以通過:
$ docker container stop my-app1 # or $ docker container kill --signal="sigkill" my-app1
退出應用,它們最終都是向容器中的 1 號進程發送了 sigkill 信號。很顯然這不是我們期望的,我們希望程序能夠收到 sigterm 信號優雅的退出。
在腳本中捕獲信號
創建另外一個啟動應用程序的腳本文件 app2.sh,內容如下:
#!/usr/bin/env bash set -x pid=0 # sigusr1-handler my_handler() { echo "my_handler" } # sigterm-handler term_handler() { if [ $pid -ne 0 ]; then kill -sigterm "$pid" wait "$pid" fi exit 143; # 128 + 15 -- sigterm } # setup handlers # on callback, kill the last background process, which is `tail -f /dev/null` and execute the specified handler trap 'kill ${!}; my_handler' sigusr1 trap 'kill ${!}; term_handler' sigterm # run application node app & pid="$!" # wait forever while true do tail -f /dev/null & wait ${!} done
這個腳本文件在啟動應用程序的同時可以捕獲發送給它的 sigterm 和 sigusr1 信號,并為它們添加了處理程序。其中 sigterm 信號的處理程序就是向我們的 node 應用程序發送 sigterm 信號。
然后創建 dockerfile2 文件,內容如下:
from iojs:onbuild copy ./app.js ./app.js copy ./app2.sh ./app2.sh copy ./package.json ./package.json run chmod +x ./app2.sh expose 3000 entrypoint ["./app2.sh"]
接下來創建鏡像:
$ docker build --no-cache -t signal-app2 -f dockerfile2 .
然后啟動容器運行應用程序:
$ docker run -it --rm -p 3000:3000 --name="my-app2" signal-app2
此時 node 應用在容器中的進程號也不是 1,但是它卻可以接收到 sigterm 信號并優雅的退出了:
讀到這里,這篇“docker容器中怎么捕獲信號”文章已經介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領會,如果想了解更多相關內容的文章,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。