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.

Avatar
Norbert Benczúr
Senior Developer / R&D Manager
comments powered by Disqus