目录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天)。
ENTRYPOINT
和CMD
的最佳实践——老是使用数组形式的写法:若是你常常在Docker Hub上查看别人的Dockerfile,会发现数组模式(例如 CMD [“/usr/bin/command”])会比shell模式用得更多(CMD /usr/bin/command)。这是由于shell模式会自动在你提供的命令前面添加一个/bin/bash -c命令,这可能会致使意外的结果。不过有时shell模式更加有用。
使用以下命令构建镜像:
$ docker build -t log-cleaner .
ENTRYPOINT
和CMD
的区别常常令人迷惑。理解的关键点是知道当一个镜像启动时,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天之前的日志文件被删除。