正确使用Dockerfile中的ENTRYPOINT命令

正确使用Dockerfile中的ENTRYPOINT命令

目录git

做者:杨冬 欢迎转载,也请保留这段声明。谢谢!
出处:https://andyyoung01.github.io/ 或 http://andyyoung01.16mb.com/github

如何正确地构建Docker镜像,对于正确使用Docker是很是关键的。若是你想要定义容器须要运行的命令,而将命令行参数留给用户提供,则使用Dockerfile中的ENTRYPOINT命令是十分方便的。docker

 

做为演示,咱们假设一个简单的场景:公司的服务器须要按期清理旧的日志文件。这虽然是一个简单的管理任务,可是很是容易出错,管理员可能会不当心删除了错误的文件。因此可使用一个Docker镜像来包装管理员运行的命令,下降这种问题出现的风险。
下面这段脚本删除/log_dir目录中久于某天的日志,天数做为命令行参数传递进来:shell

clean_logubuntu

 

1数组

2bash

3服务器

 

#!/bin/bashapp

echo "Cleaning logs over $1 days old"ui

find /log_dir -ctime "$1" -name '*log' -exec rm {} \;

 

下面在与上面脚本相同的目录下建立Dockerfile,以上面的脚本做为entrypoint

 

1

2

3

4

5

 

FROM ubuntu:14.04

ADD clean_log /usr/bin/clean_log

RUN chmod +x /usr/bin/clean_log

ENTRYPOINT ["/usr/bin/clean_log"]

CMD ["7"]

 

上面代码第2行将前面的脚本添加进镜像;第4行定义了镜像的默认执行的脚本命令;第5行定义了默认脚本命令的参数(7天)。

ENTRYPOINTCMD的最佳实践——老是使用数组形式的写法:若是你常常在Docker Hub上查看别人的Dockerfile,会发现数组模式(例如 CMD [“/usr/bin/command”])会比shell模式用得更多(CMD /usr/bin/command)。这是由于shell模式会自动在你提供的命令前面添加一个/bin/bash -c命令,这可能会致使意外的结果。不过有时shell模式更加有用。

使用以下命令构建镜像:

$ docker build -t log-cleaner .

ENTRYPOINTCMD的区别常常令人迷惑。理解的关键点是知道当一个镜像启动时,entrypoint老是被执行,即便在docker run命令后指定了镜像要运行的命令。若是是这样的话,这个命令会被认为是entrypoint的参数,替换掉CMD中的默认参数。
例如上面构建的镜像,若是这样运行docker run -it log-cleaner /bin/bash的话,并不会执行bash,而是将/bin/bash做为参数传递给脚本(这里替换掉了默认的7这个参数)。系统会提示错误的参数:

 

1

2

3

 

$ docker run -it log-cleaner /bin/bash

Cleaning logs over /bin/bash days old

find: invalid argument '-name' to '-ctime'

 

正确的使用方法以下:

$ docker run -v /var/log/myapplogs:/log_dir log-cleaner 365

上述命令将/var/log/myapplogs目录挂载到容器内部的脚本指定的目录,而且以365做为参数传递给脚本,使365天之前的日志文件被删除。

相关文章
相关标签/搜索