Dockerize a Spring Boot application
Introduction
In this post, I’d like to present a few options to ship a spring boot application in a docker container. There are many ways to dockerize a spring boot (probably a nice google hit search), but I don’t see too much discussion around the pros and cons. So let’s jump into it.
Create new project
Go to https://start.spring.io/ and create a new project. I’ll be using:
- Gradle - Groovy
- Spring Boot 3.4.2
- Java 21
- Dependencies: Spring Web
For demonstration, I’m going to add a “/ping” endpoint and it’s going to return “pong”. Just simply create PingController.java
.
package com.nukesz.demo;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class PingController {
@GetMapping("/ping")
public String ping() {
return "Pong";
}
}
Build and run the application as a normal jar:
# Build it
./gradlew build
# Run it
java -jar build/libs/dockerize-spring-boot-0.0.1-SNAPSHOT.jar
The jar is actually created with the
bootJar
task. You can learn more about it in the spring doc
Verify our REST API is working as expected:
$ curl http://localhost:8080/ping
> Pong!
Create Dockerfile
manually
Our application is ready, so let’s create a docker image for it. First let’s
FROM eclipse-temurin:21
LABEL org.opencontainers.image.authors="Norbert Benczur"
RUN mkdir /opt/app
COPY build/libs/dockerize-spring-boot-*.jar /opt/app/myapp.jar
CMD ["java", "-jar", "/opt/app/myapp.jar"]
You can build and run the Docker image:
docker build -t dockerize-spring-boot .
docker run -it -p 8080:8080 --rm dockerize-spring-boot
Verify that we can reach our REST API within the container as expected:
$ curl http://localhost:8080/ping
> Pong!
Are we done? - Not at all.
What’s the problem?
Creating Dockerfile
manually has its pros and cons. It’s the most flexible solution where you control everything. No dependency needed.
But I think the biggest drawback with this approach is that it seems everything is working, but in fact it is hiding the underlining work that is missing. The problem comes when you need more than a Hello World
example.
Repetitive
When you have more than 1 java app to dockerize, the number of dockerfiles starts to grow and you have to maintain and update each file independently.
Efficiency
In this simple example, we defined our base image and started our fat jar. But is that the most optimal way to build and run a spring boot (or any other java) application? For example, let’s change a single file in our application and build the image again:
# Let's measure the re-build
$ time ( ./gradlew build -x test; docker build -t dockerize-spring-boot . )
> ..
> => [3/3] COPY build/libs/dockerize-spring-boot-*.jar /opt/app/myapp.jar
> ..
> real 0m7,640s
So even a single change could cause the jar to be re-built and copied again. We are obviously not using the benefits of docker layers.
Can’t we leverage other people’s work rather than trying to come up with most optimal Dockerfile
ourself?
Use Buildpacks
Many of the problems that I mentioned previously comes from that the fact that we wrote our Dockerfile
manually. So let’s look another approach which does not need any Dockerfile at all.
Buildpacks transform the source code into containers without Dockerfiles. There are multiple projects that implements the Cloud Native Buildpacks (CNBs) spec and probably best two options for java application today is: Paketo and Jib.
Paketo
We can use paketo by executing the following gradle (or maven) task:
./gradlew bootBuildImage
The task will look and analyze your files, configurations and create a docker image based on it. The task’s output is really user friendly and show all the parameters that it used for building the image.
> Task :bootBuildImage
Building image 'docker.io/library/dockerize-spring-boot:0.0.1-SNAPSHOT'
> Pulling builder image 'docker.io/paketobuildpacks/builder-jammy-java-tiny:latest'
> Pulled builder image 'paketobuildpacks/builder-jammy-java-tiny@sha256:c5c53'
> Pulling run image 'docker.io/paketobuildpacks/run-jammy-tiny:latest' for platform 'linux/amd64'
> Pulled run image 'paketobuildpacks/run-jammy-tiny@sha256:0c5ac'
> Executing lifecycle version v0.20.6
> Using build cache volume 'pack-cache-7b0feb92a365.build'
> Running creator
[creator] ===> ANALYZING
[creator] Image with name "docker.io/library/dockerize-spring-boot:0.0.1-SNAPSHOT" not found
[creator] ===> DETECTING
[creator] ...
[creator] ===> RESTORING
[creator] ===> BUILDING
[creator] ...
[creator] Paketo Buildpack for Spring Boot 5.32.1
[creator] https://github.com/paketo-buildpacks/spring-boot
[creator] Build Configuration:
[creator] $BPL_JVM_CDS_ENABLED false whether to enable CDS optimizations at runtime
[creator] $BPL_SPRING_AOT_ENABLED false whether to enable Spring AOT at runtime
[creator] $BP_JVM_CDS_ENABLED false whether to enable CDS & perform JVM training run
[creator] $BP_SPRING_AOT_ENABLED false whether to enable Spring AOT
[creator] $BP_SPRING_CLOUD_BINDINGS_DISABLED false whether to contribute Spring Boot cloud bindings support
[creator] $BP_SPRING_CLOUD_BINDINGS_VERSION 1 default version of Spring Cloud Bindings library to contribute
[creator] Launch Configuration:
[creator] $BPL_SPRING_CLOUD_BINDINGS_DISABLED false whether to auto-configure Spring Boot environment properties from bindings
[creator] $BPL_SPRING_CLOUD_BINDINGS_ENABLED true Deprecated - whether to auto-configure Spring Boot environment properties from bindings
[creator] Creating slices from layers index
[creator] dependencies (19.5 MB)
[creator] spring-boot-loader (458.8 KB)
[creator] snapshot-dependencies (0.0 B)
[creator] application (35.9 KB)
[creator] Spring Cloud Bindings 2.0.4: Contributing to layer
[creator] Downloading from https://repo1.maven.org/maven2/org/springframework/cloud/spring-cloud-bindings/2.0.4/spring-cloud-bindings-2.0.4.jar
[creator] Verifying checksum
[creator] Copying to /layers/paketo-buildpacks_spring-boot/spring-cloud-bindings
[creator] Web Application Type: Contributing to layer
[creator] Servlet web application detected
[creator] Writing env.launch/BPL_JVM_THREAD_COUNT.default
[creator] Launch Helper: Contributing to layer
[creator] Creating /layers/paketo-buildpacks_spring-boot/helper/exec.d/spring-cloud-bindings
[creator] 4 application slices
[creator] Image labels:
[creator] org.opencontainers.image.title
[creator] org.opencontainers.image.version
[creator] org.springframework.boot.version
[creator] ===> EXPORTING
[creator] Adding layer 'paketo-buildpacks/...
[creator] ...
Successfully built image 'docker.io/library/dockerize-spring-boot:0.0.1-SNAPSHOT'
BUILD SUCCESSFUL in 1m 57s
Let’s run and test our the newly created images as before:
$ docker run -it -p 8081:8080 --rm dockerize-spring-boot:0.0.1-SNAPSHOT
$ curl http://localhost:8081/ping
> Pong!
This already gives us a solid ground to build upon for production and uses all the best practices almost for free. For example we already could see two benefits:
- The size of the created docker image from the “manual”
Dockerfile
is 468MB while the image created by paketo with default settings is 265MB. - The docker layers are cached. When only the java source code is changed, we can see the following message: Reused 4/5 app layer(s). So we (Paketo) can optimize our build and rebuild only what’s actually necessary.
For more information please have a look on the Spring doc Packaging OCI Images.
Jib
Another option that I would like to present is called Jib. This is part of Google’s container tools to dockerize Java applications, similar what we have seen in the previous chapter.
Let’s use it via the gradle plugin. You can add the plugin to the build.gradle
file:
plugins {
id 'com.google.cloud.tools.jib' version '3.4.4'
}
And then create a docker image:
./gradlew jibDockerBuild --image=dockerize-spring-boot-jib
Let’s run and test one last time:
$ docker run -it -p 8082:8080 --rm dockerize-spring-boot-jib:latest
$ curl http://localhost:8082/ping
> Pong!
It’s a very rich plugin, so I recommend to spend time how to configure it (for example how to change the base image).
What’s the catch?
The major drawback with these options is that it introduces a dependency to you project (or even a plugin). You have to learn, understand, configure and maintain it. Contributors to (open source) projects can come and go, and there is no guarantee that these project will stay and get updates forever.
This problem is not unique to these plugins, but general for all dependencies, libraries and frameworks that your projects relies on, so make sure that you evaluate before adding it to your project.
Summary
I hope following this post you gained the knowledge how to dockerize a spring boot application and made it clear that there are always tradeoffs to consider when deciding how to create a docker image based on your application.