在docker中进行多阶段构建?

时间:2018-10-31 23:59:04

标签: docker docker-compose

如何在docker-compose.yml中指定多阶段构建?

对于每个变体(例如dev,prod ...),我都有一个包含2个docker文件的多阶段构建:

  • dev:Dockerfile.base + Dockerfile.dev
  • 或prod:Dockerfile.base + Dockerfile.prod

文件Dockerfile.base(所有变体常见):

FROM python:3.6
RUN apt-get update && apt-get upgrade -y
RUN pip install pipenv pip
COPY Pipfile ./
# some more common configuration...

文件Dockerfile.dev

FROM flaskapp:base
RUN pipenv install --system --skip-lock --dev
ENV FLASK_ENV development
ENV FLASK_DEBUG 1

文件Dockerfile.prod

FROM flaskapp:base
RUN pipenv install --system --skip-lock
ENV FLASK_ENV production

没有docker-compose,我可以构建为:

# Building dev
docker build --tag flaskapp:base -f Dockerfile.base .
docker build --tag flaskapp:dev -f Dockerfile.dev .
# or building prod
docker build --tag flaskapp:base -f Dockerfile.base .
docker build --tag flaskapp:dev -f Dockerfile.dev .

根据compose-file doc,我可以指定要构建的Dockerfile。

# docker-compose.yml
version: '3'
services:
  webapp:
    build:
      context: ./dir
      dockerfile: Dockerfile-alternate

但是如何在docker-compose.yml中指定2个Dockerfile(用于多阶段构建)?

2 个答案:

答案 0 :(得分:11)

如评论中所述,多阶段构建涉及一个Dockerfile来执行多个阶段。您拥有的是一张普通的基本图片。

您可以使用类似的语法将它们转换为非传统的多阶段构建(我说是非传统的,因为您不执行各层之间的任何复制,而是仅使用from行从上一阶段中选取) :

FROM python:3.6 as base
RUN apt-get update && apt-get upgrade -y
RUN pip install pipenv pip
COPY Pipfile ./
# some more common configuration...

FROM base as dev
RUN pipenv install --system --skip-lock --dev
ENV FLASK_ENV development
ENV FLASK_DEBUG 1

FROM base as prod
RUN pipenv install --system --skip-lock
ENV FLASK_ENV production

然后,您可以使用--target语法来构建一个阶段或另一个阶段,或者使用诸如以下的组成文件:

# docker-compose.yml
version: '3.4'
services:
  webapp:
    build:
      context: ./dir
      dockerfile: Dockerfile
      target: prod

最大的缺点是当前的构建引擎将经历每个阶段,直到达到目标为止。构建缓存可能意味着仅需不到一秒钟的时间。 BuildKit将于18.09进行实验,并且需要docker-compose的上游支持,它将更加智能,使其仅运行所需的命令即可构建所需的目标。

所有这些,我相信这是在试图将方钉固定在圆孔中。 docker-compose开发人员鼓励用户远离在compose文件本身中进行构建,因为在swarm模式下不支持该文件。相反,推荐的解决方案是使用CI / CD构建服务器执行构建,然后将这些映像推送到注册表。然后,您可以使用docker-composedocker stack deploy甚至某些k8s等效文件运行相同的撰写文件,而无需重新设计工作流程。

答案 1 :(得分:0)

您还可以使用docker-compose文件的简洁化,包括同时包含指向现有dockerfile并运行dockerfile的{​​{1}}

相关问题