Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/openliberty/guide-testcontainers

A guide on how to test your microservices with multiple containers by using Testcontainers and JUnit.
https://github.com/openliberty/guide-testcontainers

docker jakarta-ee jakartaee junit junit5 microprofile microservices testcon

Last synced: about 2 months ago
JSON representation

A guide on how to test your microservices with multiple containers by using Testcontainers and JUnit.

Awesome Lists containing this project

README

        

// Copyright (c) 2024 IBM Corporation and others.
// Licensed under Creative Commons Attribution-NoDerivatives
// 4.0 International (CC BY-ND 4.0)
// https://creativecommons.org/licenses/by-nd/4.0/
//
// Contributors:
// IBM Corporation
//
:projectid: testcontainers
:page-layout: guide-multipane
:page-duration: 20 minutes
:page-releasedate: 2024-02-05
:page-guide-category: microprofile
:page-essential: false
:page-description: Learn how to test your microservices with multiple containers by using Testcontainers and JUnit.
:page-seo-title: Testing Java microservices using the Testcontainers and JUnit frameworks with Open Liberty and Docker containers
:page-seo-description: A tutorial and an example on how to develop and execute true-to-production integration tests in production-like environments for Java applications written with Jakarta EE and MicroProfile APIs by using Testcontainers, JUnit, Open Liberty, and Docker containers.
:guide-author: Open Liberty
:page-tags: ['microprofile', 'jakarta-ee', 'docker']
:page-related-guides: ['reactive-service-testing', 'arquillian-managed']
:page-permalink: /guides/{projectid}
:repo-description: Visit the https://openliberty.io/guides/{projectid}.html[website] for the rendered version of the guide.
:common-includes: https://raw.githubusercontent.com/OpenLiberty/guides-common/prod
:source-highlighter: prettify
:imagesdir: /img/guide/{projectid}
= Building true-to-production integration tests with Testcontainers

[.hidden]
NOTE: This repository contains the guide documentation source. To view the guide in published form, see the https://openliberty.io/guides/{projectid}.html[Open Liberty website].

Learn how to test your microservices with multiple containers by using Testcontainers and JUnit.

== What you'll learn

You'll learn how to write true-to-production integration tests for Java microservices by using https://www.testcontainers.org/[Testcontainers^] and JUnit. You'll learn to set up and configure multiple containers, including the Open Liberty Docker container, to simulate a production-like environment for your tests.

Sometimes tests might pass in development and testing environments, but fail in production because of the differences in how the application operates across these environments. Fortunately, you can minimize these differences by testing your application with the same Docker containers you use in production. This approach helps to ensure parity across the development, testing, and production environments, enhancing quality and test reliability.

=== What is Testcontainers?

Testcontainers is an open source library that provides containers as a resource at test time, creating consistent and portable testing environments. This is especially useful for applications that have external resource dependencies such as databases, message queues, or web services. By encapsulating these dependencies in containers, Testcontainers simplifies the configuration process and ensures a uniform testing setup that closely mirrors production environments.

The microservice that you'll be working with is called `inventory`. The `inventory` microservice persists data into a PostgreSQL database and supports create, retrieve, update, and delete (CRUD) operations on the database records. You'll write integration tests for the application by using Testcontainers to run it in Docker containers.

image::inventory.png[Inventory microservice,align="center",height=85%,width=85%]

// =================================================================================================
// Additional prerequisites
// =================================================================================================
== Additional prerequisites

Before you begin, Docker needs to be installed. For installation instructions, see the https://docs.docker.com/get-docker/[official Docker documentation^]. You'll test the application in Docker containers.

Make sure to start your Docker daemon before you proceed.

// =================================================================================================
// Getting Started
// =================================================================================================
[role='command']
include::{common-includes}/gitclone.adoc[]

ifdef::cloud-hosted[]
In this IBM Cloud environment, you need to change the user home to ***/home/project*** by running the following command:
```bash
sudo usermod -d /home/project theia
```
endif::[]

// =================================================================================================
// Try what you'll build
// =================================================================================================
=== Try what you'll build

The `finish` directory in the root of this guide contains the finished application. Give it a try before you proceed.

To try out the test, first go to the `finish` directory and run the following Maven goal that builds the application, starts the containers, runs the tests, and then stops the containers:

ifndef::cloud-hosted[]
[.tab_link.windows_link]
`*WINDOWS*`
[.tab_link.mac_link]
`*MAC*`
[.tab_link.linux_link]
`*LINUX*`
[.tab_content.windows_section.mac_section]
--
[role='command']
```
cd finish
mvn verify
```
--
[.tab_content.linux_section]
--
[role='command']
```
export TESTCONTAINERS_RYUK_DISABLED=true
cd finish
mvn verify
```
--
endif::[]

ifdef::cloud-hosted[]
```bash
cd /home/project/guide-testcontainers/finish
export TESTCONTAINERS_RYUK_DISABLED=true
mvn verify
```
endif::[]

You see the following output:

[role="no_copy"]
----
-------------------------------------------------------
T E S T S
-------------------------------------------------------
Running it.io.openliberty.guides.inventory.SystemResourceIT
...
Tests run: 3, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 10.118 s - in it.io.openliberty.guides.inventory.SystemResourceIT

Results:

Tests run: 3, Failures: 0, Errors: 0, Skipped: 0
----

// =================================================================================================
// Writing integration tests using Testcontainers
// =================================================================================================
== Writing integration tests using Testcontainers

Use Testcontainers to write integration tests that run in any environment with minimal setup using containers.

Navigate to the `postgres` directory.

ifdef::cloud-hosted[]
```bash
cd /home/project/guide-testcontainers/postgres
```
endif::[]

// file 0
postgres/Dockerfile
[source, Dockerfile, linenums, role="code_column"]
----
include::postgres/Dockerfile[]
----

This guide uses Docker to run an instance of the PostgreSQL database for a fast installation and setup. A [hotspot file=0]`Dockerfile` file is provided for you. Run the following command to use the Dockerfile to build the image:

[role='command']
```
docker build -t postgres-sample .
```

The PostgreSQL database is integral for the `inventory` microservice as it handles the persistence of data. Run the following command to start the PostgreSQL database, which runs the `postgres-sample` image in a Docker container and maps `5432` port from the container to your host machine:

[role='command']
```
docker run --name postgres-container --rm -p 5432:5432 -d postgres-sample
```

Retrieve the PostgreSQL container IP address by running the following command:

[role='command']
```
docker inspect -f "{{.NetworkSettings.IPAddress }}" postgres-container
```

The command returns the PostgreSQL container IP address:

[role="no_copy"]
----
172.17.0.2
----

Now, navigate to the `start` directory to begin.

ifdef::cloud-hosted[]
```bash
cd /home/project/guide-testcontainers/start
```
endif::[]

The Liberty Maven plug-in includes a `devc` goal that simplifies developing your application in a container by starting https://openliberty.io/docs/latest/development-mode.html#_container_support_for_dev_mode[dev mode^] with container support. This goal builds a Docker image, mounts the required directories, binds the required ports, and then runs the application inside of a container. Dev mode also listens for any changes in the application source code or configuration and rebuilds the image and restarts the container as necessary.

ifdef::cloud-hosted[]
In this IBM Cloud environment, you need to pre-create the ***logs*** directory by running the following commands:

```bash
mkdir -p /home/project/guide-testcontainers/start/target/liberty/wlp/usr/servers/defaultServer/logs
chmod 777 /home/project/guide-testcontainers/start/target/liberty/wlp/usr/servers/defaultServer/logs
```
endif::[]

Build and run the container by running the `devc` goal with the PostgreSQL container IP address. If your PostgreSQL container IP address is not `172.17.0.2`, replace the command with the right IP address.

[role='command']
```
mvn liberty:devc -DcontainerRunOpts="-e DB_HOSTNAME=172.17.0.2" -DserverStartTimeout=240
```

Wait a moment for dev mode to start. Some error messages are expected as a result of building the docker image. Although these messages are included on the standard error stream, in this case they are not errors, just logs of the docker build progress. After you see the following message, your Liberty instance is ready in dev mode:

[role="no_copy"]
----
**************************************************************
* Liberty is running in dev mode.
* ...
* Container network information:
* Container name: [ liberty-dev ]
* IP address [ 172.17.0.2 ] on container network [ bridge ]
* ...
----

ifndef::cloud-hosted[]
Dev mode holds your command-line session to listen for file changes. Open another command-line session to continue, or open the project in your editor.

Point your browser to the http://localhost:9080/openapi/ui URL to try out the `inventory` microservice manually. This interface provides a convenient visual way to interact with the APIs and test out their functionalities.
endif::[]

ifdef::cloud-hosted[]
Dev mode holds your command-line session to listen for file changes.

Click the following button to try out the ***inventory*** microservice manually by visiting the ***/openapi/ui*** endpoint. This interface provides a convenient visual way to interact with the APIs and test out their functionalities:

::startApplication{port="9080" display="external" name="Visit OpenAPI UI" route="/openapi/ui"}

Open another command-line session to continue.
endif::[]

=== Building a REST test client

The REST test client is responsible for sending HTTP requests to an application and handling the responses. It enables accurate verification of the application's behavior by ensuring that it responds correctly to various scenarios and conditions. Using a REST client for testing ensures reliable interaction with the `inventory` microservice across various deployment environments: local processes, Docker containers, or containers through Testcontainers.

Begin by creating a REST test client interface for the `inventory` microservice.

[role="code_command hotspot file=0" ,subs="quotes"]
----
#Create the `SystemResourceClient` class.#
`src/test/java/it/io/openliberty/guides/inventory/SystemResourceClient.java`
----

// file 0
SystemResourceClient.java
[source, java, linenums, role="code_column hide_tags=copyright"]
----
include::finish/src/test/java/it/io/openliberty/guides/inventory/SystemResourceClient.java[]
----

The [hotspot file=0]`SystemResourceClient` interface declares the [hotspot=listContents file=0]`listContents()`, [hotspot=getSystem file=0]`getSystem()`, [hotspot=addSystem file=0]`addSystem()`, [hotspot=updateSystem file=0]`updateSystem()`, and [hotspot=removeSystem file=0]`removeSystem()` methods for accessing the corresponding endpoints within the `inventory` microservice.

Next, create the `SystemData` data model for testing.

[role="code_command hotspot file=1" ,subs="quotes"]
----
#Create the `SystemData` class.#
`src/test/java/it/io/openliberty/guides/inventory/SystemData.java`
----

// file 1
SystemData.java
[source, java, linenums, role="code_column hide_tags=copyright"]
----
include::finish/src/test/java/it/io/openliberty/guides/inventory/SystemData.java[]
----

The [hotspot file=1]`SystemData` class contains the ID, hostname, operating system name, Java version, and heap size properties. The various [hotspot=getMethods file=1]`get` and [hotspot=setMethods file=1]`set` methods within this class enable you to view and edit the properties of each system in the inventory.

=== Building a test container for Open Liberty

Next, create a custom class that extends Testcontainers' generic container to define specific configurations that suit your application's requirements.

Define a custom `LibertyContainer` class, which provides a framework to start and access a containerized version of the Open Liberty application for testing.

[role="code_command hotspot file=0" ,subs="quotes"]
----
#Create the `LibertyContainer` class.#
`src/test/java/it/io/openliberty/guides/inventory/LibertyContainer.java`
----

// file 0
LibertyContainer.java
[source, java, linenums, role="code_column hide_tags=copyright"]
----
include::finish/src/test/java/it/io/openliberty/guides/inventory/LibertyContainer.java[]
----

The [hotspot file=0]`LibertyContainer` class extends the [hotspot=GenericContainer file=0]`GenericContainer` class from Testcontainers to create a custom container configuration specific to the Open Liberty application.

The [hotspot=addExposedPorts1 hotspot=addExposedPorts2 file=0]`addExposedPorts()` method exposes specified ports from the container's perspective, allowing test clients to communicate with services running inside the container. To avoid any port conflicts, Testcontainers assigns random host ports to these exposed container ports.

By default, the [hotspot=waitingFor file=0]`Wait.forLogMessage()` method directs `LibertyContainer` to wait for the specific `CWWKF0011I` log message that indicates the Liberty instance has started successfully.

The [hotspot=getBaseURL file=0]`getBaseURL()` method contructs the base URL to access the container.

For more information about Testcontainers APIs and its functionality, refer to the https://javadoc.io/doc/org.testcontainers/testcontainers/latest/index.html[Testcontainers JavaDocs^].

=== Building test cases

Next, write tests that use the `SystemResourceClient` REST client and Testcontainers integration.

[role="code_command hotspot file=0" ,subs="quotes"]
----
#Create the `SystemResourceIT` class.#
`src/test/java/it/io/openliberty/guides/inventory/SystemResourceIT.java`
----

// file 0
SystemResourceIT.java
[source, java, linenums, role="code_column hide_tags=copyright"]
----
include::finish/src/test/java/it/io/openliberty/guides/inventory/SystemResourceIT.java[]
----

// file 1
LibertyContainer.java
[source, java, linenums, role="code_column hide_tags=copyright"]
----
include::finish/src/test/java/it/io/openliberty/guides/inventory/LibertyContainer.java[]
----

// file 2
postgres/Dockerfile
[source, Dockerfile, linenums, role="code_column"]
----
include::postgres/Dockerfile[]
----

// file 3
Dockerfile
[source, Dockerfile, linenums, role="code_column"]
----
include::finish/Dockerfile[]
----

Construct the [hotspot=postgresImage file=0]`postgresImage` and [hotspot=invImage file=0]`invImage` using the `ImageFromDockerfile` class, which allows Testcontainers to build Docker images from a Dockerfile during the test runtime. For these instances, the provided Dockerfiles at the specified paths [hotspot file=2]`../postgres/Dockerfile` and [hotspot file=3]`./Dockerfile` are used to generate the respective `postgres-sample` and `inventory:1.0-SNAPSHOT` images.

Use [hotspot=GenericContainer file=0]`GenericContainer` class to create the [hotspot=postgresContainer file=0]`postgresContainer` test container to start up the `postgres-sample` Docker image, and use the [hotspot=LibertyContainer file=0]`LibertyContainer` custom class to create the [hotspot=inventoryContainer file=0]`inventoryContainer` test container to start up the `inventory:1.0-SNAPSHOT` Docker image.

As containers are isolated by default, placing both the [hotspot=LibertyContainer file=0]`LibertyContainer` and the [hotspot=postgresContainer file=0]`postgresContainer` on the same [hotspot=network1 hotspot=network2 hotspot=network3 file=0]`network` allows them to communicate by using the hostname `localhost` and the internal port `5432`, bypassing the need for an externally mapped port.

The [hotspot=waitingFor file=0]`waitingFor()` method here overrides the [hotspot=waitingFor file=1]`waitingFor()` method from [hotspot file=1]`LibertyContainer`. Given that the `inventory` service depends on a database service, ensuring that readiness involves more than just the microservice itself. To address this, the `inventoryContainer` readiness is determined by checking the [hotspot=waitingFor file=0]`/health/ready` health readiness check API, which reflects both the application and database service states. For different container readiness check customizations, see to the https://www.testcontainers.org/features/startup_and_waits/[official Testcontainers documentation^].

The [hotspot=getLogger1 hotspot=getLogger2 file=0]`LoggerFactory.getLogger()` and [hotspot=withLogConsumer1 hotspot=withLogConsumer2 file=0]`withLogConsumer(new Slf4jLogConsumer(Logger))` methods integrate container logs with the test logs by piping the container output to the specified logger.

The [hotspot=createRestClient file=0]`createRestClient()` method creates a REST client instance with the `SystemResourceClient` interface.

The [hotspot=setup file=0]`setup()` method prepares the test environment. It checks whether the test is running in dev mode or there is a local running Liberty instance, by using the [hotspot=isServiceRunning file=0]`isServiceRunning()` helper. In the case of no running Liberty instance, the test starts the [hotspot=postgresContainerStart file=0]`postgresContainer` and [hotspot=inventoryContainerStart file=0]`inventoryContainer` test containers. Otherwise, it ensures that the Postgres database is running locally.

The [hotspot=testAddSystem file=0]`testAddSystem()` verifies the [hotspot=addSystem file=0]`addSystem` and [hotspot=listContents file=0]`listContents` endpoints.

The [hotspot=testUpdateSystem file=0]`testUpdateSystem()` verifies the [hotspot=updateSystem file=0]`updateSystem` and [hotspot=getSystem file=0]`getSystem` endpoints.

The [hotspot=testRemoveSystem file=0]`testRemoveSystem()` verifies the [hotspot=removeSystem file=0]`removeSystem` endpoint.

After the tests are executed, the [hotspot=tearDown file=0]`tearDown()` method stops the containers and closes the network.

=== Setting up logs

Having reliable logs is essential for efficient debugging, as they provide detailed insights into the test execution flow and help pinpoint issues during test failures. Testcontainers' built-in `Slf4jLogConsumer` enables integration of container output directly with the JUnit process, enhancing log analysis and simplifying test creation and debugging.

[role="code_command hotspot file=0" ,subs="quotes"]
----
#Create the `log4j.properties` file.#
`src/test/resources/log4j.properties`
----

// file 0
log4j.properties
[source, properties, linenums, role='code_column']
----
include::finish/src/test/resources/log4j.properties[]
----

The [hotspot file=0]`log4j.properties` file configures the root logger, appenders, and layouts for console output. It sets the logging level to `DEBUG` for the [hotspot=package file=0]`it.io.openliberty.guides.inventory` package. This level provides detailed logging information for the specified package, which can be helpful for debugging and understanding test behavior.

=== Configuring the Maven project

Next, prepare your Maven project for test execution by adding the necessary dependencies for Testcontainers and logging, setting up Maven to copy the PostgreSQL JDBC driver during the build phase, and configuring the Liberty Maven Plugin to handle PostgreSQL dependency.

[role="code_command hotspot file=0" ,subs="quotes"]
----
#Replace the `pom.xml` file.#
`pom.xml`
----

// file 0
pom.xml
[source, XML, linenums, role='code_column']
----
include::finish/pom.xml[]
----

Add the required `dependency` for Testcontainers and Log4J libraries with `test` scope. The [hotspot=testcontainers file=0]`testcontainers` dependency offers a general-purpose API for managing container-based test environments. The [hotspot=slf4j file=0]`slf4j-reload4j` and [hotspot=slf4j-api file=0]`slf4j-api` dependencies enable the Simple Logging Facade for Java (SLF4J) API for trace logging during test execution and facilitates debugging and test performance tracking.

Also, add and configure the [hotspot=failsafe file=0]`maven-failsafe-plugin` plugin, so that the integration test can be run by the `mvn verify` command.

When you started Open Liberty in dev mode, all the changes were automatically picked up. You can run the tests by pressing the `enter/return` key from the command-line session where you started dev mode. You see the following output:

[role="no_copy"]
----
-------------------------------------------------------
T E S T S
-------------------------------------------------------
Running it.io.openliberty.guides.inventory.SystemResourceIT
it.io.openliberty.guides.inventory.SystemResourceIT - Testing by dev mode or local Liberty...
...
Tests run: 3, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 3.873 s - in it.io.openliberty.guides.inventory.SystemResourceIT

Results:

Tests run: 3, Failures: 0, Errors: 0, Skipped: 0
----

// =================================================================================================
// Running tests in a CI/CD pipeline
// =================================================================================================
== Running tests in a CI/CD pipeline

Running tests in dev mode is useful for local development, but there may be times when you want to test your application in other scenarios, such as in a CI/CD pipeline. For these cases, you can use Testcontainers to run tests against a running Open Liberty instance in a controlled, self-contained environment, ensuring that your tests run consistently regardless of the deployment context.

To test outside of dev mode, exit dev mode by pressing `CTRL+C` in the command-line session where you ran the Liberty.

Also, run the following commands to stop the PostgreSQL container that was started in the previous section:

[role='command']
```
docker stop postgres-container
```

Now, use the following Maven goal to run the tests from a cold start outside of dev mode:

[.tab_link.windows_link]
`*WINDOWS*`
[.tab_link.mac_link]
`*MAC*`
[.tab_link.linux_link]
`*LINUX*`
[.tab_content.windows_section.mac_section]
--
[role='command']
```
mvn clean verify
```
--
[.tab_content.linux_section]
--
[role='command']
```
export TESTCONTAINERS_RYUK_DISABLED=true
mvn clean verify
```
--

You see the following output:

[role="no_copy"]
----
-------------------------------------------------------
T E S T S
-------------------------------------------------------
Running it.io.openliberty.guides.inventory.SystemResourceIT
it.io.openliberty.guides.inventory.SystemResourceIT - Testing by using Testcontainers...
...
tc.postgres-sample:latest - Creating container for image: postgres-sample:latest
tc.postgres-sample:latest - Container postgres-sample:latest is starting: 7cf2e2c6a505f41877014d08b7688399b3abb9725550e882f1d33db8fa4cff5a
tc.postgres-sample:latest - Container postgres-sample:latest started in PT2.925405S
...
tc.inventory:1.0-SNAPSHOT - Creating container for image: inventory:1.0-SNAPSHOT
tc.inventory:1.0-SNAPSHOT - Container inventory:1.0-SNAPSHOT is starting: 432ac739f377abe957793f358bbb85cc916439283ed2336014cacb585f9992b8
tc.inventory:1.0-SNAPSHOT - Container inventory:1.0-SNAPSHOT started in PT25.784899S
...

Tests run: 3, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 12.208 s - in it.io.openliberty.guides.inventory.SystemResourceIT

Results:

Tests run: 3, Failures: 0, Errors: 0, Skipped: 0
----

Notice that the test initiates a new Docker container each for the PostgreSQL database and the `inventory` microservice, resulting in a longer test runtime. Despite this, cold start testing benefits from a clean instance per run and ensures consistent results. These tests also automatically hook into existing build pipelines that are set up to run the `integration-test` phase.

// =================================================================================================
// Great work! You're done!
// =================================================================================================
== Great work! You're done!

You just tested your microservices with multiple Docker containers using Testcontainers.

include::{common-includes}/attribution.adoc[subs="attributes"]