一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

云服務器|WEB服務器|FTP服務器|郵件服務器|虛擬主機|服務器安全|DNS服務器|服務器知識|Nginx|IIS|Tomcat|

服務器之家 - 服務器技術 - 服務器知識 - 詳解如何在 docker 容器中捕獲信號

詳解如何在 docker 容器中捕獲信號

2021-02-02 17:36sparkdev 服務器知識

本篇文章主要介紹了如何在 docker 容器中捕獲信號,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

我們可能都使用過 docker stop 命令來停止正在運行的容器,有時可能會使用 docker kill 命令強行關閉容器或者把某個信號傳遞給容器中的進程。這些操作的本質都是通過從主機向容器發送信號實現主機與容器中程序的交互。比如我們可以向容器中的應用發送一個重新加載信號,容器中的應用程序在接到信號后執行相應的處理程序完成重新加載配置文件的任務。本文將介紹在 docker 容器中捕獲信號的基本知識。

信號(linux)

信號是一種進程間通信的形式。一個信號就是內核發送給進程的一個消息,告訴進程發生了某種事件。當一個信號被發送給一個進程后,進程會立即中斷當前的執行流并開始執行信號的處理程序。如果沒有為這個信號指定處理程序,就執行默認的處理程序。

進程需要為自己感興趣的信號注冊處理程序,比如為了能讓程序優雅的退出(接到退出的請求后能夠對資源進行清理)一般程序都會處理 sigterm 信號。與 sigterm 信號不同,sigkill 信號會粗暴的結束一個進程。因此我們的應用應該實現這樣的目錄:捕獲并處理 sigterm 信號,從而優雅的退出程序。如果我們失敗了,用戶就只能通過 sigkill 信號這一終極手段了。除了 sigterm 和 sigkill ,還有像 sigusr1 這樣的專門支持用戶自定義行為的信號。下面的代碼簡單的說明在 nodejs 中如何為一個信號注冊處理程序:

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

關于信號的更多信息,筆者在《linux kill 命令》一文中有所提及,這里不再贅述。

容器中的信號

docker 的 stop 和 kill 命令都是用來向容器發送信號的。注意,只有容器中的 1 號進程能夠收到信號,這一點非常關鍵!
stop 命令會首先發送 sigterm 信號,并等待應用優雅的結束。如果發現應用沒有結束(用戶可以指定等待的時間),就再發送一個 sigkill 信號強行結束程序。

kill 命令默認發送的是 sigkill 信號,當然你可以通過 -s 選項指定任何信號。

下面我們通過一個 nodejs 應用演示信號在容器中的工作過程。創建 app.js 文件,內容如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
'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 文件,把上面的應用打包到鏡像中:

?
1
2
3
4
5
from iojs:onbuild
copy ./app.js ./app.js
copy ./package.json ./package.json
expose 3000
entrypoint ["node", "app"]

請注意 entrypoint 指令的寫法,這種寫法會讓 node 在容器中以 1 號進程的身份運行。

接下來創建鏡像:

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

然后啟動容器運行應用程序:

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

此時 node 應用在容器中的進程號為 1:

詳解如何在 docker 容器中捕獲信號

現在我們讓程序退出,執行命令:

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

此時應用會以我們期望的方式退出:

詳解如何在 docker 容器中捕獲信號

應用程序不是容器中的 1 號進程

創建一個啟動應用程序的腳本文件 app1.sh,內容如下:

?
1
2
#!/usr/bin/env bash
node app

然后創建 dockerfile1 文件,內容如下:

?
1
2
3
4
5
6
7
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"]

接下來創建鏡像:

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

然后啟動容器運行應用程序:

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

此時 node 應用在容器中的進程號不再是 1:

詳解如何在 docker 容器中捕獲信號

現在給 my-app1 發送 sigterm 信號試試,已經無法退出程序了!在這個場景中,應用程序由 bash 腳本啟動,bash 作為容器中的 1 號進程收到了 sigterm  信號,但是它沒有做出任何的響應動作。

我們可以通過:

?
1
2
3
$ docker container stop my-app1
# or
$ docker container kill --signal="sigkill" my-app1

退出應用,它們最終都是向容器中的 1 號進程發送了 sigkill 信號。很顯然這不是我們期望的,我們希望程序能夠收到 sigterm  信號優雅的退出。

在腳本中捕獲信號

創建另外一個啟動應用程序的腳本文件 app2.sh,內容如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
#!/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 文件,內容如下:

?
1
2
3
4
5
6
7
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"]

接下來創建鏡像:

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

然后啟動容器運行應用程序:

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

此時 node 應用在容器中的進程號也不是 1,但是它卻可以接收到 sigterm 信號并優雅的退出了:

詳解如何在 docker 容器中捕獲信號

結論

容器中的 1 號進程是非常重要的,如果它不能正確的處理相關的信號,那么應用程序退出的方式幾乎總是被強制殺死而不是優雅的退出。究竟誰是 1 號進程則主要由 entrypoint, cmd, run 等指令的寫法決定,所以這些指令的使用是很有講究的。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

原文鏈接:http://www.cnblogs.com/sparkdev/p/7598590.html?utm_source=tuicool&utm_medium=referral

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 亚洲男人网 | 精品无码国产污污污免费网站2 | 放荡警察巨r麻麻出轨小说 范冰冰特黄xx大片 饭冈加奈子在线播放观看 法国老妇性xx在线播放 | tk白嫩玉足脚心vk | 久久伊人影院 | boobsmilking流奶水野战 | 日本高清动作片www欧美 | 99久9在线视频 | 男人最爱看的网站 | 日韩大片免费观看 | 午夜影院免费看 | 男女啪啪gif | 午夜精品久久久久久久99蜜桃 | 9色视频在线观看 | 1024免费福利永久观看网站 | 美女叽叽 | 国产成人精品男人的天堂538 | 呜嗯啊野战h呻吟男男双性 污小说在线阅读 | 亚洲成综合 | 亚洲精品成人A8198A片漫画 | 精彩国产萝视频在线 | 国产成人精品一区二区不卡 | 日本高清视频网址 | free性丰满hd性欧美厨房 | 日本又大又硬又粗的视频 | 日本人与黑人做爰视频网站 | 高清视频大片免费观看 | 亚洲第一网站免费视频 | 国产综合成人亚洲区 | 国产 国语对白 露脸正在播放 | 欧美成人aletta ocean | 大jjjj免费看视频 | 蜜柚精彩在线观看 | 思思玖玖玖在线精品视频 | 牛牛影院成人免费网页 | 亚洲国产天堂久久精品网 | 国产a在线 | 日本玖玖视频 | japanesemoms乱熟| 幻女free性zoz0交 | 欧美精品一区二区三区免费播放 |