/java-docker-build-tutorial

A template project to create a minimal Docker image for a Java application. The example app uses Spring Boot to expose an HTTP endpoint.

Primary LanguageJavaApache License 2.0Apache-2.0

Project Template: Create a Docker image for a Java application

GitHub forks Docker workflow status Maven workflow status License

A template project to create a Docker image for a Java application. The example application uses Spring Boot to expose an HTTP endpoint at /welcome.

Features:

Requirements

Docker must be installed on your local machine. That's it. You do not need a Java JDK or Maven installed.

Usage and Demo

Step 1: Create the Docker image according to Dockerfile. This step uses Maven to build, test, and package the Java application according to pom.xml. The resulting image is 161MB in size, of which 44MB are the underlying alpine image.

# ***Creating an image may take a few minutes!***
$ docker build --platform linux/x86_64/v8 -t miguno/java-docker-build-tutorial:latest .

# You can also build with the new BuildKit.
# https://docs.docker.com/build/
$ docker buildx build --platform linux/x86_64/v8 -t miguno/java-docker-build-tutorial:latest .

Optionally, you can check the size of the generated Docker image:

$ docker images miguno/java-docker-build-tutorial
REPOSITORY                          TAG       IMAGE ID       CREATED         SIZE
miguno/java-docker-build-tutorial   latest    bd64d898a04e   2 minutes ago   131MB

Step 2: Start a container for the Docker image.

$ docker run -p 8123:8123 miguno/java-docker-build-tutorial:latest
Example output (click to expand)
Running container from docker image ...
Starting container for image 'miguno/java-docker-build-tutorial:latest', exposing port 8123/tcp
- Run 'curl http://localhost:8123/welcome' to send a test request to the containerized app.
- Enter Ctrl-C to stop the container.

  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/

 :: Spring Boot ::                (v3.3.3)

2024-08-26T15:45:08.859Z  INFO 1 --- [main] com.miguno.javadockerbuild.App           : Starting App v1.0.0-SNAPSHOT using Java 22.0.2 with PID 1 (/app/app.jar started by appuser in /app)
2024-08-26T15:45:08.868Z  INFO 1 --- [main] com.miguno.javadockerbuild.App           : No active profile set, falling back to 1 default profile: "default"
2024-08-26T15:45:10.930Z  INFO 1 --- [main] o.s.b.w.embedded.tomcat.TomcatWebServer  : Tomcat initialized with port 8123 (http)
2024-08-26T15:45:10.950Z  INFO 1 --- [main] o.apache.catalina.core.StandardService   : Starting service [Tomcat]
2024-08-26T15:45:10.951Z  INFO 1 --- [main] o.apache.catalina.core.StandardEngine    : Starting Servlet engine: [Apache Tomcat/10.1.28]
2024-08-26T15:45:10.991Z  INFO 1 --- [main] o.a.c.c.C.[Tomcat].[localhost].[/]       : Initializing Spring embedded WebApplicationContext
2024-08-26T15:45:10.992Z  INFO 1 --- [main] w.s.c.ServletWebServerApplicationContext : Root WebApplicationContext: initialization completed in 2004 ms
2024-08-26T15:45:12.452Z  INFO 1 --- [main] o.s.b.a.e.web.EndpointLinksResolver      : Exposing 1 endpoint beneath base path '/actuator'
2024-08-26T15:45:12.562Z  INFO 1 --- [main] o.s.b.w.embedded.tomcat.TomcatWebServer  : Tomcat started on port 8123 (http) with context path '/'
2024-08-26T15:45:12.597Z  INFO 1 --- [main] com.miguno.javadockerbuild.App           : Started App in 5.0 seconds (process running for 6.246)

Step 3: Open another terminal and access the example API endpoint of the running container.

$ curl http://localhost:8123/welcome
{"welcome":"Hello, World!"}

Local usage without Docker

You can also build, test, package, and run the Java application locally (without Docker) if you have JDK 22+ installed. You do not need to have Maven installed, because this repository contains the Maven Wrapper mvnw (use mvnw.cmd on Windows).

# Build, test, package the application locally.
$ ./mvnw clean verify package

# Run the application locally.
$ ./mvnw spring-boot:run -Dspring-boot.run.jvmArguments="-XX:+UseZGC -XX:+ZGenerational"

# Alternatively, run the application locally via its jar file.
$ java -XX:+UseZGC -XX:+ZGenerational -jar target/app.jar

Appendix

Hot reloading during development

This project uses spring-boot-devtools for fast, automatic application restarts after code changes.

  • Restarts will be triggered whenever files in the classpath changed, e.g., after you ran ./mvnw compile or after you re-built the project in your IDE.
  • This feature works both when running the application inside an IDE like IntelliJ IDEA as well as when running the application in a terminal with ./mvnw spring-boot:run.
  • Be patient. After a file changed, it may take a few seconds for the refresh to happen.

In IntelliJ IDEA, you can also enable automatic project builds for even more convenience, using the following settings. Then, whenever you modify a source file, IDEA will automatically rebuild the project in the background and thus trigger an automatic restart:

  • Settings > Build, Execution, Deployment > Compiler: [X] Build project automatically
  • Settings > Advanced Settings: [X] Allow auto-make to start even if developed application is currently running

Restart vs. Reload: If you want true hot reloads that are even faster than automatic restarts, look at tools like JRebel.

Usage with just

If you have just installed, you can run the commands above more conveniently as per this project's justfile:

$ just
Available recipes:
    [benchmarking]
    benchmark-plow        # benchmark the app's HTTP endpoint with plow (requires https://github.com/six-ddc/plow)
    benchmark-wrk         # benchmark the app's HTTP endpoint with wrk (requires https://github.com/wg/wrk)

    [development]
    analyze               # perform static code analysis
    build                 # alias for 'compile'
    clean                 # clean (remove) the build artifacts
    compile               # compile the project
    coverage              # create coverage report
    dependencies          # list dependency tree of this project
    docs                  # generate Java documentation
    format                # format sources
    format-check          # check formatting of sources (without modifying)
    infer                 # static code analysis with infer (requires https://github.com/facebook/infer)
    outdated              # list outdated dependencies
    outdated-plugins      # list outdated maven plugins
    package               # package the app to create an uber jar
    send-request-to-app   # send request to the app's HTTP endpoint (requires running app)
    site                  # generate site incl. reports for spotbugs, dependencies, javadocs, licenses
    spotbugs              # static code analysis with spotbugs
    start                 # start the app
    start-jar             # start the app via its packaged jar (requires 'package' step)
    test                  # run unit tests
    verify                # run unit and integration tests, coverage check, static code analysis

    [docker]
    docker-image-create   # create a docker image (requires Docker)
    docker-image-run      # run the docker image (requires Docker)
    docker-image-size     # size of the docker image (requires Docker)

    [maven]
    maven-active-profiles # list active profiles
    maven-all-profiles    # list all profiles
    maven-help            # show help of maven-help-plugin
    maven-lifecycles      # show maven lifecycles like 'clean', 'compile'
    maven-pom             # print effective pom.xml
    maven-system          # print platform details like system properties, env variables
    mvnw-upgrade          # upgrade maven wrapper

    [project-agnostic]
    default               # print available targets
    evaluate              # evaluate and print all just variables
    system-info           # print system information such as OS and architecture

Example:

$ just docker-image-create

References