Contents
- What you’ll learn
- Getting started
- Creating a configuration profile for the dev environment
- Creating profile-specific configuration properties
- Creating profile-specific
microprofile-config.properties
configuration files - Creating a configuration profile for the test environment
- Next steps
- Nice work! Where to next?
Tags
Externalizing environment-specific microservice configuration for CI/CD
Learn how to create environment-specific configurations for microservices by using MicroProfile Config configuration profiles for easy management and portable deployments throughout the CI/CD lifecycle.
What you’ll learn
Managing configurations for microservices can be challenging, especially when configurations require adjustments across various stages of the software development and delivery lifecycle. The MicroProfile Config configuration profile feature, also known as the Config Profile, is a direct solution to this challenge. It simplifies the management of microservice configurations across diverse environments - from development to production and throughout the continuous integration/continuous delivery (CI/CD) pipeline. By externalizing and tailoring configuration properties to each environment, the CI/CD process becomes more seamless, so you can concentrate on perfecting your application code and capabilities.
You’ll learn how to provide environment-specific configurations by using the MicroProfile Config configuration profile feature. You’ll work with the MicroProfile Config API to create configuration profiles that use profile-specific configuration properties and configuration sources.
This guide builds on the Separating configuration from code in microservices guide and the Configuring microservices guide. If you are not familiar with externalizing the configuration of microservices, it will be helpful to read the External configuration of microservices document and complete the aforementioned guides before you proceed.
The application that you will work with is a query
service, which fetches information about the running JVM from a system
microservice. You’ll use configuration profiles to externalize and manage the configurations across the development, testing, and production environments.
Getting started
The fastest way to work through this guide is to clone the Git repository and use the projects that are provided inside:
git clone https://github.com/openliberty/guide-microprofile-config-profile.git
cd guide-microprofile-config-profile
The start
directory contains the starting project that you will build upon.
The finish
directory contains the finished project that you will build.
Before you begin, make sure you have all the necessary prerequisites.
Creating a configuration profile for the dev environment
The dev environment is used to test, experiment, debug, and refine your code, ensuring an application’s functional readiness before progressing to subsequent stages in a software development and delivery lifecycle.
Navigate to the start
directory to begin.
The starting Java project, which you can find in the start
directory, is a multi-module Maven project comprised of the system
and query
microservices. Each microservice is in its own corresponding directory, system
and query
.
The system
microservice contains the three Maven build profiles: dev
, test
, and prod
, in which the dev
profile is set as the default. Each build profile defines properties for a particular deployment configuration that the microservice uses.
The MicroProfile Config configuration profile feature supplies configurations for different environments when only a single profile is active. The active profile is set using the mp.config.profile
property. You can set it in any of the configuration sources and it is read once during application startup. When a profile is active, its associated configuration properties are used. For the query
service, the mp.config.profile
property is set to dev
in its Maven pom.xml
. This Liberty configuration variable indicates to the runtime that dev
is the active configuration profile.
When you run Open Liberty in dev mode, the dev mode listens for file changes and automatically recompiles and deploys your updates whenever you save a new change.
Open a command-line session and run the following commands to navigate to the system
directory and start the system
service in the dev
environment:
cd system
mvn liberty:dev
Open another command-line session and run the following commands to navigate to the query
directory and start the query
service in the dev
environment:
cd query
mvn liberty:dev
After you see the following message, your Liberty instance is ready in dev mode:
**************************************************
* Liberty is running in dev mode.
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.
In the dev environment, the dev
configuration profile is set in the system/pom.xml
file as the configuration profile to use for running the system
service. The system
service runs on HTTP port 9081
and HTTPS port 9444
using the context root system/dev
. It uses a basic user registry with username alice
and password alicepwd
for resource authorization. Note that the basicRegistry
element is a simple registry configuration for learning purposes. For more information on user registries, see the User registries documentation.
Point your browser to the http://localhost:9085/query/systems/localhost URL.
The query
service returns the message: {"fail":"Failed to reach the client localhost."}
. This is because the current query
service uses the default properties in the query/src/main/resources/META-INF/microprofile-config.properties
file to access the system
service.
For proper communication with the development system
service, the query
service uses properties in the dev
configuration profile.
There are two ways to define configuration properties that are associated with your configuration profile. The first is as individual configuration properties associated with a configuration profile that can be specified in any kind of MicroProfile configuration source. The second is through default microprofile-config.properties
configuration files embedded in your application that can be associated with different configuration profiles. The former allows for flexibility in defining profile-specific configuration properties in the best configuration sources for your needs while the latter enables default profiles of configuration properties to be provided in your application.
Creating profile-specific configuration properties
This approach involves directly associating individual configuration properties with a configuration profile. To define a configuration property for a particular config profile, use the %<config_profile_id>.<property_name>=<value>
syntax, where <config_profile_id>
is the unique identifier for the configuration profile and <property_name>
is the name of the property that you want to set.
query/src/main/resources/META-INF/microprofile-config.properties
Configure the %dev.*
properties in the microprofile-config.properties
file based on the values from the dev
profile of the system
service.
Because the active profile is set to dev
, each %dev.*
property overrides the value of the plain non-profile-specific property. For example, in this case, the %dev.system.httpsPort
property overrides the system.httpsPort
property and the value is resolved to 9444
.
Because you are running the query
service in dev mode, the changes that you made are automatically picked up.
Try out the application at the http://localhost:9085/query/systems/localhost URL. You can see the current OS and Java version in JSON format.
Creating profile-specific microprofile-config.properties
configuration files
Creating profile-specific microprofile-config.properties
configuration files is a structured way to provide and manage more extensive sets of default configurations. You can create a configuration file for each configuration profile in the META-INF
folder on the classpath of your application by using the microprofile-config-<config_profile_id>
naming convention, where <config_profile_id>
is the unique identifier for a configuration profile. After you create the file, you can add your configuration properties to it with the standard <property_name>=<value>
syntax.
query/src/main/resources/META-INF/microprofile-config-dev.properties
Define the system.*
properties in the microprofile-config-dev.properties
file based on the values from the dev
profile of the system
service.
query/src/main/resources/META-INF/microprofile-config.properties
Remove the %dev.*
properties from the microprofile-config.properties
file.
Because the active profile is set to dev
, any system.
properties specified in the microprofile-config-dev.properties
file take precedence over the system.
property values in the microprofile-config.properties
file.
Now, point your browser to the http://localhost:9085/query/systems/localhost URL to check out the application again. You can see the current OS and Java version in JSON format.
When you are done checking out the application in dev
environment, exit dev mode by pressing CTRL+C
in the command-line sessions where you ran the system
and query
services.
Creating a configuration profile for the test environment
In CI/CD, the test environment is where integration tests ensure the readiness and quality of an application. A good testing configuration not only ensures smooth operations but also aligns the environment closely with potential production settings.
query/src/main/resources/META-INF/microprofile-config-test.properties
Define the system.*
properties in the microprofile-config-test.properties
file based on the values from the test
profile of the system
service.
query/src/test/java/it/io/openliberty/guides/query/QueryEndpointIT.java
Implement endpoint tests to test the basic functionality of the query
microservice. If a test failure occurs, you might have introduced a bug into the code.
See the following descriptions of test cases:
-
testQuerySystem()
verifies the/query/systems/{hostname}
endpoint. -
testUnknownHost()
verifies that an unknown host or a host that does not expose their JVM system properties is correctly handled with a fail message.
Running the tests in the test environment
Now, navigate to the start
directory.
Test the application under the test
environment by running the following script that contains different Maven goals to build
, start
, test
, and stop
the services.
WINDOWS
MAC
LINUX
scripts\testApp.bat
If the tests pass, you see output similar to the following example:
-------------------------------------------------------
T E S T S
-------------------------------------------------------
Running it.io.openliberty.guides.system.SystemEndpointIT
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.539 s - in it.io.openliberty.guides.system.SystemEndpointIT
Results:
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0
...
-------------------------------------------------------
T E S T S
-------------------------------------------------------
Running it.io.openliberty.guides.query.QueryEndpointIT
Tests run: 2, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 1.706 s - in it.io.openliberty.guides.query.QueryEndpointIT
Results:
Tests run: 2, Failures: 0, Errors: 0, Skipped: 0
Next steps
Deploying the application to a Kubernetes environment using the Open Liberty Operator is an optional learning step in this guide.
To further explore deploying microservices using Kubernetes and the Open Liberty Operator, you can read the following guides:
A secure production environment is essential for application security. In the previous sections, you learned how to use the MicroProfile Config API to externalize credentials and other properties for accessing the system
service. This strategy makes the application more adaptable to different environments without the need to change code and rebuild your application.
In the this section, you’ll learn how to use Kubernetes secrets to provide the credentials and how to pass them to the query
service by using MicroProfile Config.
Deploying the application in the prod environment with Kubernetes
Before deploying, create the Dockerfile files for both system
and query
microservices. Then, build their .war
files and Docker images in the start
directory.
mvn -P prod clean package
docker build -t system:1.0-SNAPSHOT system/.
docker build -t query:1.0-SNAPSHOT query/.
The Maven clean
and package
goals can clean the target
directories and build the .war
application files from scratch. The microprofile-config-dev.properties
and microprofile-config-test.properties
files of the query
microservice are excluded from the prod
build. The default microprofile-config.properties
file is automatically applied.
The Docker build
command packages the .war
files of the system
and query
microservices with their default configuration into your Docker images.
After building the images, you can create a Kubernetes secret for storing sensitive data such as credentials.
kubectl create secret generic sys-app-credentials \
--from-literal username=[username] \
--from-literal password=[password]
For more information about managing secrets, see the Managing Secrets using kubectl documentation.
Finally, write up the deploy.yaml
deployment file to configure the deployment of the system
and query
microservices by using the Open Liberty Operator. The sys-app-credentials
Kubernetes secrets set the environment variables DEFAULT_USERNAME
and DEFAULT_PASSWORD
for the system
microservice, and SYSTEM_USER
and SYSTEM_PASSWORD
for the query
microservice.
If you want to override another property, you can specify it in the env
sections of the deploy.yaml
file. For example, set the CONTEXT_ROOT
environment variable in the system
deployment and the SYSTEM_CONTEXTROOT
environment variable in the query
deployment.
After the images and the secret are ready, you can deploy the microservices to your production environment with Kubernetes.
kubectl apply -f deploy.yaml
1<?xml version="1.0" encoding="UTF-8"?>
2<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
3
4 <modelVersion>4.0.0</modelVersion>
5
6 <groupId>io.openliberty.guides</groupId>
7
8 <artifactId>guide-microprofile-config-profile-system</artifactId>
9 <version>1.0-SNAPSHOT</version>
10 <packaging>war</packaging>
11
12 <properties>
13 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
14 <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
15 <maven.compiler.source>11</maven.compiler.source>
16 <maven.compiler.target>11</maven.compiler.target>
17 </properties>
18
19 <dependencies>
20 <dependency>
21 <groupId>jakarta.platform</groupId>
22 <artifactId>jakarta.jakartaee-api</artifactId>
23 <version>10.0.0</version>
24 <scope>provided</scope>
25 </dependency>
26 <dependency>
27 <groupId>org.eclipse.microprofile</groupId>
28 <artifactId>microprofile</artifactId>
29 <version>6.1</version>
30 <type>pom</type>
31 <scope>provided</scope>
32 </dependency>
33
34 <!-- For tests -->
35 <dependency>
36 <groupId>org.junit.jupiter</groupId>
37 <artifactId>junit-jupiter</artifactId>
38 <version>5.11.3</version>
39 <scope>test</scope>
40 </dependency>
41 <dependency>
42 <groupId>org.jboss.resteasy</groupId>
43 <artifactId>resteasy-client</artifactId>
44 <version>6.2.11.Final</version>
45 <scope>test</scope>
46 </dependency>
47 <dependency>
48 <groupId>org.jboss.resteasy</groupId>
49 <artifactId>resteasy-json-binding-provider</artifactId>
50 <version>6.2.11.Final</version>
51 <scope>test</scope>
52 </dependency>
53 </dependencies>
54
55 <build>
56 <finalName>${project.artifactId}</finalName>
57 <plugins>
58 <plugin>
59 <groupId>org.apache.maven.plugins</groupId>
60 <artifactId>maven-war-plugin</artifactId>
61 <version>3.4.0</version>
62 </plugin>
63 <plugin>
64 <groupId>io.openliberty.tools</groupId>
65 <artifactId>liberty-maven-plugin</artifactId>
66 <version>3.11.1</version>
67 </plugin>
68 <plugin>
69 <groupId>org.apache.maven.plugins</groupId>
70 <artifactId>maven-surefire-plugin</artifactId>
71 <version>3.5.2</version>
72 </plugin>
73 <plugin>
74 <groupId>org.apache.maven.plugins</groupId>
75 <artifactId>maven-failsafe-plugin</artifactId>
76 <version>3.5.2</version>
77 </plugin>
78 </plugins>
79 </build>
80
81 <profiles>
83 <profile>
85 <id>dev</id>
86 <activation>
87 <activeByDefault>true</activeByDefault>
88 </activation>
90
91 <properties>
92 <system.service.root>localhost:9081</system.service.root>
95 <liberty.var.http.port>9081</liberty.var.http.port>
98 <liberty.var.https.port>9444</liberty.var.https.port>
101 <liberty.var.default.username>alice</liberty.var.default.username>
104 <liberty.var.default.password>alicepwd</liberty.var.default.password>
107 <liberty.var.context.root>system/dev</liberty.var.context.root>
110 </properties>
111
112 <build>
113 <plugins>
115 <plugin>
116 <groupId>org.apache.maven.plugins</groupId>
117 <artifactId>maven-failsafe-plugin</artifactId>
118 <configuration>
119 <systemPropertyVariables>
120 <system.service>${system.service.root}</system.service>
121 <system.context>${liberty.var.context.root}</system.context>
122 <system.user>${liberty.var.default.username}</system.user>
123 <system.pwd>${liberty.var.default.password}</system.pwd>
124 </systemPropertyVariables>
125 </configuration>
126 </plugin>
128 </plugins>
129 </build>
130 </profile>
132
134 <profile>
135 <id>test</id>
136 <activation>
137 <activeByDefault>false</activeByDefault>
138 </activation>
139 <properties>
141 <system.service.root>localhost:9082</system.service.root>
143 <liberty.var.http.port>9082</liberty.var.http.port>
145 <liberty.var.https.port>9445</liberty.var.https.port>
147 <liberty.var.default.username>bob</liberty.var.default.username>
150 <liberty.var.default.password>bobpwd</liberty.var.default.password>
153 <liberty.var.context.root>system/test</liberty.var.context.root>
156 </properties>
157 <build>
158 <plugins>
159 <plugin>
160 <groupId>org.apache.maven.plugins</groupId>
161 <artifactId>maven-failsafe-plugin</artifactId>
162 <configuration>
163 <systemPropertyVariables>
164 <system.service>${system.service.root}</system.service>
165 <system.context>${liberty.var.context.root}</system.context>
166 <system.user>${liberty.var.default.username}</system.user>
167 <system.pwd>${liberty.var.default.password}</system.pwd>
168 </systemPropertyVariables>
169 </configuration>
170 </plugin>
171 </plugins>
172 </build>
173 </profile>
175
177 <profile>
178 <id>prod</id>
179 <activation>
180 <activeByDefault>false</activeByDefault>
181 </activation>
182 </profile>
184 </profiles>
185
186</project>187
Nice work! Where to next?
Nice work! You just learned how to use the MicroProfile Config’s configuration profile feature to configure your application for multiple CI/CD environments.
Feel free to try one of the related guides. They demonstrate new technologies that you can learn to expand on what you built in this guide.
Externalizing environment-specific microservice configuration for CI/CD by Open Liberty is licensed under CC BY-ND 4.0
What did you think of this guide?
Thank you for your feedback!
What could make this guide better?
Raise an issue to share feedback
Create a pull request to contribute to this guide
Need help?
Ask a question on Stack Overflow