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