小知识:详解如何在 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猿站网-插图

应用程序不是容器中的 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 容器中捕获信号-2猿站网-插图

现在给 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 容器中捕获信号-3猿站网-插图

结论

容器中的 1 号进程是非常重要的,如果它不能正确的处理相关的信号,那么应用程序退出的方式几乎总是被强制杀死而不是优雅的退出。究竟谁是 1 号进程则主要由 entrypoint, cmd, run 等指令的写法决定,所以这些指令的使用是很有讲究的。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

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

声明: 猿站网有关资源均来自网络搜集与网友提供,任何涉及商业盈利目的的均不得使用,否则产生的一切后果将由您自己承担! 本平台资源仅供个人学习交流、测试使用 所有内容请在下载后24小时内删除,制止非法恶意传播,不对任何下载或转载者造成的危害负任何法律责任!也请大家支持、购置正版! 。本站一律禁止以任何方式发布或转载任何违法的相关信息访客发现请向站长举报,会员发帖仅代表会员个人观点,并不代表本站赞同其观点和对其真实性负责。本网站的资源部分来源于网络,如有侵权烦请发送邮件至:2697268773@qq.com进行处理。
建站知识

小知识:详解linux下mnt目录作用

2023-4-12 23:56:09

建站知识

小知识:如何用nginx配置wordpress的方法示例

2023-4-13 0:55:27

0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧
个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索