在 docker 容器中捕获信号

咱们可能都使用过 docker stop 命令来中止正在运行的容器,有时可能会使用 docker kill 命令强行关闭容器或者把某个信号传递给容器中的进程。这些操做的本质都是经过从主机向容器发送信号实现主机与容器中程序的交互。好比咱们能够向容器中的应用发送一个从新加载信号,容器中的应用程序在接到信号后执行相应的处理程序完成从新加载配置文件的任务。本文将介绍在 docker 容器中捕获信号的基本知识。html

信号(linux)

信号是一种进程间通讯的形式。一个信号就是内核发送给进程的一个消息,告诉进程发生了某种事件。当一个信号被发送给一个进程后,进程会当即中断当前的执行流并开始执行信号的处理程序。若是没有为这个信号指定处理程序,就执行默认的处理程序。
进程须要为本身感兴趣的信号注册处理程序,好比为了能让程序优雅的退出(接到退出的请求后可以对资源进行清理)通常程序都会处理 SIGTERM 信号。与 SIGTERM 信号不一样,SIGKILL 信号会粗暴的结束一个进程。所以咱们的应用应该实现这样的目录:捕获并处理 SIGTERM 信号,从而优雅的退出程序。若是咱们失败了,用户就只能经过 SIGKILL 信号这一终极手段了。除了 SIGTERM 和 SIGKILL ,还有像 SIGUSR1 这样的专门支持用户自定义行为的信号。下面的代码简单的说明在 nodejs 中如何为一个信号注册处理程序:node

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

关于信号的更多信息,笔者在《linux kill 命令》一文中有所说起,这里再也不赘述。linux

容器中的信号

Docker 的 stop 和 kill 命令都是用来向容器发送信号的。注意,只有容器中的 1 号进程可以收到信号,这一点很是关键!
stop 命令会首先发送 SIGTERM 信号,并等待应用优雅的结束。若是发现应用没有结束(用户能够指定等待的时间),就再发送一个 SIGKILL 信号强行结束程序。
kill 命令默认发送的是 SIGKILL 信号,固然你能够经过 -s 选项指定任何信号。docker

下面咱们经过一个 nodejs 应用演示信号在容器中的工做过程。建立 app.js 文件,内容以下:json

'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 信号注册了处理程序。接下来咱们将介绍以不一样的方式在容器中运行程序时信号的处理状况。bash

应用程序做为容器中的 1 号进程

建立 Dockerfile 文件,把上面的应用打包到镜像中:服务器

FROM iojs:onbuild
COPY ./app.js ./app.js
COPY ./package.json ./package.json
EXPOSE 3000
ENTRYPOINT ["node", "app"]

请注意 ENTRYPOINT 指令的写法,这种写法会让 node 在容器中以 1 号进程的身份运行。app

接下来建立镜像:ui

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

而后启动容器运行应用程序:spa

$ 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 信号并优雅的退出了:

结论

容器中的 1 号进程是很是重要的,若是它不能正确的处理相关的信号,那么应用程序退出的方式几乎老是被强制杀死而不是优雅的退出。究竟谁是 1 号进程则主要由 EntryPoint, CMD, RUN 等指令的写法决定,因此这些指令的使用是颇有讲究的。