Skip to content

Day 13 - Multistage Builds

Published:ย atย 12:00 AM

Welcome to Day 13 of our Docker adventure! Today, weโ€™ll explore multistage builds - a powerful feature that helps us create smaller, more secure Docker images. Weโ€™ll transform our previous Go application into a lean, production-ready container using multistage builds.

What are Multistage Builds?

Multistage builds allow us to use multiple FROM statements in our Dockerfile. Each FROM instruction starts a new build stage, and we can selectively copy artifacts from one stage to another. This means we can include build tools and dependencies in earlier stages without bloating our final image.

Before: Single Stage Build

Letโ€™s look at our previous Go application Dockerfile:

FROM golang:latest

WORKDIR /app
COPY . .

RUN go build -o main .

EXPOSE 8080
CMD ["./main"]

This Dockerfile works, but it includes the entire Go toolchain and build dependencies in our final image. Letโ€™s check its size:

$ docker build -t hello-world-go .
$ docker images hello-world-go
REPOSITORY       TAG       IMAGE ID       CREATED      SIZE
hello-world-go   latest    6209fff2e822   2 days ago   928MB

Nearly 1GB for a simple Go application! ๐Ÿ˜ฑ

After: Multistage Build

Hereโ€™s how we can improve it using multistage builds:

# Build stage
FROM golang:latest AS builder

WORKDIR /app
COPY . .

RUN go build -o main .

# Final stage
FROM alpine:3.18

WORKDIR /app
COPY --from=builder /app/main .

EXPOSE 8080
CMD ["./main"]

Letโ€™s break down whatโ€™s happening:

  1. The first stage (named builder) uses the full Go image to compile our application
  2. The second stage starts fresh from a minimal Alpine Linux image
  3. We copy only the compiled binary from the builder stage
  4. Everything else from the build stage is discarded

Now letโ€™s check the size:

$ docker build -t hello-world-go-multistage .
$ docker images hello-world-go-multistage
REPOSITORY                  TAG       IMAGE ID       CREATED         SIZE
hello-world-go-multistage   latest    dc5a7b64247a   6 seconds ago   14.9MB

Thatโ€™s a 98% reduction in size! ๐ŸŽ‰

More Examples

Multistage builds are not only useful for Go applications. They can be applied to any language or framework. Here are some more examples:

Node.js Application

Hereโ€™s how a multistage build looks for a Node.js static website (like adventofdocker.com):

# Build stage - compiles the application
FROM node:lts AS base
WORKDIR /app
COPY package*.json ./
RUN npm install
COPY . .
RUN npm run build

# Runtime stage - serves the static files
FROM nginx:mainline-alpine-slim AS runtime
COPY --from=base ./app/dist /usr/share/nginx/html
EXPOSE 80

This example shows how to build and serve a static website using two stages:

  1. The build stage uses Node.js to install dependencies and build the application
  2. The runtime stage uses a lightweight nginx image to serve only the built static files

This approach significantly reduces the final image size by excluding Node.js, npm, and all development dependencies from the runtime image.

Benefits of Multistage Builds

  1. Smaller Image Size: Final images contain only whatโ€™s necessary to run the application
  2. Better Security: Fewer components mean a smaller attack surface
  3. Faster Deployments: Smaller images are faster to push and pull
  4. Clean Separation: Build-time dependencies are completely separated from runtime

Best Practices

Conclusion

Multistage builds are a game-changer for containerized applications. They help us maintain lean, secure, and efficient Docker images without compromising our development workflow. Whether youโ€™re working with Go, Node.js, Java, or any other language, multistage builds should be part of your Docker best practices!

Tomorrow, weโ€™ll explore common debugging techniques to help you solve issues faster. Until then, happy containerizing! ๐Ÿณ

Jonas


Previous Post
Day 12 - Docker Compose Deep Dive
Next Post
Day 14 - Debugging Docker
Sponsor logo

Sliplane

Deploy your Docker Apps straight from your Github repository in less than 2 minutes with sliplane.io

Learn More โ†’