8

我工作的公司的开发人员要求我用 Docker 做一些不同的事情,然后我也习惯了。目标是拥有 2 个具有以下职责的容器:

容器 A:节点容器,它将构建前端反应应用程序并将捆绑包放入名为app/dist/. 完成后,容器将停止运行。

容器 B:一个高山 nginx 容器,它将从/usr/share/nginx/html/app.

已在容器 A 中构建的文件将使用将安装<Container A>/app/dist到的卷提供给容器<Container B>/usr/share/nginx/html/appB。

请注意,在公共可访问端口和 nginx 容器之间有一个 HAProxy 层,该容器称为app.

上面的任务正在使用 docker compose 文件进行编排,如下所示:

version: '2'
volumes:
  webapp_build_volume: {}
services:
  webapp_build:
    build:
      context: .
      dockerfile: 'config/nginx/dockerfile-builder'
    volumes:
      - webapp_build_volume:/app/dist
      - webapp_static_volume:/app/src/app/static
  app:
    build:
      context: 'config/haproxy'
      dockerfile: 'dockerfile-app-haproxy'
    links:
      - web
    volumes:
      - /var/run/docker.sock:/var/run/docker.sock
    ports:
      - '80:80'
      - '1936:1936'
  web:
    build:
      context: .
      dockerfile: 'config/nginx/dockerfile-web'
    environment:
      - EXCLUDE_PORTS=443
      - VIRTUAL_HOST=*
    depends_on:
      - webapp_build
    volumes:
       - webapp_build_volume:/usr/share/nginx/html/app

这目前仅在第一次构建 docker compose 文件时有效。创建卷后,卷中的文件不再更新。我读过命名卷在建立后无法更新,但我无法确认。我找到了涉及运行的解决方法,docker-compose rm --force && docker volume webapp_build rm但如果可能的话,我不想杀死缓存的容器,因为 CI 服务会变得太慢。

如果我能澄清任何事情,请告诉我(我知道这里有很多活动部分)。请注意,我也在使用 docker 2 beta,尽管我看不出这会如何改变我在这里所做的任何事情。

4

1 回答 1

7

这有点难以理解,但听起来您正在构建一个图像,将文件输出到您认为是一个卷中,并尝试使用它来填充另一个正在运行的容器使用的命名卷。

您的困惑很可能是构建容器不会安装卷,卷仅安装在正在运行的容器中。命名卷确实有一个功能,它将由映像的内容填充,但仅当您挂载一个空的命名卷时。您似乎在第一次构建+运行时利用了此功能,但在以后的构建中将无法再次使用。如果您在没有卷的情况下运行构建容器,您会发现您的文件按预期存在。

您可以轻松更新命名卷。我想到了两个选择。一种是使用您当前的进程,但将卷挂载点更改为“/target”之类的内容,并作为您CMD的构建容器,将源的内容复制到“/target”。那看起来像:

Dockerfile

...
RUN compile-cmd --output-to /local/build/dir

入口点.sh:

cp -a /local/build/dir/* /target/

码头工人-compose.yml:

version: '2'
services:
  webapp_build:
    build:
      context: .
      dockerfile: 'config/nginx/dockerfile-builder'
    volumes:
      - webapp_build_volume:/target
...

第二种选择是根本不在容器构建中执行此操作,而是使用您的应用程序编译先决条件制作一个容器。然后将您的应用程序代码作为卷安装到此容器中,使用 aCMDENTRYPOINT获取代码卷内容,对其进行编译,并将其输出到也已安装的命名卷。然后,您无需构建构建容器,只需运行安装了两个卷的编译容器。

入口点.sh:

compile-cmd --input-src=/source --output-to /target

码头工人-compose.yml:

version: '2'
services:
  webapp_build:
    volumes:
      - app/source:/source
      - webapp_build_volume:/target
...
于 2016-08-04T18:51:57.557 回答