溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

docker容器中怎么捕獲信號(hào)

發(fā)布時(shí)間:2022-05-25 11:21:19 來源:億速云 閱讀:179 作者:iii 欄目:大數(shù)據(jù)

本文小編為大家詳細(xì)介紹“docker容器中怎么捕獲信號(hào)”,內(nèi)容詳細(xì),步驟清晰,細(xì)節(jié)處理妥當(dāng),希望這篇“docker容器中怎么捕獲信號(hào)”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學(xué)習(xí)新知識(shí)吧。

信號(hào)(linux)

信號(hào)是一種進(jìn)程間通信的形式。一個(gè)信號(hào)就是內(nèi)核發(fā)送給進(jìn)程的一個(gè)消息,告訴進(jìn)程發(fā)生了某種事件。當(dāng)一個(gè)信號(hào)被發(fā)送給一個(gè)進(jìn)程后,進(jìn)程會(huì)立即中斷當(dāng)前的執(zhí)行流并開始執(zhí)行信號(hào)的處理程序。如果沒有為這個(gè)信號(hào)指定處理程序,就執(zhí)行默認(rèn)的處理程序。

進(jìn)程需要為自己感興趣的信號(hào)注冊(cè)處理程序,比如為了能讓程序優(yōu)雅的退出(接到退出的請(qǐng)求后能夠?qū)Y源進(jìn)行清理)一般程序都會(huì)處理 sigterm 信號(hào)。與 sigterm 信號(hào)不同,sigkill 信號(hào)會(huì)粗暴的結(jié)束一個(gè)進(jìn)程。因此我們的應(yīng)用應(yīng)該實(shí)現(xiàn)這樣的目錄:捕獲并處理 sigterm 信號(hào),從而優(yōu)雅的退出程序。如果我們失敗了,用戶就只能通過 sigkill 信號(hào)這一終極手段了。除了 sigterm 和 sigkill ,還有像 sigusr1 這樣的專門支持用戶自定義行為的信號(hào)。下面的代碼簡單的說明在 nodejs 中如何為一個(gè)信號(hào)注冊(cè)處理程序:

process.on('sigterm', function() {
 console.log('shutting down...');
});

關(guān)于信號(hào)的更多信息,筆者在《》一文中有所提及,這里不再贅述。

容器中的信號(hào)

docker 的 stop 和 kill 命令都是用來向容器發(fā)送信號(hào)的。注意,只有容器中的 1 號(hào)進(jìn)程能夠收到信號(hào),這一點(diǎn)非常關(guān)鍵!
stop 命令會(huì)首先發(fā)送 sigterm 信號(hào),并等待應(yīng)用優(yōu)雅的結(jié)束。如果發(fā)現(xiàn)應(yīng)用沒有結(jié)束(用戶可以指定等待的時(shí)間),就再發(fā)送一個(gè) sigkill 信號(hào)強(qiáng)行結(jié)束程序。

kill 命令默認(rèn)發(fā)送的是 sigkill 信號(hào),當(dāng)然你可以通過 -s 選項(xiàng)指定任何信號(hào)。

下面我們通過一個(gè) nodejs 應(yīng)用演示信號(hào)在容器中的工作過程。創(chuàng)建 app.js 文件,內(nèi)容如下:

'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]);
 });
});

這個(gè)應(yīng)用是一個(gè) http 服務(wù)器,監(jiān)聽端口 3000,為 sigint 和 sigterm 信號(hào)注冊(cè)了處理程序。接下來我們將介紹以不同的方式在容器中運(yùn)行程序時(shí)信號(hào)的處理情況。

應(yīng)用程序作為容器中的 1 號(hào)進(jìn)程

創(chuàng)建 dockerfile 文件,把上面的應(yīng)用打包到鏡像中:

from iojs:onbuild
copy ./app.js ./app.js
copy ./package.json ./package.json
expose 3000
entrypoint ["node", "app"]

請(qǐng)注意 entrypoint 指令的寫法,這種寫法會(huì)讓 node 在容器中以 1 號(hào)進(jìn)程的身份運(yùn)行。

接下來創(chuàng)建鏡像:

$ docker build --no-cache -t signal-app -f dockerfile .

然后啟動(dòng)容器運(yùn)行應(yīng)用程序:

$ docker run -it --rm -p 3000:3000 --name="my-app" signal-app

此時(shí) node 應(yīng)用在容器中的進(jìn)程號(hào)為 1:

docker容器中怎么捕獲信號(hào)

現(xiàn)在我們讓程序退出,執(zhí)行命令:

$ docker container kill --signal="sigterm" my-app

此時(shí)應(yīng)用會(huì)以我們期望的方式退出:

docker容器中怎么捕獲信號(hào)

應(yīng)用程序不是容器中的 1 號(hào)進(jìn)程

創(chuàng)建一個(gè)啟動(dòng)應(yīng)用程序的腳本文件 app1.sh,內(nèi)容如下:

#!/usr/bin/env bash
node app

然后創(chuàng)建 dockerfile1 文件,內(nèi)容如下:

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"]

接下來創(chuàng)建鏡像:

$ docker build --no-cache -t signal-app1 -f dockerfile1 .

然后啟動(dòng)容器運(yùn)行應(yīng)用程序:

$ docker run -it --rm -p 3000:3000 --name="my-app1" signal-app1

此時(shí) node 應(yīng)用在容器中的進(jìn)程號(hào)不再是 1:

docker容器中怎么捕獲信號(hào)

現(xiàn)在給 my-app1 發(fā)送 sigterm 信號(hào)試試,已經(jīng)無法退出程序了!在這個(gè)場景中,應(yīng)用程序由 bash 腳本啟動(dòng),bash 作為容器中的 1 號(hào)進(jìn)程收到了 sigterm  信號(hào),但是它沒有做出任何的響應(yīng)動(dòng)作。

我們可以通過:

$ docker container stop my-app1
# or
$ docker container kill --signal="sigkill" my-app1

退出應(yīng)用,它們最終都是向容器中的 1 號(hào)進(jìn)程發(fā)送了 sigkill 信號(hào)。很顯然這不是我們期望的,我們希望程序能夠收到 sigterm  信號(hào)優(yōu)雅的退出。

在腳本中捕獲信號(hào)

創(chuàng)建另外一個(gè)啟動(dòng)應(yīng)用程序的腳本文件 app2.sh,內(nèi)容如下:

#!/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

這個(gè)腳本文件在啟動(dòng)應(yīng)用程序的同時(shí)可以捕獲發(fā)送給它的 sigterm 和 sigusr1 信號(hào),并為它們添加了處理程序。其中 sigterm 信號(hào)的處理程序就是向我們的 node 應(yīng)用程序發(fā)送 sigterm 信號(hào)。

然后創(chuàng)建 dockerfile2 文件,內(nèi)容如下:

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"]

接下來創(chuàng)建鏡像:

$ docker build --no-cache -t signal-app2 -f dockerfile2 .

然后啟動(dòng)容器運(yùn)行應(yīng)用程序:

$ docker run -it --rm -p 3000:3000 --name="my-app2" signal-app2

此時(shí) node 應(yīng)用在容器中的進(jìn)程號(hào)也不是 1,但是它卻可以接收到 sigterm 信號(hào)并優(yōu)雅的退出了:

docker容器中怎么捕獲信號(hào)

讀到這里,這篇“docker容器中怎么捕獲信號(hào)”文章已經(jīng)介紹完畢,想要掌握這篇文章的知識(shí)點(diǎn)還需要大家自己動(dòng)手實(shí)踐使用過才能領(lǐng)會(huì),如果想了解更多相關(guān)內(nèi)容的文章,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI