Dockerfile 最佳实践


Docker 官方关于 Dockerfile 最佳实践原文链接地址 https://docs.docker.com/develop/develop-images/dockerfile_best-practices/

Docker 可以通过从 Dockerfile 包含所有命令的文本文件中读取指令自动构建镜像,以便构建给定镜像

Dockerfiles 使用特定的格式并使用一组特定的指令,您可以在 Dockerfile Reference 页面上了解基础知识,如果你是新手写作 Dockerfile,你应该从那里开始

本文档介绍了由 Docker,Inc. 和 Docker 社区推荐的用于构建高效镜像的最佳实践和方法,要查看许多实践和建议,请查看 Dockerfile for buildpack-deps

一般准则和建议

容器应该是短暂的

通过 Dockerfile 构建的镜像所启动的容器应该尽可能短暂(生命周期短),「短暂」意味着可以停止和销毁容器,并且创建一个新容器并部署好所需的设置和配置工作量应该是极小的,我们可以查看下 12 Factor(12要素)应⽤程序⽅法 的进程部分,可以让我们理解这种无状态方式运行容器的动机

建立上下文

当你发出一个 docker build 命令时,当前的工作目录被称为构建上下文,默认情况下,Dockerfile 就位于该路径下,当然您也可以使用 -f 参数来指定不同位置,无论 Dockerfile 在什么地方,当前目录中的所有文件内容都将作为构建上下文发送到 Docker 守护进程中去

下面是一个构建上下文的示例,为构建上下文创建一个目录并 cd 放入其中,将 "hello" 写入一个文本文件 hello,然后创建一个 Dockerfile 并运行 cat,从构建上下文 (.) 中构建图像

$ mkdir myproject && cd myproject
$ echo "hello" > hello
$ echo -e "FROM busybox\nCOPY /hello /\nRUN cat /hello" > Dockerfile
$ docker build -t helloapp:v1

现在移动 Dockerfile 和 hello 到不同的目录,并建立了乳香的第二个版本(不依赖于缓存中的最后一个版本),使用 -f 指向 Dockerfile 并指定构建上下文的目录

$ mkdir -p dockerfiles context
$ mv Dockerfile dockerfiles && mv hello context
$ docker build --no-cache -t helloapp:v2 -f dockerfiles/Dockerfile context

在构建的时候包含不需要的文件会导致更大的构建上下文和更大的镜像大小,这会增加构建时间,拉取和推送镜像的时间以及容器的运行时间大小,要查看你的构建环境有多大,请在构建系统时查找这样的消息

Dockerfile:
Sending build context to Docker daemon 187.8MB

使用 .dockerignore 文件

使用 Dockerfile 构建镜像时最好是将 Dockerfile 放置在一个新建的空目录下,然后将构建镜像所需要的文件添加到该目录中,为了提高构建镜像的效率,你可以在目录下新建一个 .dockerignore 文件来指定要忽略的文件和目录,.dockerignore 文件的排除模式语法和 Git 的 .gitignore 文件相似

使用多阶段构建

在 Docker 17.05 以上版本中,你可以使用多阶段构建来减少所构建镜像的大小

避免安装不必要的包

为了降低复杂性、减少依赖、减小文件大小和构建时间,应避免安装额外的或者不必要的软件包,例如,不要在数据库镜像中包含一个文本编辑器

一个容器只专注做一件事情

应该保证在一个容器中只运行一个进程,将多个应用解耦至不同容器中,保证了容器的横向扩展和复用,例如一个 web 应用程序可能包含三个独立的容器:web 应用、数据库、缓存,每个容器都是独立的镜像,分开运行,但这并不是说一个容器就只能跑一个进程,因为有的程序可能会自行产生其他进程,例如 Celery 就可以有很多个工作进程,虽然“每个容器跑一个进程”是一条很好的法则,但这并不是一条硬性的规定,我们主要是希望一个容器至关注一件事情,尽量保持干净和模块化

如果容器互相依赖,可使用 Docker 容器网络 来把这些容器连接起来

最小化镜像层数

在 Docker 17.05 甚至更早(1.10前),尽量减少镜像层数是非常重要的,不过现在的版本已经有了一定改善

  • 在 1.10 后,只有 RUNCOPYADD 指令会创建层,其他指令会创建临时的中间镜像,但是不会直接增加构建的镜像大小

当然减少 RUNCOPYADD 的指令仍然是很有必要的,但是我们也需要在 Dockerfile 可读性(也包括长期的可维护性)和减少层数之间做一个平衡

对多行参数排行

只要有可能,就将多行参数按字母顺序排序(比如要安装多个包时),这可以帮用户避免重复包含同一个包,更新包列表时也更容易,也更易阅读和审查,建议在反斜杠符号 \ 之前添加一个空格,可增加可读性

RUN apt-get update && apt-get install -y \
  bzr \
  cvs \
  git \
  mercurial \
  subversion

构建缓存

在镜像的构建过程中,Docker 根据 Dockerfile 指定的顺序执行每个指令,在执行每条指令前,Docker 都会在缓存中查找是否已经存在可重用镜像,如果有就使用现存的镜像,不再重复创建,当然如果你不想在构建过程中使用缓存,可以在 docker build 命令中使用 --no-cache=ture 选项

若想在构建的过程中使用缓存,那么了解什么时候可以什么时候无法找到匹配的镜像就很重要了,Docker 中缓存遵循的基本规则如下

  • 从一个基础镜像开始(FROM 命令指定),下一条指令将和该基础镜像的所有镜像进行匹配,检查这些子镜像被创建时使用的指令是否和被检查的指令完全一致,如果不一致,则缓存失败
  • 在大多数情况下,只需要简单地对比 Dockerfile 中的指令和子镜像,然而,有些指令需要更多的检查和解释
  • 对于 ADDCOPY 指令,镜像中对应文件的内容也会被检查,每个文件都会计算出一个检验值,这些文件的修改时间和最后访问时间不会被纳入校验的范围,在缓存的查找过程中,会将这些校验和已存在镜像中的文件校验值进行对比,如果文件有任何改变,比如内容和元数据,则缓存失败
  • 一旦缓存失败,所有后续的 Dockerfile 指令都将产生新的镜像,缓存不会被使用

Dockerfile 指令

下面是一下常用的 Dockerfile 指令,根据上面的建议和下面这些指令的合理使用,可以帮助我们编写高效且易维护的 Dockerfile 文件

  • FROM
    尽可能使用当前官方仓库作为构建镜像的基础,推荐使用 Alpine 镜像,因为它被严格控制并保持最小尺寸(目前小于 6M),但它仍是完整发行版

  • LABEL
    可给镜像添加标签来帮助组织镜像、记录许可信息、辅助自动化构建等,每个标签一行,由 LABEL 开头加上一个或多个标签对
    下面的示例展示了各种不同的可能格式 # 开头行是注释内容

# Set one or more individual labels
LABEL com.example.version="0.0.1-beta"
LABEL vendor="ACME Incorporated"
LABEL com.example.release-date="2020-03-23"
LABEL com.example.version.is-production=""

一个镜像可以包含多个标签,在 1.10 前,建议将多有标签合并为一条 LABEL 指令,以防止创建额外的层,但是现在不再是必须的了,以上内容也可写成下面这样

# Set multiple labels at once, using line-continuation characters to break long lines
LABEL vendor=ACME\ Incorporated \
      com.example.is-production="" \
      com.example.version="0.0.1-beta" \
      com.example.release-date="2020-03-23"

关于标签可以接受的键值对,参考 Understanding object labels

  • RUN
    为了保持 Dockerfile 文件的可读性,以及可维护性,建议将长的或复杂的 RUN 指令用反斜杠 \ 分割成多行
    RUN 指令最常见的用法是安装包用的 apt-get,因为 RUN apt-get 指令会安装包,所以有几个问题需要注意
    1. 不要使用 RUN apt-get upgradedist-upgrade,如果基础镜像中的某个包过时了,你应该联系它的维护者,如果你确定某个特定的包,比如 foo,需要升级,使用 apt-get install -u foo 即可,该指令会自动升级 foo 包
    2. 永远将 RUN apt-get updateapt-get install 组合成一条 RUN 声明
RUN apt-get update && apt-get install -y \
      package-bar \
      package-baz \
      package-foo

apt-get update 放在一条单独的 RUN 声明中会导致缓存问题以及后续的 apt-get install 失败

FROM ubuntu:14.04
RUN apt-get update
RUN apt-get install -y curl

构建镜像后,所有的层都在 Docker 的缓存中,假设你后来又修改了其中的 apt-get install 添加了一个包

FROM ubuntu:14.04
RUN apt-get update
RUN apt-get install -y curl nginx

Docker 发现修改后的 RUN apt-get update 指令和之前的完全一致,所以,apt-get update 不会执行,而是使用此前的缓存镜像,因为 apt-get update 没有运行,后面的 apt-get install 可能安装的是过时的 curl 和 nginx 版本
使用 RUN apt-get update && apt-get install -y 可以确保你的 Dockerfiles 每次安装的都是包的最新版本,而且这个过程不需要进一步的编码或额外干预,这项技术叫做 cache busting(缓存破坏),你也可以显示指定一个包的版本号来达到 cache-busting,这就是所谓的固定版本

RUN apt-get update && apt-get install -y \
    package-bar \
    package-baz \
    package-foo=1.3.*

固定版本会迫使构建过程检索特定的版本,而不管缓存中有什么,这项技术也可以减少因所需包中未预料到的变化而导致的失败
下面是一个 RUN 指令的示例模板,展示了所有关于 apt-get 的建议

RUN apt-get update && apt-get install -y \
    aufs-rools \
    automake \
    build-essential \
    curl \
    dpkg-sig \
    libcap-dev \
    libsqlite3-dev \
    mercurial \
    reperpeo \
    ruby1.9.1 \
    ruby1.9.1-dev \
    s3cmd=1.1.* \
&& rm -rf /var/lib/apt/lists/*

其中 s3cmd 指令指定了一个版本号 1.1.*,如果之前的镜像使用的是更旧的版本,指定新的版本会导致 apt-get update 缓存失效并确保安装的是新版本,另外,清理掉 apt 缓存 var/lib/apt/lists 可以减小镜像大小,因为 RUN 指令的开头为 apt-get update,包缓存总是会在 apt-get install 之前刷新

  • CMD
    CMD 指令用于执行目标镜像中包含的软件和任何参数,CMD 几乎都是以 CMD ["executable", "param1", "param2"...] 的形式使用,因此,若创建镜像的目的是为了部署某个服务(比如 Apache),你可能会执行类似于 CMD ["apache2", "-DFOREGROUND"] 形式的命令
    多数情况下,CMD 都需要一个交互式的 shell(bash,Python,perl 等),例如 CMD ["perl", "-de0"],或者 CMD ["PHP", "-a"],使用这种形式意味着,当你执行类似 docker run -it python 时,你会进入一个准备好的 shell 中
    CMD 在极少的情况下才会以 CMD ["param", "param"] 的形式与 ENTRYPOINT 协同使用,除非你和你的镜像使用者都对 ENTRYPOINT 的工作方式十分熟悉

  • EXPOSE
    EXPOSE 指令用于指定容器将要监听的端口,因此,你应该为你的应用程序使用常见的端口
    例如,提供 Apache web 服务的镜像应该使用 EXPOSE 80,而提供 MongeDB 服务的镜像使用 EXPOSE 27017
    对于外部访问,用户可以在执行 docker run 时使用一个标志来指示如何将制定的端口映射到所选择的端口

  • ENV
    为了方便新程序运行,可使用 ENV 来为容器中安装的程序更新 PATH 环境变量,如使用 ENV PATH /usr/local/nginx/bin:$PATH 来确保 CMD ["nginx"] 能正确运行
    ENV 指令也可用于为你想要容器化的服务提供必要的环境变量,比如 Postgres 需要的 PGDATA,最后,ENV 也能用于设置常见的版本号

ENV PG_MAJOR 9.3
ENV PG_VERSION 9.3.4
RUN curl -SL http://example.com/postgres-$PG_VERSION.tar.xz | tar -xJC /usr/src/postgress 
&& ...ENV PATH /usr/local/postgres-$PG_MAJOR/bin:$PATH

类似于程序中的常量,这种方法可以让你只需改变 ENV 指令来自动的改变容器中的软件版本

  • ADD 和 COPY
    虽然 ADDCOPY 功能类似,但一般优先使用 COPY,因此它比 ADD 更透明,COPY 只支持简单将本地文件拷贝到容器中,而 ADD 有一些并不明显的功能(比如本地 tar 提取和远程 URL 支持),因此,ADD 的最佳用例是将本地 tar 文件自动提取到镜像中,如 ADD rootfs.tar.xz
    若 Dockerfile 有多个步骤需要使用上下文中不同的文件,单独 COPY 每个文件,而不是一次性的 COPY 所有文件,这将保证每个步骤的构建缓存只在特定的文件变化时失效
COPY requirements.txt /tmp/
RUN pip install --requirement /tmp/requirements.txt
COPY . /tmp/

如果将 COPY . /tmp/ 放置在 RUN 指令之前,只要 . 目录中任何一个文件变化,都会导致后续指令的缓存失效
为了让镜像尽量小,最好不要使用 ADD 指令从远程 URL 获取包,而是使用 curlwget,这样可在文件提取完后删掉不需要的文件来避免在镜像中额外添加一层,如尽量避免如下方式

ADD http://example.com/big.tar.xz /usr/src/things/
RUN tar -xJf /usr/src/things/big.tar.xz -C /usr/src/things
RUN make -C /usr/src/things all

而是应使用如下方式

RUN mkdir -p /usr/src/things \
    && curl -SL http://example.com/big.tar/xz \
    | tar -xJC /usr/src/things \
    && make -C /usr/src/things all

上面使用了管道操作,故没有中间文件需要删除,对于其他不需要 ADD 的自动提取功能的文件或目录,应该使用 COPY

  • ENTRYPOINT
    ENTRYPOINT 的最佳用处是设置镜像的主命令,允许将镜像当成命令本身来运行(用 CMD 提供默认选项)
    如下示例提供了命令行工具 s3cmd
ENTRYPOINT ["s3cmd"]
CMD ["--help"]

现在直接运行该镜像创建的容器汇显示命令帮助

$ docker run s3cmd

或提供正确的参数来执行某个命令

$ docker run s3cmd ls s3://mybucket

这样镜像名可当成命令行的参考,ENTRYPOINT 指令也可结合一个辅助脚本使用,和此前命令行风格类似,即使启动工具需要不止一个步骤
例如,Postgres 官方镜像使用如下脚本作为 ENTRYPOINT

#!/bin/bash
set -e
if [ "$1" = 'postgres' ];then
  chown -R postgres "$PGDATA"

  if [ -z "$(ls -A "$PGDATA")" ]; then
    gosu postgres initdb
  fi

  exec gosu postgres "$@"fi
exec "%@"

注意:该脚本使用了 Bash 内置命令 exec,所以最后运行的进程就是容器的 PID 为 1 的进程,这样,进程就可以接收到任何发送给容器的 Unix 信号了

该辅助脚本被拷贝到容器,并在容器启动时通过 ENTRYPOINT 执行

COPY ./docker-entrypoint.sh /
ENTRYPOINT ["/docker-entrypoint.sh"]

该脚本可让用户用几种不同的方式和 Postgres 交互,你可以很简单地启动 Postgres

$ docker run postgres

也可执行 Postgres 并传递参数

$ docker run postgres postgres --help

最后,你还可以启动另一个完全不同的工具,如 Bash

$ docker run --rm -it postgres bash
  • VOLUME
    VOLUME 指令用于暴露任何数据库存储文件,配置文件,或容器创建的文件目录,强烈建议使用 VOLUME 来管理镜像中的可变部分和用户可以改变的部分

  • USER
    若某个服务不需要特权执行,建议使用 USER 指令切换到非 root 用户,先在 Dockerfile 中使用类似 RUN groupadd -r postgres && useradd -r -g postgres postgres 的指令创建用户和用户组

注意:在镜像中,用户和用户组每次被分配的 UID/GID 都是不确定的,下次重新构建镜像时被分配到的 UID/GID 可能会不一样,若要依赖确定的 UID/GID,应该显式的指定一个 UID/GID

应避免使用 sudo,因为它不可预期的 TTY 和信号转发行为可能造成的问题比它能解决的问题还多,若真的需要和 sudo 类似的功能(例如,以 root 权限初始化某个守护进程,以非 root 权限执行它),你可使用 gosu
最后,为了减少层数和复杂度,避免频繁使用 USER 来回切换用户

  • WORKDIR
    为了清晰性和可靠性,应该总是在 WORKDIR 中使用绝对路径,另外,应使用 WORKDIR 来替代类似于 RUN cd ... && do-something 指令,后者难以阅读、排错和维护

  • ONBUILD
    格式:ONBUILD<其他指令>,ONBUILD 是一个特殊的指令,它后面跟的是其他指令,比如 RUNCOPY 等,而这些指令,在当前镜像构建时并不会被执行,只有当前镜像为基础镜像,去构建下一级镜像的时候才会被执行,Dockerfile 中的其它指令都是为了定制当前镜像而准备的,唯有 ONBUILD 是为了帮助别人定制自己而准备的

假设我们要制作 Node.js 所写的应用镜像,我们都知道 Node.js 使用 npm 进行包管理,所有依赖、配置、启动信息等会放到 package.json 文件里,在拿到程序代码后,需要先进行 npm install 才可以获得所有需要的依赖,然后就可以通过 npm start 来启动应用,因此,一般来说会这样写 Dockerfile

FROM node:slim
RUN mkdir /app
WORKDIR /app
COPY ./package.json /app
RUN [ "npm", "install" ]
COPY . /app/
CMD [ "npm", "start" ]

把这个 Dockerfile 放到 Node.js 项目的根目录,构建好镜像后,即可直接拿来启动容器运行,但是若还有第二个 Node.js 项目也产不多呢?那就再把这个 Dockerfile 复制到第二个项目中,若有第三个项目呢?继续复制?文件的副本越多,版本控制就月困难,让我们继续看这样的场景维护问题

若第一个 Node.js 项目在开发过程中,发现这个 Dockerfile 里存在问题,比如敲错字了、或者需要安装额外的包,然后开发人员修复了这个 Dockerfile,再次构建,问题解决,第一个项目没问题了,但第二个项目呢?虽然最初 Dockerfile 是复制、粘贴自第一个项目的,但并不会因为第一个项目修复了他们的 Dockerfile,而第二个项目的 Dockerfile 就会被自动修复

那可不可以做一个基础镜像,然后各个项目使用这个基础镜像呢?这样基础镜像更新,各个项目不用同步 Dockerfile 变化,重新构建后就继承了基础镜像的更新?好吧,可以,来看这样的结果,上面的 Dockerfile 将变为

FROM node:slim
RUN mkdir /app
WORKDIR /app
CMD [ "npm", "start" ]

这里我们把项目相关的构建指令拿出来,放到子项目中,假设这个基础镜像的名字为 my-node 的话,各个项目内的 Dockerfile 应变为

FROM my-node
COPY ./package.json /app
RUN [ "npm", "install" ]
COPY . /app/

基础镜像变化后,各个项目都用这个 Dockerfile 重新构建镜像,会继承基础镜像的更新

那么,问题解决了么?没有,准确说,只解决了一半,若这个 Dockerfile 里面有些东西需要调整呢?比如 npm install 都需要加一些参数,那怎么办?这一行 RUN 是不可能放入基础镜像的,因为涉及到了当前项目的 ./package.json,难道又要一个个修改么?所以说,这样制作基础镜像,只解决了原来的 Dockerfile 的前 4 条指令的变化问题,而后面三条指令的变化则完全没办法处理

ONBUILD 可解决这个问题,让我们用 ONBUILD 重新写一次基础镜像的 Dockerfile

FROM node:slim
RUN mkdir /app
WORKDIR /app
ONBUILD COPY ./package.json /app
ONBUILD RUN [ "npm", "install" ]
ONBUILD COPY . /app/
CMD [ "npm", "start" ]

这次我们回到原始的 Dockerfile,但是这次将项目相关的指令加上 ONBUILD,这样在构建基础镜像的时候,这三行并不会执行,然后各个项目的 Dockerfile 就变成了简单的

FROM my-node

是的,只有这么一行,当在各个项目目录中,用这个只有一行的 Dockerfile 构建镜像时,之前基础镜像的那三行 ONBUILD 就会开始执行,成功的将当前项目的代码复制进镜像、并且针对本项目执行 npm install,生成应用镜像

官方仓库示例

这些官方仓库的 Dockerfile 都是参考典范:https://github.com/docker-library/docs

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容