前言
Docker Buildx 是一个 docker CLI 插件,其扩展了 docker 命令,支持 Moby BuildKit 提供的功能。提供了与 docker build 相同的用户体验,并增加了许多新功能。
该功能仅适用于 Docker v19.03+ 版本
BuildKit
BuildKit 是下一代的镜像构建组件,在 https://github.com/moby/buildkit 开源。
注意:如果您的镜像构建使用的是云服务商提供的镜像构建服务(腾讯云容器服务、阿里云容器服务等),由于上述服务提供商的 Docker 版本低于 18.09,BuildKit 无法使用,将造成镜像构建失败。建议使用 BuildKit 构建镜像时使用一个新的 Dockerfile 文件(例如 Dockerfile.buildkit)
目前,Docker Hub 自动构建已经支持 buildkit,具体参考 https://github.com/docker-practice/docker-hub-buildx
Dockerfile 新增指令详解
启用 BuildKit 之后,我们可以使用下面几个新的 Dockerfile 指令来加快镜像构建。
RUN --mount=type=cache
目前,几乎所有的程序都会使用依赖管理工具,例如 Go 中的 go mod、Node.js 中的 npm 等等,当我们构建一个镜像时,往往会重复的从互联网中获取依赖包,难以缓存,大大降低了镜像的构建效率。
例如一个前端工程需要用到 npm:
1 | FROM node:alpine as builder |
使用多阶段构建,构建的镜像中只包含了目标文件夹 dist,但仍然存在一些问题,当 package.json 文件变动时,RUN npm i && rm -rf ~/.npm 这一层会重新执行,变更多次后,生成了大量的中间层镜像。
为解决这个问题,进一步的我们可以设想一个类似 数据卷 的功能,在镜像构建时把 node_modules 文件夹挂载上去,在构建完成后,这个 node_modules 文件夹会自动卸载,实际的镜像中并不包含 node_modules 这个文件夹,这样我们就省去了每次获取依赖的时间,大大增加了镜像构建效率,同时也避免了生成了大量的中间层镜像。
BuildKit 提供了 RUN --mount=type=cache 指令,可以实现上边的设想。
1 | # syntax = docker/dockerfile:experimental |
由于 BuildKit 为实验特性,每个 Dockerfile 文件开头都必须加上如下指令
1 | # syntax = docker/dockerfile:experimental |
第一个 RUN 指令执行后,id 为 my_app_npm_module 的缓存文件夹挂载到了 /app/node_modules 文件夹中。多次执行也不会产生多个中间层镜像。
第二个 RUN 指令执行时需要用到 node_modules 文件夹,node_modules 已经挂载,命令也可以正确执行。
第三个 RUN 指令将上一阶段产生的文件复制到指定位置,from 指明缓存的来源,这里 builder 表示缓存来源于构建的第一阶段,source 指明缓存来源的文件夹。
上面的 Dockerfile 中 --mount=type=cache,... 中指令作用如下:
| Option | Description |
|---|---|
id | id 设置一个标志,以便区分缓存。 |
target (必填项) | 缓存的挂载目标文件夹。 |
ro,readonly | 只读,缓存文件夹不能被写入。 |
sharing | 有 shared private locked 值可供选择。sharing 设置当一个缓存被多次使用时的表现,由于 BuildKit 支持并行构建,当多个步骤使用同一缓存时(同一 id)会发生冲突。shared 表示多个步骤可以同时读写,private 表示当多个步骤使用同一缓存时,每个步骤使用不同的缓存,locked 表示当一个步骤完成释放缓存后,后一个步骤才能继续使用该缓存。 |
from | 缓存来源(构建阶段),不填写时为空文件夹。 |
source | 来源的文件夹路径。 |
RUN --mount=type=bind
该指令可以将一个镜像(或上一构建阶段)的文件挂载到指定位置。
1 | # syntax = docker/dockerfile:experimental |
RUN --mount=type=tmpfs
该指令可以将一个 tmpfs 文件系统挂载到指定位置。
1 | # syntax = docker/dockerfile:experimental |
RUN --mount=type=secret
该指令可以将一个文件(例如密钥)挂载到指定位置。
1 | # syntax = docker/dockerfile:experimental |
1 | $ docker build -t test --secret id=aws,src=$HOME/.aws/credentials . |
RUN --mount=type=ssh
该指令可以挂载 ssh 密钥。
1 | # syntax = docker/dockerfile:experimental |
1 | $ eval $(ssh-agent) |
docker-compose build 使用 Buildkit
设置 COMPOSE_DOCKER_CLI_BUILD=1 环境变量即可使用。
官方文档
https://github.com/moby/buildkit/blob/master/frontend/dockerfile/docs/experimental.md
使用 buildx 构建镜像
使用
你可以直接使用 docker buildx build 命令构建镜像。
1 | $ docker buildx build . |
Buildx 使用 BuildKit 引擎 进行构建,支持许多新的功能,具体参考 Buildkit 一节。
官方文档
https://docs.docker.com/engine/reference/commandline/buildx/
使用 buildx 构建多种系统架构支持的 Docker 镜像
在之前的版本中构建多种系统架构支持的 Docker 镜像,要想使用统一的名字必须使用 $ docker manifest 命令。
在 Docker 19.03+ 版本中可以使用 $ docker buildx build 命令使用 BuildKit 构建镜像。该命令支持 --platform 参数可以同时构建支持多种系统架构的 Docker 镜像,大大简化了构建步骤。
新建 builder 实例
Docker for Linux 不支持构建 arm 架构镜像,我们可以运行一个新的容器让其支持该特性,Docker 桌面版无需进行此项设置。
1 | $ docker run --rm --privileged tonistiigi/binfmt:latest --install all |
由于 Docker 默认的 builder 实例不支持同时指定多个 --platform,我们必须首先创建一个新的 builder 实例。同时由于国内拉取镜像较缓慢,我们可以使用配置了 镜像加速地址 的 dockerpracticesig/buildkit:master 镜像替换官方镜像。
如果你有私有的镜像加速器,可以基于 https://github.com/docker-practice/buildx 构建自己的 buildkit 镜像并使用它。
1 | # 适用于国内环境 |
构建镜像
新建 Dockerfile 文件。
1 | FROM --platform=$TARGETPLATFORM alpine |
使用 $ docker buildx build 命令构建镜像,注意将 myusername 替换为自己的 Docker Hub 用户名。
--push 参数表示将构建好的镜像推送到 Docker 仓库。
1 | $ docker buildx build --platform linux/arm,linux/arm64,linux/amd64 -t myusername/hello . --push |
在不同架构运行该镜像,可以得到该架构的信息。
1 | # arm |
架构相关变量
Dockerfile 支持如下架构相关的变量
TARGETPLATFORM
构建镜像的目标平台,例如 linux/amd64, linux/arm/v7, windows/amd64。
TARGETOS
1 | TARGETPLATFORM` 的 OS 类型,例如 `linux`, `windows |
TARGETARCH
1 | TARGETPLATFORM` 的架构类型,例如 `amd64`, `arm |
TARGETVARIANT
1 | TARGETPLATFORM` 的变种,该变量可能为空,例如 `v7 |
BUILDPLATFORM
构建镜像主机平台,例如 linux/amd64
BUILDOS
1 | BUILDPLATFORM` 的 OS 类型,例如 `linux |
BUILDARCH
1 | BUILDPLATFORM` 的架构类型,例如 `amd64 |
BUILDVARIANT
1 | BUILDPLATFORM` 的变种,该变量可能为空,例如 `v7 |
使用举例
例如我们要构建支持 linux/arm/v7 和 linux/amd64 两种架构的镜像。假设已经生成了两个平台对应的二进制文件:
bin/dist-linux-armbin/dist-linux-amd64
那么 Dockerfile 可以这样书写:
1 | FROM scratch |
