构建多平台镜像

多平台镜像使用场景

我们知道Docker镜像是支持多平台(不同的操作系统/架构)的,比如linux/amd64,linux/arm64,linux/riscv64等,当我们需要在不同平台使用容器运行我们的镜像的时候,通常可能会考虑分别编译各个平台的镜像文件,然后打上不同的tag用来区分平台,使用的时候也同样需要根据实际运行的平台在配置文件中选择不同的tag,这样就导致配置文件无法良好的共用,使用起来相当的不方便。

多平台镜像就是解决此问题的一个方案,那么什么是多平台镜像呢?我们以MySQL的官方镜像为例(Docker Hub上的大多数Docker官方镜像都提供了多平台镜像)

可以看到,该镜像支持在linux/amd64和linux/arm64平台下运行,通过执行相同的命令docker pull mysql:8.0获取镜像,docker会自动根据manifest的描述找到并下载适合当前系统架构的镜像文件。

如此,便可以在多个平台之间共用同一份配置文件,而无需多余的处理,是不是方便很多呢?


如何构建多平台镜像

接下来介绍两种构建多平台镜像的方法

方法一(推荐)

此方法需要开启docker的containerd-snapshotter特性

开启containerd-snapshotter

修改/etc/docker/daemon.json文件,添加

text 复制代码
"features": {
  "containerd-snapshotter": true
}

重启docker服务

注意:开启该配置以后,会导致原来下载的镜像文件被隐藏,开启和关闭该配置情况下的镜像文件是隔离存储的(OrbStack下验证)

OrbStack在设置里面的Docker功能页添加

Docker Desktop也可以参考docs.docker.com/desktop/con...

查看当前构建工具支持的平台

shell 复制代码
docker buildx inspect --bootstrap

其中的Platforms就是当前支持编译构建的平台

编译多平台镜像

编译指定的多平台镜像加载到本地,平台之间需使用逗号隔开

shell 复制代码
docker buildx build --platform linux/amd64,linux/arm64 -t martindai/wechat-robot:1.0 --load .

如果不想把镜像加载到本地,想要直接推送到仓库,修改--load--push即可

查看镜像列表

shell 复制代码
docker images

这里你会发现,有两个镜像文件,其中第一个就是我们编译的多平台镜像,文件大小比较大,而第二个则是适用于当前平台的镜像文件。

推送到仓库(确保已经登录)

shell 复制代码
docker push martindai/wechat-robot:1.0

可以看到,我们的多平台镜像就完成了。

方法二

我们可以使用命令查看一下方法一编译出来的多平台镜像的manifest信息

shell 复制代码
docker manifest inspect martindai/wechat-robot:1.0

可以看到,本质上这个多平台镜像只是一个索引,并不包含实际的文件,实际背后就是适用于两个平台的独立的镜像文件,docker在使用该镜像的时候会解析该索引文件,然后选择拉取合适的实际的镜像文件,相当于对于使用者屏蔽了平台这一层的信息。

方法一其实就是编译完多个平台的镜像以后,自动创建了一个索引,然后把各个平台的镜像做了一个关联。

那么方法二就是要手动创建这个索引,镜像的关联完全由我们自己控制。

下面开始操作

注意:此方法不能开启containerd-snapshotter特性

交叉编译多平台镜像

通过tag区分平台

shell 复制代码
docker buildx build --platform linux/amd64 -t martindai/wechat-robot:1.1-amd64 .
docker buildx build --platform linux/arm64 -t martindai/wechat-robot:1.1-arm64 .

或通过镜像名区分平台

shell 复制代码
docker buildx build --platform linux/amd64 -t martindai/wechat-robot-amd64:1.1 .
docker buildx build --platform linux/arm64 -t martindai/wechat-robot-arm64:1.1 .

PS:下面操作基于tag区分平台

推送到仓库

手动关联的镜像必须要在仓库里面有才行,所以需要先把编译好的单平台镜像推送到仓库

shell 复制代码
docker push martindai/wechat-robot:1.1-amd64
docker push martindai/wechat-robot:1.1-arm64

创建manifest关联镜像

shell 复制代码
docker manifest create martindai/wechat-robot:1.1 martindai/wechat-robot:1.1-amd64 martindai/wechat-robot:1.1-arm64 --amend

确认manifest的信息

shell 复制代码
docker manifest inspect martindai/wechat-robot:1.1

可以看到该manifest跟方法一的manifest是类似的(不是完全一样,类型其实是不一样的,只是效果类似),也是关联了两个平台镜像。

如果需要修改manifest,可以使用如下命令

shell 复制代码
docker manifest annotate --arch arm64 martindai/wechat-robot:1.1 martindai/wechat-robot-arm64:1.1

该命令表示,修改martindai/wechat-robot:1.1的manifest的arm64架构关联的镜像为martindai/wechat-robot-arm64:1.1

当然也可以删除manifest,重新创建

shell 复制代码
docker manifest rm martindai/wechat-robot:1.1

推送manifest到仓库

信息确认没问题以后,把创建的menifest推送到远程仓库

shell 复制代码
docker manifest push martindai/wechat-robot:1.1

可以看到仓库多了一个多平台镜像,并且关联的就是我们之前上传的两个单平台镜像。


总结

方法一使用起来比较方便,也是个人比较推荐的,可以配置在稳定的测试/生产环境使用。

方法二使用起来稍微会麻烦一点,但是会比较灵活,比较适合一些定制化/开发场景。

两种方法都可以完成创建多平台镜像的工作,具体使用就看个人根据实际情况选择。

相关推荐
wusam2 小时前
螺蛳壳里做道场:老破机搭建的私人数据中心---Centos下Docker学习04(环境准备)
学习·docker·centos
wusam5 小时前
螺蛳壳里做道场:老破机搭建的私人数据中心---Centos下Docker学习03(网络及IP规划)
运维·服务器·网络·docker·容器
一直在进步的派大星6 小时前
Docker 从安装到实战
java·运维·docker·微服务·容器
技术钱11 小时前
docker简介
运维·docker·容器
roman_日积跬步-终至千里11 小时前
【docker】docker常见命令
运维·docker·容器
tangdou36909865515 小时前
Docker系列-超级详细教你Linux安装并使用docker compose,如何使用docker-compose安装sqlserver
docker·容器·sql server
tangdou36909865516 小时前
手把手非常详细图文并茂教你 Docker 部署 SQL Server
docker·容器·sql server
飘逸高铁侠1 天前
docker export/import 和 docker save/load 的区别
docker·容器·eureka
大宇进阶之路1 天前
docker运行arm64架构的镜像、不同平台镜像构建
docker·微服务·架构
长天一色1 天前
【Docker从入门到进阶】04.高效实践
运维·docker·容器