【51CTO.com快译】由于Docker容器是小小的软件盒子,你可以将它们从一台计算机复制和转移到另一计算机。然而这不常讨论,可能是由于大企业自行构建库,可以定制自己的映像,并在它们觉得合适时跨服务器分发。但如果你只是偶尔需要移动一个或多个容器,没必要搞得这么复杂。
你可能在本地计算机上使用Docker实例,决定将它转移到功能更强大的服务器上。或者你可能只想在多台计算机上部署定制的容器,对容器进行“复制粘贴”。其他时候你可能对云计算提供商不满意,想另换一家。无论是什么情况,都有一个很简单的过程,下面会详细介绍。
保存来自源主机的容器映像
不需要先停止容器,但强烈建议这么做。你将对Docker实例中的数据创建快照。如果这么做时实例在运行,最终出现在快照中的一些文件有可能不完整。想象一下有人上传了一个500MB的文件。上载250MB后,你发出docker commit命令。上传随后继续,但是当你在另一台主机上恢复该Docker映像时,500MB中只有250MB可用。
所以,如果可以,先停止实例。
- docker stop NAME_OF_INSTANCE
Docker容器由普通的初始映像构建。随着时间的推移,你将自己的更改添加到该基本映像。容器内运行的进程也可能保存自己的数据或进行其他更改。想保留这一切,你可以将此新状态提交到新映像。
注意,如果实例目前在运行中,该操作会在其内容保存期间暂停该实例。如果向容器添加了大量数据,此操作需要较长时间才能完成。如果这是个问题,可以输入docker commit -p=false NAME_OF_INSTANCE mycontainerimage而不是下一个命令,避免这个暂停。然而,除非绝对必要,否则别这么做。在这种情况下,创建数据不一致/不完整的映像的可能性会加大。
在本教程中,为因此生成的映像选择了普通名称:mycontainerimage。如果你愿意,可以更改此名称。如果更改名称,记得在遇到它的所有后续命令中替换它。
- docker commit NAME_OF_INSTANCE mycontainerimage
现在,将此映像保存到文件,并压缩它。
- docker save mycontainerimage | gzip > mycontainerimage.tar.gz
接下来,使用你青睐的文件传输方法,将mycontainerimage.tar.gz复制到想把容器迁移过去的那个主机。
将容器映像加载到目标主机上
登录到你已将映像转移过去的主机后,将其导入到Docker。
- gunzip -c mycontainerimage.tar.gz | docker load
由于你从未在这里初始化该容器,还无法用docker start启动它。相反,你首次运行该Docker实例时,发出在过去使用的同一个命令。现在唯一的区别是你最后会使用“mycontainerimage”,而不是在过去使用的任何映像。
下一个命令只是一个例子;除非适用于你,否则别复制粘贴该命令。(首次运行映像时不需要特殊参数)
- docker run -d --name=PICK_NAME_FOR_CONTAINER mycontainerimage
相比之下,在下面这个命令示例中,需要参数--publish将主机上的端口80转发到容器上的端口80:
- docker run -d --name=http-server --publish 80:80 mycontainerimage
之后,你可以使用docker stop和docker start命令正常停止和启动该容器。
无需创建文件即可传输映像
有时你可能想跳过创建mycontainerimage.tar.gz文件这一步。也许你没有足够的磁盘空间,因为容器里面有大量数据。你可以保存、压缩、传输和解压缩映像,并将映像加载到目标主机上,一个命令即可搞定。运行前面讨论的docker commit命令后,你可以使用:
- docker save mycontainerimage | gzip | ssh root@203.0.113.1 'gunzip | docker load'
它在Windows上应该也管用,因为现在有内置的SSH客户端(PuTTY不再必不可少)。
然后,继续使用适用于你情况的docker run命令。
结束语
作为偶尔移动容器的临时方案,docker save和docker load很棒。但请记住,如果你经常移动容器,可能应该改而构建自己的专有库。
原文标题:How to Copy/Move a Docker Container to Another Host,作者:Alexandru Andrei
【51CTO译稿,合作站点转载请注明原文译者和出处为51CTO.com】