Docker可以从Dockerfile
中一步一步的读取指令来自动的创建镜像,常使用Dockerfile来创建用户自定义的镜像。格式如下:
|
|
虽然前面的指令大小写不敏感,但习惯性的还是建议大写。docker是严格按照顺序(#
注释起来的忽略)运行指令的。
下面逐个来介绍几个必要的指令。
FROM
|
|
在Dockerfile中第一条非注释INSTRUCTION一定是FROM
,它决定了以哪一个镜像作为基准,<image>
首选本地是否存在,如果不存在则会从公共仓库下载(当然也可以使用私有仓库的格式)。
RUN
|
|
RUN
指令会在当前镜像的顶层执行任何命令,并commit成新的(中间)镜像,提交的镜像会在后面继续用到。
上面看到RUN
后的格式有两种写法。
shell格式,相当于执行/bin/sh -c "<command>"
:
|
|
exec格式,不会触发shell,所以$HOME
这样的环境变量无法使用,但它可以在没有bash
的镜像中执行,而且可以避免错误的解析命令字符串:
|
|
ENTRYPOINT
ENTRYPOINT
命令设置在容器启动时执行命令,如果有多个ENTRYPOINT
指令,那只有最后一个生效。有以下两种命令格式:
|
|
比如:
|
|
使用exec格式,在docker run <image>
的所有参数,都会追加到ENTRYPOINT
之后,并且会覆盖CMD
所指定的参数(如果有的话)。当然可以在run
时使用--entrypoint
来覆盖ENTRYPOINT
指令。
使用shell格式,ENTRYPOINT
相当于执行/bin/sh -c <command..>
,这种格式会忽略docker run
和CMD
的所有参数。
以推荐使用的exec格式为例:
我们可以使用ENTRYPOINT
来设置基本不会变化的命令,用CMD
来设置其它的可能改变的默认启动命令或选项(docker run
会覆盖的)。
|
|
docker build -t registry.tp-link.net:8000/ubuntu:dockerfile_test .
运行
|
|
如果在使用的docker版本在v1.3及以上,则可以使用docker exec
继续在容器中验证,看到完整的top命令docker exec -it test ps aux
CMD
|
|
一个Dockerfile里只能有一个CMD,如果有多个,只有最后一个生效。CMD
指令的主要功能是在build完成后,为了给docker run
启动到容器时提供默认命令或参数,这些默认值可以包含可执行的命令,也可以只是参数(此时可执行命令就必须提前在ENTRYPOINT
中指定)。
它与ENTRYPOINT
的功能极为相似,区别在于如果docker run
后面出现与CMD
指定的相同命令,那么CMD会被覆盖;而ENTRYPOINT
会把容器名后面的所有内容都当成参数传递给其指定的命令(不会对命令覆盖)。另外CMD
还可以单独作为ENTRYPOINT
的所接命令的可选参数。
CMD
与RUN
的区别在于,RUN
是在build
成镜像时就运行的,先于CMD
和ENTRYPOINT
的,CMD
会在每次启动容器的时候运行,而RUN
只在创建镜像时执行一次,固化在image中。
举例1:
|
|
因为echo run_arg
覆盖了CMD
。如果run
后没有echo run_arg
,则输出CMD_args
。
举例2:
|
|
因为echo run_arg
追加到ENTRYPOIINT
的echo
后面了。如果在ENTRYPOINT
后再加入一行CMD ["CMD_args"]
,则结果依旧,除非去掉run后的所有参数。
当出现ENTRYPOINT
指令时CMD
指令只可能(当ENTRYPOINT
指令使用exec方式执行时)被当做ENTRYPOINT
指令的参数使用,其他情况则会被忽略。
EXPOSE
EXPOSE
指令告诉容器在运行时要监听的端口,但是这个端口是用于多个容器之间通信用的(links),外面的host是访问不到的。要把端口暴露给外面的主机,在启动容器时使用-p
选项。
示例:
|
|
ADD
|
|
将文件<src>
拷贝到container的文件系统对应的路径<dest>
下。
<src>
可以是文件、文件夹、URL,对于文件和文件夹<src>
必须是在Dockerfile的相对路径下(build context path),即只能是相对路径且不能包含../path/
。
<dest>
只能是容器中的绝对路径。如果路径不存在则会自动级联创建,根据你的需要是<dest>
里是否需要反斜杠/
,习惯使用/
结尾从而避免被当成文件。
示例:
|
|
另外ADD
支持远程URL获取文件,但官方认为是strongly discouraged
,建议使用wget
或curl
代替。
ADD
还支持自动解压tar文件,比如ADD trusty-core-amd64.tar.gz /
会线自动解压内容再COPY到在容器的/
目录下。
ADD只有在build镜像的时候运行一次,后面运行container的时候不会再重新加载,也就是你不能在运行时通过这种方式向容器中传送文件,-v
选项映射本地到容器的目录。
COPY
Same as ‘ADD’ but without the tar and remote url handling.
COPY
的语法与功能与ADD
相同,只是不支持上面讲到的<src>
是远程URL、自动解压这两个特性,但是Best Practices for Writing Dockerfiles建议尽量使用COPY
,并使用RUN
与COPY
的组合来代替ADD
,这是因为虽然COPY
只支持本地文件拷贝到container,但它的处理比ADD
更加透明,建议只在复制tar文件时使用ADD
,如ADD trusty-core-amd64.tar.gz /
。
ENV
用于设置环境变量:
|
|
设置了后,后续的RUN命令都可以使用,当运行生成的镜像时这些环境变量依然有效,如果需要在运行时更改这些环境变量可以在运行docker run
时添加-env <key>=<value>
参数来修改。
VOLUME
VOLUME指令用来在容器中设置一个挂载点,可以用来让其他容器挂载以实现数据共享或对容器数据的备份、恢复或迁移。请参考文章docker容器间通信
WORKDIR
WORKDIR指
令用于设置Dockerfile
中的RUN
、CMD
和ENTRYPOINT
指令执行命令的工作目录(默认为/
目录),该指令在Dockerfile
文件中可以出现多次,如果使用相对路径则为相对于WORKDIR
上一次的值,例如WORKDIR /a
,WORKDIR b
,RUN pwd
最终输出的当前目录是/a/b
。(RUN cd /a/b
,RUN pwd
是得不到/a/b
的)
ONBUILD
ONBUILD
指令用来设置一些触发的指令,用于在当该镜像被作为基础镜像来创建其他镜像时(也就是Dockerfile
中的FROM
为当前镜像时)执行一些操作,ONBUILD中
定义的指令会在用于生成其他镜像的Dockerfile
文件的FROM
指令之后被执行,上述介绍的任何一个指令都可以用于ONBUILD
指令,可以用来执行一些因为环境而变化的操作,使镜像更加通用。
注意:
- ONBUILD中定义的指令在当前镜像的build中不会被执行。
- 可以通过查看
docker inspect <image>
命令执行结果的OnBuild键来查看某个镜像ONBUILD指令定义的内容。 - ONBUILD中定义的指令会当做引用该镜像的Dockerfile文件的FROM指令的一部分来执行,执行顺序会按ONBUILD定义的先后顺序执行,如果ONBUILD中定义的任何一个指令运行失败,则会使FROM指令中断并导致整个build失败,当所有的ONBUILD中定义的指令成功完成后,会按正常顺序继续执行build。
- ONBUILD中定义的指令不会继承到当前引用的镜像中,也就是当引用ONBUILD的镜像创建完成后将会清除所有引用的ONBUILD指令。
- ONBUILD指令不允许嵌套,例如
ONBUILD ONBUILD ADD . /data
是不允许的。 - ONBUILD指令不会执行其定义的FROM或MAINTAINER指令。
例如,Dockerfile
使用如下的内容创建了镜像 image-A :
|
|
如果基于 image-A 创建新的镜像时,新的Dockerfile
中使用FROM image-A
指定基础镜像时,会自动执行ONBUILD
指令内容,等价于在后面添加了两条指令。
|
|
USER
为运行镜像时或者任何接下来的RUN
指令指定运行用户名或UID:
USER daemon
MAINTAINER
使用MAINTAINER指令来为生成的镜像署名作者
MAINTAINER author's name mailaddress
The .dockerignore
file
.dockerignore
用来忽略上下文目录中包含的一些image用不到的文件,它们不会传送到docker daemon。规则使用go语言的匹配语法。如:
|
|
更多内容参考Dockerfile最佳实践系列。官方有个Dockerfile tutorial练习Dockerfile的写法,非常简单但对于养成良好的格式、注释有一些帮助。
Dockerfile示例
下面的Dockerfile
是MySQL官方镜像的构建过程。从ubuntu基础镜像开始构建,安装mysql-server、配置权限、映射目录和端口,CMD
在从这个镜像运行到容器时启动mysql。其中VOLUME
定义的两个可挂载点,用于在host中挂载,因为数据库保存在主机上而非容器中才是比较安全的。
|
|
使用:
|
|
(提示,上述第一条命令,如果你的host不可以连接Docker Hub,那么需要在启动docker服务时使用HTTP_PROXY=
——用于build的时更新下载软件,同时执行docker build
的终端设置http_proxy
和https_proxy
用于下载Dockerfile)
运行:
|
|