在文件更改时重建Docker容器
为了运行ASP.NET
Core应用程序,我生成了一个dockerfile,该文件构建了该应用程序,并将源代码复制到了容器中,该容器由Git使用Jenkins获取。因此,在我的工作区中,我在dockerfile中执行以下操作:
WORKDIR /appCOPY src src
虽然Jenkins使用Git正确更新了主机上的文件,但Docker并未将其应用于我的映像。
我的基本构建脚本:
#!/bin/bashimageName=xx:my-image
containerName=my-container
docker build -t $imageName -f Dockerfile .
containerRunning=$(docker inspect --format="{{ .State.Running }}" $containerName 2> /dev/null)
if [ "$containerRunning" == "true" ]; then
docker stop $containerName
docker start $containerName
else
docker run -d -p 5000:5000 --name $containerName $imageName
fi
我尝试了各种不同的操作,例如--rm
和--no-cache
参数,docker run
并 在 构建新容器 之前
停止/删除了容器。我不确定我在做什么错。似乎docker正在正确更新映像,因为的调用COPY src src
会导致图层ID且没有缓存调用:
Step 6 : COPY src src ---> 382ef210d8fd
推荐的更新容器的方法是什么?
我的典型场景是:应用程序在Docker容器中的服务器上运行。现在,应用程序的某些部分已更新,例如通过修改文件。现在,容器应运行新版本。Docker似乎建议构建一个新映像而不是修改现有容器,因此我认为像我一样进行重建的一般方法是正确的,但是实现中的一些细节必须改进。
回答:
经过一些研究和测试,我发现我对Docker容器的生命存在一些误解。在此期间重建映像时,仅重新启动容器并不会使Docker使用新映像。相反,Docker仅
在 创建容器 之前 获取映像。因此,运行容器后的状态是持久的。
为什么需要删除
因此,仅重建和重新启动是不够的。我认为容器就像服务一样工作:停止服务,进行更改,重新启动它,它们将适用。那是我最大的错误。
由于容器是永久性的,因此您必须先使用容器将其删除docker rm <ContainerName>
。删除容器后,您不能简单地通过来启动它docker
start。必须使用来完成此操作docker run
,本身使用最新的图像来创建新的容器实例。
容器应尽可能独立
有了这些知识,就可以理解为什么将数据存储在容器中被视为不良做法,而Docker建议改为使用数据卷/挂载主机目录:由于必须销毁容器来更新应用程序,因此内部存储的数据也将丢失。这会导致额外的工作来关闭服务,备份数据等。
因此,将这些数据完全从容器中排除是一个明智的解决方案:当数据安全地存储在主机上并且容器仅包含应用程序本身时,我们不必担心我们的数据。
为什么-rf
不能真正帮助您
该docker run
命令有一个名为的 清理
开关-rf
。它将停止永久保留docker容器的行为。使用-rf
,Docker将在退出容器后销毁该容器。但是此开关有两个问题:
- Docker还删除了没有与容器相关联的名称的卷,这可能会杀死您的数据
- 使用此选项,将无法使用
-d
switch 在后台运行容器
虽然该-rf
开关是节省开发过程中的工作以进行快速测试的不错选择,但它不适用于生产环境。特别是由于缺少在后台运行容器的选项,因此通常需要这样做。
如何取出容器
我们可以通过简单地删除容器来绕过这些限制:
docker rm --force <ContainerName>
的--force
(或-f
其中使用SIGKILL上运行的容器)开关。相反,您也可以在以下之前停止容器:
docker stop <ContainerName>docker rm <ContainerName>
两者相等。docker
stop也正在使用SIGTERM。但是,使用--force
switch会缩短脚本,尤其是在使用CI服务器时:docker
stop如果容器未运行,则会引发错误。这将导致Jenkins和许多其他CI服务器错误地认为构建失败。要解决此问题,您必须首先检查容器是否正在按问题中的方式运行(请参阅containerRunning
变量)。
用于重建Docker容器的完整脚本
根据这一新知识,我通过以下方式修复了脚本:
#!/bin/bashimageName=xx:my-image
containerName=my-container
docker build -t $imageName -f Dockerfile .
echo Delete old container...
docker rm -f $containerName
echo Run new container...
docker run -d -p 5000:5000 --name $containerName $imageName
这完美地工作:)
以上是 在文件更改时重建Docker容器 的全部内容, 来源链接: utcz.com/qa/429506.html