Dockerfile的ENTRYPOINT和CMD

Dockerfile的ENTRYPOINT和CMD都是用来定义命令的执行入口。

  1. 定义ENTRYPOINT
$ cat Dockerfile 
FROM oraclelinux
ADD ./docker-entry.sh   /docker-entry.sh
ENTRYPOINT /docker-entry.sh

$ cat docker-entry.sh
#!/usr/bin/env bash
echo "Entry of ENTRYPOINT, ARGS[#]=$#"
for ((i = 0; i <= $#; i++ )); do
  echo "ENTRYPOINT ARGS[${i}]=[${!i}]"
done

$ docker build -t testimage

$ docker run --rm testimage
Entry of ENTRYPOINT, ARGS[#]=0

在docker run的时候ENTRYPOINT指定的命令被调用了。

  1. 定义CMD
$ cat Dockerfile 
FROM oraclelinux
ADD ./docker-cmd.sh   /docker-cmd.sh
CMD /docker-cmd.sh

$ cat docker-cmd.sh
#!/usr/bin/env bash
echo "Entry of CMD, ARGS[#]=$#"
for ((i = 0; i <= $#; i++ )); do
  echo "CMD ARGS[${i}]=[${!i}]"
done

$ docker build -t testimage

$ docker run --rm testimage
Entry of CMD, ARGS[#]=0

这里看起来,ENTRYPOINT和CMD的行为是一样的,没有啥区分。

  1. 同时定义ENTRYPOINT和CMD
$ cat Dockerfile 
FROM oraclelinux
ADD ./docker-cmd.sh   /docker-cmd.sh
ADD ./docker-entry.sh   /docker-entry.sh
CMD        /docker-cmd.sh
ENTRYPOINT /docker-entry.sh

$ docker build -t testimage

$ docker run --rm testimage
Entry of ENTRYPOINT, ARGS[#]=0
  • 同时定义了ENTRYPOINT和CMD时,CMD被忽略。
  1. ENTRYPOINT和CMD的两种写法

4.1 SHELL
格式:<ENTRYPOINT|CMD> executable param1 param2 ...
4.2 EXEC
格式:<ENTRYPOINT|CMD> ["executable", "param1", "param2", ...]

  1. 使用EXEC的方式重新执行上面三个case

5.1 定义ENTRYPOINT

$ docker run --rm testimage
Entry of ENTRYPOINT, ARGS[#]=0

5.2 定义CMD

$ docker run --rm testimage
Entry of CMD, ARGS[#]=0

5.3 同时定义ENTRYPOINT和CMD

Entry of ENTRYPOINT, ARGS[#]=1
ENTRYPOINT ARGS[0]=[/docker-entry.sh]
ENTRYPOINT ARGS[1]=[/docker-cmd.sh]

看到没有5.3这里和前面使用SHELL格式的有重大不一样,CMD的内容被作为参数传递给了ENTRYPOINT,虽然目前ENTRYPOINT没有用到,但是我们知道就可以用了。

  1. 使用run自带命令

6.1 ENTRYPOINT + SHELL格式

$ docker run --rm testimage date
Entry of ENTRYPOINT, ARGS[#]=0
ENTRYPOINT ARGS[0]=[/docker-entry.sh]

6.2 CMD + SHELL格式

$ docker run --rm testimage date
Sun Dec 30 04:44:33 UTC 2018

6.3 ENTRYPOINT + CMD + SHELL格式

$ docker run --rm testimage date
Entry of ENTRYPOINT, ARGS[#]=0
ENTRYPOINT ARGS[0]=[/docker-entry.sh]

6.4 ENTRYPOINT + EXEC格式

$ docker run --rm testimage date
Entry of ENTRYPOINT, ARGS[#]=1
ENTRYPOINT ARGS[0]=[/docker-entry.sh]
ENTRYPOINT ARGS[1]=[date]

6.5 CMD + EXEC格式

$ docker run --rm testimage date
Sun Dec 30 04:47:12 UTC 2018

6.6 ENTRYPOINT + CMD + EXEC格式

$ docker run --rm testimage date
Entry of ENTRYPOINT, ARGS[#]=1
ENTRYPOINT ARGS[0]=[/docker-entry.sh]
ENTRYPOINT ARGS[1]=[date]
  1. 总结

7.1. ENTRYPOINT和CMD都可以用来指定容器的入口命令。
7.2 ENTRYPOINT具有优先权

  • 如果定义了ENTRYPOINT,那么执行ENTRYPOINT,忽略CMD
  • 如果没有定义ENTRYPOINT,那么执行CMD;如果CMD也没有则失败。

7.3 在EXEC模式下,CMD会被作为ARG[1]传递给ENTRYPOINT

  • 这样ENTRYPOINT可以决定是否调用CMD的内容。
  • 在SHELL模式下则不会。
  • 如果没有ENTRYPOINT,如前所述,则CMD直接被执行了。

7.3 docker run的时候如果后面指定命令,那么这个命令是CMD的替换。

  • 规则遵从CMD的处理流程。

7.4 因此建议ENTRYPOINT和CMD均采用EXEC的调用方式

  • 这样CMD的内容可以被ENTRYPOINT调用,而且
  • 原始CMD的内容可以在命令行被替换。

完整例子:

$ cat Dockerfile
FROM oraclelinux

ADD ./docker-cmd.sh     /docker-cmd.sh
ADD ./docker-entry.sh   /docker-entry.sh

ENTRYPOINT  [ "/docker-entry.sh" ]
CMD         [ "/docker-cmd.sh" ]

$ docker build . -t testimage

$ docker run --rm testimage
Entry of ENTRYPOINT, ARGS[#]=1
ENTRYPOINT ARGS[0]=[/docker-entry.sh]
ENTRYPOINT ARGS[1]=[/docker-cmd.sh]
$@=[/docker-cmd.sh]
Entry of CMD, ARGS[#]=0
CMD ARGS[0]=[/docker-cmd.sh]

$ docker run --rm testimage date
Entry of ENTRYPOINT, ARGS[#]=1
ENTRYPOINT ARGS[0]=[/docker-entry.sh]
ENTRYPOINT ARGS[1]=[date]
$@=[date]
Sun Dec 30 05:03:34 UTC 2018

$ docker run --rm testimage date "+%s"
Entry of ENTRYPOINT, ARGS[#]=2
ENTRYPOINT ARGS[0]=[/docker-entry.sh]
ENTRYPOINT ARGS[1]=[date]
ENTRYPOINT ARGS[2]=[+%s]
$@=[date +%s]
1546146222
  1. EXEC模式的缺陷

8.1 EXEC不能使用环境变量

例如在SHELL模式下:

$ cat Dockerfile
FROM oraclelinux

ADD ./docker-entry.sh   /docker-entry.sh

ENV VAR Hello

ENTRYPOINT  "/docker-entry.sh" "${VAR}"

$ docker run --rm testimage
Entry of ENTRYPOINT, ARGS[#]=1
ENTRYPOINT ARGS[0]=[/docker-entry.sh]
ENTRYPOINT ARGS[1]=[Hello]
$@=[Hello]

而同样的功能如果在EXEC模式下:

$ cat Dockerfile
FROM oraclelinux

ADD ./docker-entry.sh   /docker-entry.sh

ENV VAR Hello

ENTRYPOINT  [ "/docker-entry.sh", "${VAR}" ]

$ docker run --rm testimage
Entry of ENTRYPOINT, ARGS[#]=1
ENTRYPOINT ARGS[0]=[/docker-entry.sh]
ENTRYPOINT ARGS[1]=[${VAR}]
$@=[${VAR}]

这个环境变量$VAR没有被替换掉,而是源文本的方式穿下去了。

解决这个问题的办法使用"bash -c"来调用ENTRYPOINT指令:

$ cat Dockerfile
FROM oraclelinux

ADD ./docker-entry.sh   /docker-entry.sh

ENV VAR Hello

ENTRYPOINT  [ "/bin/bash", "-c", "/docker-entry.sh ${VAR}" ]

$ docker run --rm testimage
Entry of ENTRYPOINT, ARGS[#]=1
ENTRYPOINT ARGS[0]=[/docker-entry.sh]
ENTRYPOINT ARGS[1]=[Hello]
$@=[Hello]
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Dockerfile 有两个启动配置, CMD 和 ENTRYPOINT , 可以在 Dockerfile 中来配...
    JaeGwen阅读 14,443评论 1 1
  • 透过别人的故事,看自己。首先,这个别人得是自己所欣赏的人,不然就没有写下去的意义了。 今天白杨师姐在“百人百天陪伴...
    颜妍陪你坚持阅读 3,538评论 2 2
  • 一首歌的时间,做一个“逃兵”。走过的路、读过的书、身体所经历过的疼痛,他们全都算数。细腻的心、乱飞的思绪,没心没肺...
    醬醬子阅读 1,612评论 0 0
  • 辛弃疾怀诗。 岳鹏举志词。 看宋朝、又上愁眉。 可叹未江山一统, 偏安地,国分离。 古史尽闻悲。 今人当谨思。 洞...
    西山有晴雪阅读 5,712评论 31 65
  • 让品清告诉大家为什么️这么爱三草这个品牌,并3年如一日坚持着我的热爱,坚持着我的分享❤️ 三草两木发展史️ 蟹老板...
    品微阅读 1,397评论 0 0