Docker-compose - 使用重用 maven 存储库的 maven 构建

tm1*_*701 7 maven docker docker-compose

使用 Maven 构建 Spring-boot 映像时,我现在在 Dockerfile 中执行此操作。Maven 将下载所有依赖项,然后编译我的文件。这需要相当长的时间。

如何通过 docker-compose (Dockerfile) 指定构建过程重新使用我的“Windows10 Home”Maven 存储库?因此,(新)下载的数量很少。我的开发环境:我使用 Docker 快速入门终端,因此使用 docker-machine。

这是我的 docker-compose.yml 文件的一部分:

version: '3'
services:
  spring-boot-app:
    image: spring-boot-image
    build:
      context: ./
      dockerfile: Dockerfile
    depends_on:
      - mysql-docker-container
    ports:
      - 8087:8080
    volumes:
      - $HOME/.m2/repository:/root/.m2/repository
      - /data/spring-boot-app
Run Code Online (Sandbox Code Playgroud)

我的 Dockerfile 是:

FROM java:8
FROM maven:alpine
WORKDIR /app
COPY . /app
RUN mvn -v
RUN mvn clean install -DskipTests
EXPOSE 8080
LABEL maintainer=“xyz@holland.nl”
ADD ./target/spring-boot-example-0.0.1-SNAPSHOT.jar /developments/
ENTRYPOINT ["java","-jar","/developments/spring-boot-example-0.0.1-SNAPSHOT.jar"]
Run Code Online (Sandbox Code Playgroud)

tm1*_*701 7

@Jack Gore - 非常感谢你给我指路。我看过那个帖子,但还没有明确的答案。在深入研究所有单独的答案之后,这些信息为我提供了症状的答案以及如何解决类似问题的见解。

解答:通过 Dockerfile,您可以通过 1 个或多个后续镜像层构建最终镜像。为防止该步骤每次都重新下载依赖项,您可以为依赖项的“下载世界”创建一个镜像层。只有在 pom.xml 文件中的依赖项更改时才会重新下载。

为此,您将 Dockerfile 拆分为 2 个构建步骤:首先复制 pom.xml 文件并构建它。这将创建一个具有所有依赖项的图像层。第二步是构建应用程序。

FROM java:8
FROM maven:alpine

# image layer
WORKDIR /app
ADD pom.xml /app
RUN mvn verify clean --fail-never

# Image layer: with the application
COPY . /app
RUN mvn -v
RUN mvn clean install -DskipTests
EXPOSE 8080
ADD ./target/your.jar /developments/
ENTRYPOINT ["java","-jar","/developments/your.jar"]
Run Code Online (Sandbox Code Playgroud)

然后你会得到以下构建场景:

  • 第一次你建立这个(泊坞窗版本。)的依赖被下载和步骤2中的应用程序JAR是构建。
  • 当您立即重新构建依赖项 (pom.xml) 和应用程序源时,不会更改。因此,不需要更改图像层。构建很快就准备好了。
  • 如果您更改了 1 个应用程序源文件,则只会下载少量下载并构建应用程序。所以你不是在下载世界。
  • 如果更改 pom.xml 文件,从而更改依赖项,则所有依赖项下载都已完成。

分离图像层的影响通过一些非常短的视频来展示

这种构建方式的缺点当然是最终(生产)图像包含的不仅仅是应用程序。里面不仅有 JAR,还有一些依赖。

如何解决这个图像太大:

  • 通过技巧减小图像尺寸。使用选项:--squash。这在这篇文章中有解释。
  • 将构建过程(此处:maven 和/或 ng build --prod)分离为 docker 构建步骤之前的一个步骤。然后执行 docker build 并仅将 jar 文件放入最终映像中。
  • 将 CI/CD 环境与例如 Jenkinsfile(管道)一起使用。在 Jenkins 管道中,您首先构建映像。所有依赖项都已经存在。您只需重建应用程序。您执行 docker build 等。

就我而言,这是可能的最佳选择。您可以自动化该过程并保持较低的图像大小。

  • 对于镜像大小,Docker 现在支持[多阶段构建](https://docs.docker.com/develop/develop-images/multistage-build/)。因此,基本上在构建 jar 后,您可以创建一个新的生产映像,并从开发映像复制 jar。因此,您的生产映像中没有任何构建/测试依赖项。 (3认同)