on
Multi-stage building with Docker
Docker supports a feature, where you can create multiple containers with a single dockerfile and at the same time share the contents between the containers. This feature gives you the possibility of having multi-stage builds and opens one significant spot for creating repeatable builds.
The idea behind it is simple, you can have multiple FROM
statements inside the
same Dockerfile
. Each FROM
statement can use different images, therefore
defines it’s own dependencies and allows you to connect the containers from one
base to the other by using COPY --from=0or COPY --from=previous-stage
.
Each stage could be responsible for doing one thing only(don’t forget the unix philosphy), e.g. run unit tests.
Multi-Stage Builds in dotnet core
Here is a simplified example of a multi-stage Dockerfile which is used to build, test a dotnet core 2.0 application, plus a ready to run container.
# 1st stage
FROM microsoft/dotnet:2-sdk-jessie as builder
RUN mkdir /app
COPY . /app
WORKDIR /app
RUN dotnet restore \
&& dotnet build -c Release /app/src/MyProject \
&& dotnet build -c Release /app/test/Tests.MyProject \
&& dotnet test /app/test/Tests.MyProject \
&& dotnet publish -c Release -o /app/bin/MyProject /app/src/MyProject
# 2nd stage
FROM microsoft/dotnet:2.0.0-runtime-jessie
RUN mkdir -p /opt/app/bin
COPY --from=builder /app/bin /opt/app/bin
COPY --from=builder /app/startup.sh /opt/app
ENTRYPOINT ["/opt/app/startup.sh"]
Why use it
The benefits are very clear:
- you create builds that are CI/CD independent, i.e. you don’t care if you run a build with TeamCity or Gitlab or Jenkins.
- you have one host dependency, i.e. docker
- you commit in your repository the steps of your build, which means that you can have different steps on different branches (it might not work very well in TeamCity, but still it’s possible).
- consistent build environment, i.e. what is inside docker does not change, they are repeatable and consistent.
There is one minor issue, you need to remove a dangling image at the end of the build process ;)
Conclusion
The ability to repeat one build with the same consistency is crucial in today’s constantly changing dependencies. Docker allows you to maintain the same environment, even by making sure your build steps are the same. If you are already using docker it’s dead simple to use multi-stage builds. If you are new to docker you will still grasp the concept quickly.
References
NOTE: originally posted on medium https://medium.com/@ipinak/multi-stage-building-with-docker-37cd10a44e3c