This document contains the hands-on lab modules for the Open Liberty Masterclass. It is intended to be used in conjunction with taught materials, however, feel free to give it a try, even if you're not in a Masterclass.
- Open Liberty Masterclass
- Table of Contents
- Before you begin
- The Application
- Module 1: Build
- Module 2: Feature-based Build
- Module 3: Application APIs
- Module 4: Server Configuration
- Module 5: Externalizing Configuration
- Module 6: Integration Testing
- Module 7: Docker
- Module 8: Testing in Containers
- Module 9: Support Licensing
- Conclusion
- A Java 8 JDK (e.g. https://adoptopenjdk.net/?variant=openjdk8&jvmVariant=openj9)
- Apache Maven (https://maven.apache.org/)
- A git client (https://git-scm.com/downloads)
- An editor with Java support (e.g. Eclipse, VS Code, IntelliJ)
- Docker
- Windows: Set up Docker for Windows as described at https://docs.docker.com/docker-for-windows/.
- Mac: Set up Docker for Mac as described at https://docs.docker.com/docker-for-mac/.
If you will be taking the Masterclass at a location with limited network bandwidth, it is recommended you do the following beforehand in order to populate your local .m2 repo and Docker cache.
git clone https://github.com/gcharters/open-liberty-masterclass.git
cd open-liberty-masterclass/finish/coffee-shop
mvn install
docker build -t masterclass:coffee-shop .
cd ../barista
mvn install
docker build -t masterclass:barista .
The application consists of two Microservices; coffee-shop
and barista
. The coffee-shop
service allows you to place an order and the barista
service services the making of the coffee.
^|
|| orderCoffee()
||
||
┌───|v────────┐ startCoffeeBrew() ┌─────────────┐
│ coffee-shop │---------------------->│ barista │
└─────────────┘<----------------------└─────────────┘
The completed code for the Masterclass is provided in the open-liberty-masterclass/finish
directory. To work through the Masterclass you will develop in the open-liberty-masterclass/start
directory.
Liberty has support for building and deploying applications using Maven and Gradle. The source and documentation for these plugins can be found here:
The Masterclass will make use of the liberty-maven-plugin
.
Take a look at the maven build file for the coffee-shop project: open-liberty-masterclass/start/barista/pom.xml
Go to the barista project:
cd open-liberty-masterclass/start/barista
Build and run the barista service:
mvn install liberty:run
Visit: http://localhost:9081/openapi/ui
This page is an OpenAPI UI that lets you try out the barista service.
Click on POST
and then Try it out
Under Example Value
specify:
{
"type": "ESPRESSO"
}
Click on Execute
Scroll down and you should see the server response code of 201
. This says that the barista request to make an ESPRESSO
was successfully Created
.
The liberty-maven-plugin
lets you specify which Liberty features you want to build against.
Take a look at the maven build file for the coffee-shop project: open-liberty-masterclass/start/coffee-shop/pom.xml
In order for the plugin to know what features are available, we need to tell it where to find the feature information. This is done with the following <dependencyManagement/>
section:
<dependencyManagement>
<dependencies>
<dependency>
<groupId>io.openliberty.features</groupId>
<artifactId>features-bom</artifactId>
<version>RELEASE</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
We can now specify which features we want to build against.
In the same coffee-shop/pom.xml
locate the <dependencies/>
section. You'll see, for example, that we're depending on jaxrs-2.1
because we're using this feature to implement the REST service:
<dependencies>
<!--Open Liberty features -->
<dependency>
<groupId>io.openliberty.features</groupId>
<artifactId>jaxrs-2.1</artifactId>
<type>esa</type>
<scope>provided</scope>
</dependency>
...
</dependencies>
Let's add add dependency on the MicroProfile OpenAPI
feature so we can try the coffee-shop
service out.
Add the following dependency to the coffee-shop/pom.xml
<dependency>
<groupId>io.openliberty.features</groupId>
<artifactId>mpOpenAPI-1.1</artifactId>
<type>esa</type>
<scope>provided</scope>
</dependency>
The above dependency will cause the feature to be installed during the build, but we also need to tell the server to load it at runtime.
Open the file open-liberty-masterclass/start/coffee-shop/src/main/liberty/config/server.xml
This file is the configuration for the coffee-shop
server.
Near the top of the file, you'll see the following <featureManager/>
entry:
<featureManager>
<feature>jaxrs-2.1</feature>
<feature>ejbLite-3.2</feature>
<feature>cdi-2.0</feature>
<feature>beanValidation-2.0</feature>
<feature>mpHealth-2.0</feature>
<feature>mpConfig-1.3</feature>
<feature>mpRestClient-1.3</feature>
<feature>jsonp-1.1</feature>
</featureManager>
This entry lists all the features to be loaded by the server. Add the following entry inside the <featureManager/>
element:
<feature>mpOpenAPI-1.1</feature>
Build and run the coffee-shop service:
mvn install liberty:run
Visit: http://localhost:9080/openapi/ui
As with the barista service, this is an Open API UI page that lets to try out the service API for the coffee-shop service.
For a full list of all the features available, see https://openliberty.io/docs/ref/feature/.
Open Liberty has support for many standard APIs out of the box, including all the latest Java EE 8 APIs and the latest MicroProfile APIs. To lead in the delivery of new APIs, a new version of Liberty is released every 4 weeks and aims to provide MicroProfile implementations soon after they are finalized.
As we've seen, to use a new feature, we need to add them to the build. There is no need to add a dependency on the APIs for the feature because each feature depends on the APIs. That means during build, the API dependencies are automatically added from maven central.
For example, take a look at: https://search.maven.org/artifact/io.openliberty.features/mpMetrics-2.0/19.0.0.8/esa
You'll see in the XML on the left that this feature depends on:
<dependency>
<groupId>io.openliberty.features</groupId>
<artifactId>com.ibm.websphere.appserver.org.eclipse.microprofile.metrics-2.0</artifactId>
<version>19.0.0.8</version>
<type>esa</type>
</dependency>
Which depends on the Metrics API from Eclipse MicroProfile:
<dependency>
<groupId>org.eclipse.microprofile.metrics</groupId>
<artifactId>microprofile-metrics-api</artifactId>
<version>2.0.0</version>
</dependency>
And so during build, this API will be added for you.
We're now going to add Metrics to the coffee-shop
. Edit the open-liberty-masterclass/start/coffee-shop/pom.xml
file and add the following dependency:
<dependency>
<groupId>io.openliberty.features</groupId>
<artifactId>mpMetrics-2.0</artifactId>
<type>esa</type>
</dependency>
Build the project:
mvn install
You should see that during the build, the following features are installed, and include mpMetrics-2.0:
[INFO] [AUDIT ] CWWKF0012I: The server installed the following features: [beanValidation-2.0, cdi-2.0, distributedMap-1.0, ejbLite-3.2, el-3.0, jaxrs-2.1, jaxrsClient-2.1, jndi-1.0, json-1.0, jsonp-1.1, mpConfig-1.3, mpHealth-2.0, mpMetrics-2.0, mpOpenAPI-1.1, mpRestClient-1.3, servlet-4.0, ssl-1.0].
Now we have the API available, we can update the application to include a metric which will count the number of times a coffee order is requested. In the file open-liberty-masterclass/start/coffee-shop/src/main/java/com/sebastian_daschner/coffee_shop/boundary/OrdersResource.java
, add the following @Counted
annotation to the orderCoffee
method:
@POST
@Counted(name="order", displayName="Order count", description="Number of times orders requested.")
public Response orderCoffee(@Valid @NotNull CoffeeOrder order) {
...
}
You'll also need to add the following package import:
import org.eclipse.microprofile.metrics.annotation.Counted;
Rebuild the project:
mvn install
In the previous module you added the mpMetrics-2.0
feature to the Liberty build. This makes the feature available for use by the Liberty runtime, but as we saw with the mpOpenAPI
feature loading the feature at runtime is a separate explicit choice.
Open the file open-liberty-masterclass/start/coffee-shop/src/main/liberty/config/server.xml
Near the top of the file, you'll see the following <featureManager/>
entry:
<featureManager>
<feature>jaxrs-2.1</feature>
<feature>ejbLite-3.2</feature>
<feature>cdi-2.0</feature>
<feature>beanValidation-2.0</feature>
<feature>mpHealth-2.0</feature>
<feature>mpConfig-1.3</feature>
<feature>mpRestClient-1.3</feature>
<feature>jsonp-1.1</feature>
<feature>mpOpenAPI-2.0</feature>
</featureManager>
Add the following inside the <featureManager/>
element to include the mpMetrics-2.0
feature:
<feature>mpMetrics-2.0</feature>
In the open-liberty-masterclass/start/coffee-shop
directory, build the updated application and start the server:
mvn install liberty:run
You should now see a message for a new metrics endpoint that looks like:
[INFO] [AUDIT ] CWWKT0016I: Web application available (default_host): http://localhost:9080/metrics/
Open the metrics endpoint in your browser. You should see a message like this:
Error 403: Resource must be accessed with a secure connection try again using an HTTPS connection.
If you take a look at the server output, you should see the following error:
[INFO] [ERROR ] CWWKS9113E: The SSL port is not active. The incoming http request cannot be redirected to a secure port. Check the server.xml file for configuration errors. The https port may be disabled. The keyStore element may be missing or incorrectly specified. The SSL feature may not be enabled.
It's one thing to configure the server to load a feature, but many Liberty features require additional configuration. The complete set of Liberty features and their configuration can be found here: https://openliberty.io/docs/ref/config/.
The error message suggests we need to add a keyStore
and one route to solve this would be to add a keyStore
and user registry (e.g. a basicRegistry
for test purposes). However, if we take a look at the configuration for mpMetrics (https://openliberty.io/docs/ref/config/#mpMetrics.html) we can see that it has an option to turn the metrics endpoint authentication off.
Add the following to the open-liberty-masterclass/start/coffee-shop/src/main/liberty/config/server.xml
<mpMetrics authentication="false" />
Rebuild, restart the server and visit the metrics endpoint, you should see a number of metrics automatically generated by the JVM:
TYPE base:classloader_total_loaded_class_count counter
# HELP base:classloader_total_loaded_class_count Displays the total number of classes that have been loaded since the Java virtual machine has started execution.
base:classloader_total_loaded_class_count 10616
...
This doesn't contain the metrics you added because the service hasn't been called and so no application metrics have been recorded. Use the OpenAPI UI (http://localhost:9080/openapi/ui/) to send a few requests to the service.
As with the barista
service, you'll need to specify the following payload for the POST
request:
{
"type": "ESPRESSO"
}
Reload the metrics page and at the bottom of the metrics results you should see:
...
# TYPE application:com_sebastian_daschner_coffee_shop_boundary_orders_resource_order counter
# HELP application:com_sebastian_daschner_coffee_shop_boundary_orders_resource_order Number of times orders requested.
application:com_sebastian_daschner_coffee_shop_boundary_orders_resource_order 3
If you're familiar with the concept of 12-factor applications (see http://12factor.net) you'll know that factor III states that an applications configuration should be stored in the environment. Config here is referring to things which vary between development, staging and production. In doing so you can build the deployment artefact once and deploy it in the different environments unchanged.
Liberty lets your application pick up configuration from a number of sources, such as environment variables, bootstrap.properties and Kubernetes configuration.
Bootstrap.properties lets you provide simple configuration values to substitute in the server configuration and also to use within the application. The following example replaces the hard-coded base URL the coffee-shop
service uses to talk to the barista
service, as well as the ports it exposes.
In the open-liberty-masterclass/start/coffee-shop/pom.xml
file, in the existing <properties/>
element, add the following port and url values:
<properties>
...
<testServerHttpPort>9080</testServerHttpPort>
<testServerHttpsPort>9443</testServerHttpsPort>
<baristaBaseURL>http://localhost:9081</baristaBaseURL>
...
</properties>
This <properties/>
element is where the property values are set that can then be re-used within the maven project.
In the <bootstrapProperties/>
section of the liberty-maven-plugin
configuration, add the following:
<bootstrapProperties>
...
<env.default_http_port>${testServerHttpPort}</env.default_http_port>
<env.default_https_port>${testServerHttpsPort}</env.default_https_port>
<default_barista_base_url>${baristaBaseURL}</default_barista_base_url>
</bootstrapProperties>
The above takes the properties we defined in the maven project and passes them to Liberty as bootstrap properties.
Note, we're using the env.
prefix because in the Docker modules of this Masterclass you will set these through environment variables. Note, also the names use underscores (_
) so they can be passed as environment variables.
Build the project:
mvn install
The liberty-maven-plugin
generated the following file target/liberty/wlp/usr/servers/defaultServer/bootstrap.properties
which contains the configuration that will be loaded and applied to the server configuration. If you view the file you'll see the values you specified:
# Generated by liberty-maven-plugin
default_barista_base_url=http://localhost:9081
env.default_http_port=9080
env.default_https_port=9443
war.name=coffee-shop.war
We now need to change the server configuration to use these values. In the open-liberty-masterclass/start/coffee-shop/src/main/liberty/config/server.xml
file, change this line:
<httpEndpoint host="*" httpPort="9080" httpsPort="9443" id="defaultHttpEndpoint"/>
to
<httpEndpoint host="*" httpPort="${env.default_http_port}" httpsPort="${env.default_https_port}" id="defaultHttpEndpoint"/>
Next we'll use the default_barista_base_url
in the code to avoid hard-coding the location of the barista
service.
Edit the file open-liberty-masterclass/start/coffee-shop/src/main/java/com/sebastian_daschner/coffee_shop/control/Barista.java
Change:
String baristaBaseURL = "http://localhost:9081";
To:
@Inject
@ConfigProperty(name="default_barista_base_url")
String baristaBaseURL;
You'll also need to add the following imports:
import javax.inject.Inject;
import org.eclipse.microprofile.config.inject.ConfigProperty;
This is using the MicroProfile Config specification to inject the configuration value. Configuration can come from a number of sources, including bootstrap.properties
.
We also need to make the same changes to the CoffeeShopHealth of the coffee-shop
service. Edit the file: open-liberty-masterclass/start/coffee-shop/src/main/java/com/sebastian_daschner/coffee_shop/boundary/CoffeeShopHealth.java
Change:
String baristaBaseURL = "http://localhost:9081";
To:
@Inject
@ConfigProperty(name="default_barista_base_url")
String baristaBaseURL;
Add the following imports:
import javax.inject.Inject;
import org.eclipse.microprofile.config.inject.ConfigProperty;
For more information on MicroProfile Config see https://openliberty.io/guides/microprofile-config.html.
Rebuild the code, start the coffee-shop
and barista
servers and try out the endpoint through the Open API UI. You can also try out the health endpoint at http://localhost:9080/health
.
Tests are essential for developing maintainable code. Developing your application using bean-based component models like CDI makes your code easily unit-testable. Integration Tests are a little more challenging. In this section you'll add a barista
service integration test using the maven-failsafe-plugin
. During the build, the Liberty server will be started along with the barista
application deployed, the test will be run and then the server will be stopped. The starting and stopping of the Liberty server is configured by the Liberty parent pom (see https://search.maven.org/artifact/net.wasdev.wlp.maven.parent/liberty-maven-app-parent/2.6.3/pom), which is configured as the parent of the Masterclass poms.
Because we're going to be testing a REST POST
request, we need JAX-RS client support and also support for serializing json
into the request. We also need junit
for writing the test. Add these dependencies to the open-liberty-masterclass/start/barista/pom.xml
:
<!-- Test dependencies -->
<dependency>
<groupId>org.junit.jupiter</groupId>
<artifactId>junit-jupiter</artifactId>
<version>5.4.2</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-rs-mp-client</artifactId>
<version>3.3.0</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.jaxrs</groupId>
<artifactId>jackson-jaxrs-json-provider</artifactId>
<version>2.9.3</version>
<scope>test</scope>
</dependency>
Note, the later Testing in Containers
module requires the JUnit 5 Jupiter API so we're using the same API here.
Note the <scope/>
of the dependencies is set to test
because we only want the dependencies to be used during testing.
Next add maven-failsafe-plugin
configuration at the end of the <plugins/>
section:
<plugins>
...
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-failsafe-plugin</artifactId>
<version>3.0.0-M1</version>
<executions>
<execution>
<phase>integration-test</phase>
<id>integration-test</id>
<goals>
<goal>integration-test</goal>
</goals>
<configuration>
<trimStackTrace>false</trimStackTrace>
<!-- Starting in version 3.0 of the liberty-maven-plugin these will be set automatically -->
<systemProperties>
<liberty.test.port>${testServerHttpPort}</liberty.test.port>
</systemProperties>
</configuration>
</execution>
<execution>
<id>verify-results</id>
<goals>
<goal>verify</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
Note, this configuration makes the port of the server available to the test as a system property called liberty.test.port
.
Finally, add the test code. Create a file called, open-liberty-masterclass/start/barista/src/test/java/com/sebastian-daschner/barista/it/BaristaIT.java
and add the following:
package com.sebastian_daschner.barista.it;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;
import org.junit.BeforeClass;
import javax.inject.Inject;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.BeforeAll;;
import javax.ws.rs.client.Client;
import javax.ws.rs.client.ClientBuilder;
import javax.ws.rs.client.Entity;
import javax.ws.rs.client.WebTarget;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.MediaType;
import com.fasterxml.jackson.jaxrs.json.JacksonJsonProvider;
import com.sebastian_daschner.barista.boundary.BrewsResource;
import com.sebastian_daschner.barista.entity.CoffeeBrew;
import com.sebastian_daschner.barista.entity.CoffeeType;
public class BaristaIT {
private static String URL;
@BeforeAll
public static void init() {
String port = System.getProperty("liberty.test.port");
URL = "http://localhost:" + port + "/barista/resources/brews";
}
@Test
public void testService() throws Exception {
Client client = null;
WebTarget target = null;
try {
client = ClientBuilder.newClient().register(JacksonJsonProvider.class);
target = client.target(URL);
} catch (Exception e) {
client.close();
throw e;
}
CoffeeBrew brew = new CoffeeBrew();
brew.setType(CoffeeType.POUR_OVER);
Response response = target.request(MediaType.APPLICATION_JSON).post(Entity.json(brew));
try {
if (response == null) {
assertNotNull("GreetingService response must not be NULL", response);
} else {
assertEquals("Response must be 200 OK", 200, response.getStatus());
}
} finally {
response.close();
}
}
}
This test sends a json
request to the barista
service and checks for a 200 OK
response.
Re-build and run the tests:
mvn install
In the output of the build, you should see:
-------------------------------------------------------
T E S T S
-------------------------------------------------------
Running com.sebastian_daschner.barista.it.BaristaIT
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 1.365 sec - in com.sebastian_daschner.barista.it.BaristaIT
Results :
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0
We're now going to dockerize the two services and show how we can override the defaults to re-wire the two services. We're going to use a Docker user-defined network (see https://docs.docker.com/network/network-tutorial-standalone/#use-user-defined-bridge-networks) because we'll be running them on the same host and it keeps things simple. For real-world production deployments you would use a Kubernetes environment, such as IBM Cloud Private or the IBM Cloud Kubernetes Service.
Take a look at the open-liberty-masterclass/start/coffee-shop/Dockerfile
:
FROM open-liberty:kernel-java8-ibm as server-setup
COPY /target/defaultServer.zip /config/
USER 0
RUN apt-get update \
&& apt-get install -y --no-install-recommends unzip \
&& unzip /config/defaultServer.zip \
&& mv /wlp/usr/servers/defaultServer/* /config/ \
&& rm -rf /config/wlp \
&& rm -rf /config/defaultServer.zip \
&& apt-get remove -y unzip
USER 1001
RUN rm /opt/ol/wlp/usr/servers/defaultServer/bootstrap.properties
FROM open-liberty
LABEL maintainer="Graham Charters" vendor="IBM" github="https://github.com/WASdev/ci.maven"
COPY --from=server-setup /config/ /config/
EXPOSE 9080 9443
This Dockerfile uses Docker build stages. The first stage gets all the application and server configuration contents into the right location and the second builds the actual final image. Using stages means any temporary files from the first stage don't end up in the final image, so it's smaller.
The FROM
statement is building this image using the Open Liberty kernel image (see https://hub.docker.com/_/open-liberty/ for the available images). The second RUN
removes the bootstrap.properties
file to avoid accidentally using it and avoid conflicts with the environment variables we will pass in later through Docker. The EXPOSE
makes the two server ports available outside the container.
Let's build the docker image. In the open-liberty-masterclass/start/coffee-shop
directory, run (note the period (.
) at the end of the line is important):
docker build -t masterclass:coffee-shop .
In the open-liberty-masterclass/start/barista
directory, run (note the period (.
) at the end of the line is important):
docker build -t masterclass:barista .
Next, create the user-defined bridge network:
docker network create --driver bridge masterclass-net
You can now run the two Docker containers and get them to join the same bridge network. Providing names to the containers makes those names available for DNS resolution within the bridge network so there's no need to use ip addresses.
Run the barista
container:
docker run --network=masterclass-net --name=barista masterclass:barista
Note, we don't need map the barista
service ports outside the container because the bridge network gives access to the other containers on the same network.
Next, we're going to run the coffee-shop
container. For it to work we'll need to provide new values for ports and the location of the barista service. Run the coffee-shop
container
docker run -p 9080:9080 -p 9445:9443 --network=masterclass-net --name=coffee-shop -e default_barista_base_url='http://barista:9081' -e default_http_port=9080 -e default_https_port=9443 masterclass:coffee-shop
You can take a look at the bridge network using:
docker network inspect masterclass-net
You'll see something like:
[
{
"Name": "masterclass-net",
...
"IPAM": {
"Driver": "default",
"Options": {},
"Config": [
{
"Subnet": "172.19.0.0/16",
"Gateway": "172.19.0.1"
}
]
},
...
"Containers": {
"0fc740d52f2ed8dfdb04127fe3e49366dcbeb7924fee6b0cbf6f891c0909b0e8": {
"Name": "coffee-shop",
"EndpointID": "157d697fb4bff2722d654c68e3a5e5fe7554a91e860213d22362cd7cc074fc8f",
"MacAddress": "02:42:ac:13:00:02",
"IPv4Address": "172.19.0.2/16",
"IPv6Address": ""
},
"2b78ebf13596147042c8f2f5bd3171ca1c6f77241f419472010ddc2f28fd7a0c": {
"Name": "barista",
"EndpointID": "c93163547eb7e3c2c84dd0f72beb77127cfc319b6d9d7f6d9d99e17b85ff6d30",
"MacAddress": "02:42:ac:13:00:03",
"IPv4Address": "172.19.0.3/16",
"IPv6Address": ""
}
},
"Options": {},
"Labels": {}
}
]
If you need to remove a container, use:
docker container rm <container name>
You should now be able to load the coffee-shop
service's Open API page and call the service. Give it a try.
The above works fine, but still has a metrics endpoint with authentication turned off. We'll now show how configDropins/overrides
can be used to override existing, or add new, server configuration. For example, this can be used to add server configuration in a production environment. The approach we're going to take is to use a Docker volume, but in a real-world scenario you would use Kubernetes ConfigMaps and secrets to include the production server configuration, security configuration and environment variables.
In fact, unlike what we have done here, the best practice is to build an image that does not contain any environment specific configuration (such as the unsecured endpoint in our example) and then add those things through external configuration in the development, staging and production environments. The goal is to ensure deployment of the image without configuration doesn't not cause undesirable results such as security vulnerabilities or talking to the wrong data sources.
Take a look at the file open-liberty-masterclass/start/coffee-shop/configDropins/overrides/metrics-prod.xml
:
<?xml version="1.0" encoding="UTF-8"?>
<server description="Coffee Shop Server">
<featureManager>
<feature>mpMetrics-2.0</feature>
</featureManager>
<mpMetrics authentication="true" />
<!--
Note, this configuration is for demo purposes
only and MUST NOT BE USED IN PRODUCTION AS IT
IS INSECURE. -->
<variable name="admin.password" value="change_it" />
<variable name="keystore.password" value="change_it" />
<quickStartSecurity userName="admin" userPassword="${admin.password}"/>
<keyStore id="defaultKeyStore" password="${keystore.password}"/>
</server>
You'll see that this turns metrics authentication on and sets up some simple security required for securing/accessing the metrics endpoint. Note, this configuration really is NOT FOR PRODUCTION, it's simply aiming to show how to override, or provide new, server configuration.
If you're on a unix-based OS, in the open-liberty-masterclass/start/coffee-shop
directory, run the coffee-shop
container:
docker run -p 9080:9080 -p 9445:9443 --network=masterclass-net --name=coffee-shop -e default_barista_base_url='http://barista:9081' -e default_http_port=9080 -e default_https_port=9443 -v $(pwd)/configDropins/overrides:/opt/ol/wlp/usr/servers/defaultServer/configDropins/overrides masterclass:coffee-shop
The above relies on pwd
to fill in the docker volume source path. If you're on Windows, replace $(pwd)
with the absolute path to the open-liberty-masterclass/start/coffee-shop
directory in the above command.
You should see the following message as the server is starting:
[AUDIT ] CWWKG0102I: Found conflicting settings for mpMetrics configuration.
Property authentication has conflicting values:
Value false is set in file:/opt/ol/wlp/usr/servers/defaultServer/server.xml.
Value true is set in file:/opt/ol/wlp/usr/servers/defaultServer/configDropins/overrides/metrics-prod.xml.
Property authentication will be set to true.
This shows that we have turned metrics authentication back on.
Access the metrics endpoint at: https://localhost:9445/metrics
You will see that the browser complains about the certificate. This is a self-signed certificate generated by Liberty for test purposes. Accept the exception (note, Firefox may not allow you to do this in which case you'll need to use a different browser). You'll be presented with a login prompt. Sign in with userid admin
and password change_it
(the values in the metrics-prod.xml
).
We saw in an earlier module, how to perform Integration Tests against the application running in the server. We then showed how to package the application and server and run them inside a Docker container. Assuming we're going to deploy our application in production inside Containers it would be a good idea to actually performs tests against that configuration. The more we can make our development and test environments the same as production, the less likely we are to encounter issues in production. MicroShed Testing (microshed.org) is a project that enables us to do just that.
Let's create a new Integration Test that will perform the same test, but inside a running container. In the Barista project, add the follow dependencies to the pom.xml
file in the <dependencies>
element:
<!-- For MicroShed Testing -->
<dependency>
<groupId>com.github.microshed.microshed-testing</groupId>
<artifactId>microshed-testing-liberty</artifactId>
<version>0.4.1-beta</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-log4j12</artifactId>
<version>1.7.26</version>
<scope>test</scope>
</dependency>
Add the following <repository>
element to the pom.xml
. This should be as a peer of the <properties>
element:
<repositories>
<!-- https://jitpack.io/#microshed/microshed-testing -->
<repository>
<id>jitpack.io</id>
<url>https://jitpack.io</url>
</repository>
</repositories>
The MicroShed Testing project is not released to Maven Central at the moment and so this entries tells maven about the repository from where it can be downloaded.
Create a new Integration Test called BaristaContainerIT.java
in the directory start/barista/src/test/java/com/sebastian_daschner/barista/it
and add the following code:
package com.sebastian_daschner.barista.it;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;
import javax.inject.Inject;
import javax.ws.rs.core.Response;
import org.junit.jupiter.api.Test;
import org.microshed.testing.jupiter.MicroShedTest;
import org.microshed.testing.testcontainers.MicroProfileApplication;
import org.testcontainers.junit.jupiter.Container;
import com.sebastian_daschner.barista.boundary.BrewsResource;
import com.sebastian_daschner.barista.entity.CoffeeBrew;
import com.sebastian_daschner.barista.entity.CoffeeType;
@MicroShedTest
public class BaristaContainerIT {
@Container
public static MicroProfileApplication app = new MicroProfileApplication()
.withAppContextRoot("/barista")
.withExposedPorts(9081, 9444)
.withReadinessPath("/health");
@Inject
public static BrewsResource brews;
@Test
public void testService() throws Exception {
CoffeeBrew brew = new CoffeeBrew();
brew.setType(CoffeeType.POUR_OVER);
Response response = brews.startCoffeeBrew(brew);
try {
if (response == null) {
assertNotNull("GreetingService response must not be NULL", response);
} else {
assertEquals("Response must be 200 OK", 200, response.getStatus());
}
} finally {
response.close();
}
}
}
You'll see that the class is marked as a MicroShed test with the @MicroShedTest
annotation.
The test also contains the following Container configuration:
@Container
public static MicroProfileApplication app = new MicroProfileApplication()
.withAppContextRoot("/barista")
.withExposedPorts(9081, 9444)
.withReadinessPath("/health");
You'll see that the unit test is like any other.
We need to configure log4j
in order to see the detailed progress of the MicroShed test. In the directory start/barista/src/test/resources/
create the file log4j.properties
and add the following configuration to it:
log4j.rootLogger=INFO, stdout
log4j.appender=org.apache.log4j.ConsoleAppender
log4j.appender.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%r %p %c %x - %m%n
log4j.logger.org.microshed=DEBUG
Build and run the test:
mvn install
You should see the following output:
[INFO] -------------------------------------------------------
[INFO] T E S T S
[INFO] -------------------------------------------------------
[INFO] Running com.sebastian_daschner.barista.it.BaristaContainerIT
0 INFO org.microshed.testing.jupiter.MicroShedTestExtension - Using ApplicationEnvironment class: org.microshed.testing.testcontainers.config.HollowTestcontainersConfiguration
70 INFO org.testcontainers.dockerclient.DockerClientProviderStrategy - Loaded org.testcontainers.dockerclient.UnixSocketClientProviderStrategy from ~/.testcontainers.properties, will try it first
710 INFO org.testcontainers.dockerclient.UnixSocketClientProviderStrategy - Accessing docker with local Unix socket
710 INFO org.testcontainers.dockerclient.DockerClientProviderStrategy - Found Docker environment with local Unix socket (unix:///var/run/docker.sock)
868 INFO org.testcontainers.DockerClientFactory - Docker host IP address is localhost
914 INFO org.testcontainers.DockerClientFactory - Connected to docker:
Server Version: 19.03.1
API Version: 1.40
Operating System: Docker Desktop
Total Memory: 1998 MB
1638 INFO org.testcontainers.utility.RegistryAuthLocator - Credential helper/store (docker-credential-desktop) does not have credentials for quay.io
2627 INFO org.testcontainers.DockerClientFactory - Ryuk started - will monitor and terminate Testcontainers containers on JVM exit
ℹ︎ Checking the system...
✔ Docker version should be at least 1.6.0
✔ Docker environment should have more than 2GB free disk space
2827 INFO org.microshed.testing.testcontainers.MicroProfileApplication - Discovered ServerAdapter: class org.testcontainers.containers.liberty.LibertyAdapter
2828 INFO org.microshed.testing.testcontainers.MicroProfileApplication - Using ServerAdapter: org.testcontainers.containers.liberty.LibertyAdapter
2834 DEBUG org.microshed.testing.testcontainers.config.TestcontainersConfiguration - No networks explicitly defined. Using shared network for all containers in class com.sebastian_daschner.barista.it.BaristaContainerIT
2842 INFO org.microshed.testing.testcontainers.config.HollowTestcontainersConfiguration - exposing port: 9081 for container alpine:3.5
2843 INFO org.microshed.testing.testcontainers.config.HollowTestcontainersConfiguration - exposing port: 9444 for container alpine:3.5
2844 INFO org.microshed.testing.testcontainers.config.TestcontainersConfiguration - Starting containers in parallel for class com.sebastian_daschner.barista.it.BaristaContainerIT
2845 INFO org.microshed.testing.testcontainers.config.TestcontainersConfiguration - java.util.concurrent.CompletableFuture@465232e9[Completed normally]
2848 INFO org.microshed.testing.testcontainers.config.TestcontainersConfiguration - All containers started in 3ms
2868 DEBUG org.microshed.testing.jaxrs.RestClientBuilder - no classes implementing Application found in pkg: com.sebastian_daschner.barista.boundary
2868 DEBUG org.microshed.testing.jaxrs.RestClientBuilder - checking in pkg: com.sebastian_daschner.barista
2873 DEBUG org.microshed.testing.jaxrs.RestClientBuilder - Using ApplicationPath of 'resources'
2874 INFO org.microshed.testing.jaxrs.RestClientBuilder - Building rest client for class com.sebastian_daschner.barista.boundary.BrewsResource with base path: http://localhost:9081/barista/resources and providers: [class org.microshed.testing.jaxrs.JsonBProvider]
3273 DEBUG org.microshed.testing.jupiter.MicroShedTestExtension - Injecting rest client for public static com.sebastian_daschner.barista.boundary.BrewsResource com.sebastian_daschner.barista.it.BaristaContainerIT.brews
3419 INFO org.microshed.testing.jaxrs.JsonBProvider - Sending data to server: {"type":"POUR_OVER"}
[INFO] Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 3.93 s - in com.sebastian_daschner.barista.it.BaristaContainerIT
[INFO] Running com.sebastian_daschner.barista.it.BaristaIT
[INFO] Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.271 s - in com.sebastian_daschner.barista.it.BaristaIT
[INFO]
[INFO] Results:
[INFO]
[INFO] Tests run: 2, Failures: 0, Errors: 0, Skipped: 0
[INFO]
[INFO]
Open Liberty is Open Source under the Eclipse Public License v1, as a result there is no fee to use in production. Community support is available via StackOverflow, Gitter, or the mail list, and bugs can be raised in github (https://github.com/openliberty/open-liberty). Commercial support from IBM is available for Open Liberty, you can find out more on the IBM Marketplace. The WebSphere Liberty product is built on Open Liberty, there is no migration required to use WebSphere Liberty, you simply point to WebSphere Liberty in your build. Users of WebSphere Liberty get support for the packaged Open Liberty function.
WebSphere Liberty is also available in Maven Central - see https://search.maven.org/search?q=g:com.ibm.websphere.appserver.runtime
You can use WebSphere Liberty for development even if you haven't purchased it, but if you have production entitlement you can easily change to use it, as follows:
In the open-liberty-masterclass/start/coffee-shop/pom.xml
change these two lines from:
<groupId>io.openliberty</groupId>
<artifactId>openliberty-kernel</artifactId>
To:
<groupId>com.ibm.websphere.appserver.runtime</groupId>
<artifactId>wlp-kernel</artifactId>
Rebuild and re-start the coffee-shop
service:
mvn install liberty:run
Try the service out using the Open API Web page and you should see the behavior is identical. Not surprising since the code is identical, from the same build, just built into WebSphere Liberty.
Thanks for trying the Open Liberty Masterclass. If you're interested in finding out more, please visit http://openliberty.io, and for more hands-on experience, why not try the Open Liberty Guides - http://openliberty.io/guides.