docker stop不适用于节点进程

时间:2015-01-15 18:05:01

标签: node.js ubuntu docker sigterm

我希望能够在docker容器中运行node,然后能够运行docker stop <container>。这应该会使SIGTERM上的容器停止,而不是超时并执行SIGKILL。不幸的是,我似乎遗漏了一些东西,我发现的信息似乎与其他部分相矛盾。

这是一个测试Dockerfile:

FROM ubuntu:14.04
RUN apt-get update && apt-get install -y curl
RUN curl -sSL http://nodejs.org/dist/v0.11.14/node-v0.11.14-linux-x64.tar.gz | tar -xzf -
ADD test.js /
ENTRYPOINT ["/node-v0.11.14-linux-x64/bin/node", "/test.js"]

这是Dockerfile中引用的test.js

var http = require('http');

var server = http.createServer(function (req, res) {
  console.log('exiting');
  process.exit(0);
}).listen(3333, function (err) {
  console.log('pid is ' + process.pid)
});

我像这样构建它:

$ docker build -t test .

我这样运行:

$ docker run --name test -p 3333:3333 -d test

然后我跑:

$ docker stop test

因此SIGTERM显然不起作用,导致它在10秒后超时然后死亡。

我发现如果我通过sh -c启动节点任务,那么我可以使用^C从交互式(-it)容器中删除它,但我仍然无法获取docker stop工作。这与我读过的评论相矛盾,说sh没有传递信号,但可能同意我读过的其他评论说PID 1没有得到SIGTERM(因为它已经开始了)通过sh,它将是PID 2)。

最终目标是能够在新手工作中运行docker start -a ...并能够停止服务并实际退出容器。

2 个答案:

答案 0 :(得分:2)

好的,我自己想出了一个解决方法,我会冒昧地回答,希望能帮助别人。它并没有完全回答为什么信号之前没有工作,但它确实给了我想要的行为。

使用baseimage-docker似乎解决了这个问题。以下是我使用上面的最小测试示例进行的操作:

按原样保留test.js

Dockerfile修改为如下所示:

FROM phusion/baseimage:0.9.15

# disable SSH
RUN rm -rf /etc/service/sshd /etc/my_init.d/00_regen_ssh_host_keys.sh

# install curl and node as before
RUN apt-get update && apt-get install -y curl
RUN curl -sSL http://nodejs.org/dist/v0.11.14/node-v0.11.14-linux-x64.tar.gz | tar -xzf -

# the baseimage init process
CMD ["/sbin/my_init"]

# create a directory for the runit script and add it
RUN mkdir /etc/service/app
ADD run.sh /etc/service/app/run

# install the application
ADD test.js /

baseimage-docker包含一个init进程(/sbin/my_init),用于处理启动其他进程和处理zombie processes。它使用runit进行服务监督。因此,Dockerfile将my_init进程设置为在引导时运行的命令,并为runit添加脚本/etc/service以进行提取。

run.sh脚本很简单:

#!/bin/sh
exec /node-v0.11.14-linux-x64/bin/node /test.js

不要忘记chmod +x run.sh

默认情况下,如果服务出现故障,runit将自动重启服务。

按照这些步骤(以及之前的构建,运行和停止),容器会及时正确地响应其关闭请求。

答案 1 :(得分:0)

我不知道这是否为时已晚。但是我这样做的方法是捕获SIGINT(JavaScript中的中断信号)。

var process = require('process')
process.on('SIGINT', () => {
  console.info("Interrupted")
  process.exit(0)
})

按Ctrl + C时应该可以解决问题