DZone
Thanks for visiting DZone today,
Edit Profile
  • Manage Email Subscriptions
  • How to Post to DZone
  • Article Submission Guidelines
Sign Out View Profile
  • Post an Article
  • Manage My Drafts
Over 2 million developers have joined DZone.
Log In / Join
Please enter at least three characters to search
Refcards Trend Reports
Events Video Library
Refcards
Trend Reports

Events

View Events Video Library

Zones

Culture and Methodologies Agile Career Development Methodologies Team Management
Data Engineering AI/ML Big Data Data Databases IoT
Software Design and Architecture Cloud Architecture Containers Integration Microservices Performance Security
Coding Frameworks Java JavaScript Languages Tools
Testing, Deployment, and Maintenance Deployment DevOps and CI/CD Maintenance Monitoring and Observability Testing, Tools, and Frameworks
Culture and Methodologies
Agile Career Development Methodologies Team Management
Data Engineering
AI/ML Big Data Data Databases IoT
Software Design and Architecture
Cloud Architecture Containers Integration Microservices Performance Security
Coding
Frameworks Java JavaScript Languages Tools
Testing, Deployment, and Maintenance
Deployment DevOps and CI/CD Maintenance Monitoring and Observability Testing, Tools, and Frameworks

Because the DevOps movement has redefined engineering responsibilities, SREs now have to become stewards of observability strategy.

Apache Cassandra combines the benefits of major NoSQL databases to support data management needs not covered by traditional RDBMS vendors.

The software you build is only as secure as the code that powers it. Learn how malicious code creeps into your software supply chain.

Generative AI has transformed nearly every industry. How can you leverage GenAI to improve your productivity and efficiency?

Related

  • Managed Scheduled Executor Service vs EJB Timer
  • A Systematic Approach for Java Software Upgrades
  • From J2EE to Jakarta EE
  • A Maven Story

Trending

  • Designing AI Multi-Agent Systems in Java
  • A Guide to Using Amazon Bedrock Prompts for LLM Integration
  • Knowledge Graph Embeddings and NLP Innovations
  • Power BI Embedded Analytics — Part 3: Power BI Embedded Demo
  1. DZone
  2. Coding
  3. Java
  4. Getting Started With Jakarta EE and Eclipse MicroProfile

Getting Started With Jakarta EE and Eclipse MicroProfile

In this article, get started with Jakarta EE and Eclipse MicroProfile.

By 
Philip Riecks user avatar
Philip Riecks
·
Updated Apr. 21, 20 · Opinion
Likes (9)
Comment
Save
Tweet
Share
8.2K Views

Join the DZone community and get the full member experience.

Join For Free

Never heard of Jakarta EE and Eclipse MicroProfile yet? Then it's time to give it the first try and see how the Java Enterprise standard evolved. By developing a small application, this post demonstrates how to use these two technologies to build cloud-ready and state-of-the-art enterprise applications.

What Is Jakarta EE?

In a nutshell, it's a set of specifications (JAX-RS, CDI, JPA, JSON-P, etc.) to write enterprise applications with Java. These specifications are documents defining the API and interaction of a technology.

In Java words: the specification documents are like interfaces with formal definitions and the actual implementation of these interfaces is up to an application server vendor (e.g. WildFly, Open Liberty, Payara, TomEE). To ensure a vendor does not implement the specification in the wrong way, each specification provides a Technology Compatibility Kit (TCK). These are a set of tests to verify an implementation is compliant to the specification. Once an application server passes the TCKs for all specifications, it's compliant to the whole standard.

For many years, Java EE has been the standard way to write enterprise applications. Java EE (named J2EE in the earlier days) was for a long time maintained by Oracle. In 2017 Oracle decided to no longer evolve the enterprise standard under its umbrella and handed it over to the Eclipse Foundation. Due to legal reasons, they had to rebrand it and decided to name it Jakarta EE.

You can find all specifications of Jakarta EE at the official homepage and will also find more information about how the specification process now works at the Eclipse Foundation.

What Is Eclipse MicroProfile?

As the process of releasing and adapting new features for Java EE was rather slow, a group of vendors and community members decided to create MicroProfile in 2016. The main goal is to keep up with changes in the industry an optimize the existing platform for a microservice architecture.

Eclipse MicroProfile currently contains twelve specifications. Four of them are also part of Jakarta EE. This allows you to create standalone applications with just MicroProfile. All specifications are vendor-agnostic and the implementation is shipped with the application server.

You can see the Jakarta EE specifications as a solid foundation to build enterprise applications. On top of this, Eclipse MicroProfile fills the gap to build distributed systems following the Twelve-Factor App methodology.

Bootstrap Your First Project

Let's bootstrap a Jakarta EE and Eclipse MicroProfile with Java 11 using Maven. The pom.xml is pretty straightforward:

XML
 




x


 
1
<project xmlns="http://maven.apache.org/POM/4.0.0"
2
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
3
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
4
  <modelVersion>4.0.0</modelVersion>
5
 
          
6
  <groupId>de.rieckpil.blog</groupId>
7
  <artifactId>jakarta-ee-microprofile</artifactId>
8
  <version>1.0-SNAPSHOT</version>
9
  <packaging>war</packaging>
10
 
          
11
  <properties>
12
    <maven.compiler.source>11</maven.compiler.source>
13
    <maven.compiler.target>11</maven.compiler.target>
14
    <jakarta.jakartaee-api.version>8.0.0</jakarta.jakartaee-api.version>
15
    <microprofile.version>3.2</microprofile.version>
16
    <failOnMissingWebXml>false</failOnMissingWebXml>
17
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
18
    <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
19
  </properties>
20
 
          
21
  <dependencies>
22
    <dependency>
23
      <groupId>jakarta.platform</groupId>
24
      <artifactId>jakarta.jakartaee-api</artifactId>
25
      <version>${jakarta.jakartaee-api.version}</version>
26
      <scope>provided</scope>
27
    </dependency>
28
    <dependency>
29
      <groupId>org.eclipse.microprofile</groupId>
30
      <artifactId>microprofile</artifactId>
31
      <version>${microprofile.version}</version>
32
      <type>pom</type>
33
      <scope>provided</scope>
34
    </dependency>
35
  </dependencies>
36
 
          
37
  <build>
38
    <finalName>${project.artifactId}</finalName>
39
  </build>
40
</project>



Both dependencies for Jakarta EE and Eclipse MicroProfile are marked as provided. As the application server provides the actual implementation of the different specifications, we don't have to ship any library with our war  file.

This results in a so-called thin war of just some kilobytes and hence making deployment fast.

When it comes to deploying your application, you have to pick an application server. All Jakarta EE compliant application servers are listed on the following overview page and most of them also implement the Eclipse MicroProfile specification.

For this demo, I'm choosing Open Liberty, a light-weight open-source application server from IBM.

Manually downloading configuring the application server is sometimes a hurdle for Jakarta EE newcomers. Fortunately, the Open Liberty team provides an easy-to-use Maven plugin for rapid development experience. Once you include it to your  pom.xml, starting/stopping a local Open Liberty server is no big deal:

XML
 




x


 
1
<plugin>
2
  <groupId>io.openliberty.tools</groupId>
3
  <artifactId>liberty-maven-plugin</artifactId>
4
  <version>3.1</version>
5
</plugin>



This plugin also comes with a dev mode to have hot-reloading during local development and even shorter feedback cycles:  mvn liberty:dev.

Configure Open Liberty for Jakarta EE and Eclipse MicroProfile

First, let's configure Open Liberty. For this, create a  server.xml  file inside  src/main/liberty/config  with the following content:

XML
 




xxxxxxxxxx
1
17


 
1
<?xml version="1.0" encoding="UTF-8"?>
2
<server description="new server"> 
3
  
4
  <featureManager>
5
    <feature>javaee-8.0</feature>
6
    <feature>microProfile-3.2</feature>
7
  </featureManager>
8
  
9
  
10
  <mpMetrics authentication="false"/>
11
 
          
12
  
13
  <httpEndpoint id="defaultHttpEndpoint" httpPort="9080" httpsPort="9443"/>
14
 
          
15
  
16
  <quickStartSecurity userName="duke" userPassword="duke"/>
17
 
          
18
  
19
  <ssl id="defaultSSLConfig" trustDefaultCerts="true"/>
20
</server>
21
 
          



Open Liberty provides a simple way to configure the features we need for our application. For this demo, I'm including the features  javaee-8.0  and  microProfile-3.2. They will pull all specifications of Jakarta EE and Eclipse MicroProfile. As the Jakarta EE 8 specification versions are compatible with Java EE 8, we can use javaee-8.0 for our Jakarta EE application.

It's also possible to define only the features that are actually in-use and cherry-pick them to improve memory footprint and startup time.

For convenience, I'm disabling authentication for the MicroProfile metrics endpoint using mpMetrics. Furthermore, httpEndpoint  defines which ports Open Liberty will use. The quickStartSecurity  configures a default user to access restricted pages of the application server. Using the ssl  configuration we instruct Open Liberty to use the default certificates provided by the JDK. We'll need this, as our application will, later on, call a remote API using HTTPS.

For more information on how to configure Open Liberty, consider using this cheat sheet.

Create a JAX-RS Endpoint

The goal of this application is to return a quote of the day. We'll expose this information using a JAX-RS (Jakarta RESTful Web Services) endpoint.

You can bootstrap a JAX-RS application by providing a subclass of javax.ws.rs.core.Application on your classpath:

Java
 




x


 
1
@ApplicationPath("resources")
2
public class JAXRSConfiguration extends Application { 
3
}



The application path will be used as a prefix for all endpoints of the application. The actual public endpoint is defined in a different class and can be accessed via HTTP GET at /resources/quotes:

Java
 




x


 
1
@Path("quotes")
2
public class QuotesResource {
3
  
4
  @Inject
5
  @RestClient
6
  private QuotesClient quotesClient;
7
  
8
  @GET
9
  @Metered
10
  @Produces(MediaType.APPLICATION_JSON)
11
  public Response getQuoteOfTheDay() { 
12
    
13
    JsonObject quoteOfTheDay = quotesClient.getQuoteOfTheDay();
14
    JsonPointer quotePointer = Json.createPointer("/contents/quotes/0/quote");
15
    JsonObject result = Json.createObjectBuilder()
16
      .add("quoteOfTheDay", quotePointer.getValue(quoteOfTheDay)).build();
17
        
18
    return Response.ok(result).build();
19
  }
20
}



Our JAX-RS endpoint requires an instance of the  QuotesClient  to work. This client is using the MicroProfile Rest Client specification. You'll see the client in the following chapter.  

With CDI (Contexts and Dependency Injection) Jakarta EE provides a specification to make use of dependency injection. The  @Inject annotation tells the CDI container to inject an instance of the QuotesClient  to our class.

Multiple things are happening inside the  getQuoteOfTheDay  method. First, we get the JSON response of the remote API using the QuotesClient. As the actual quote is nested inside the JSON, I'm creating a JsonPointer  (part of JSON-P) to easily access the field inside the nested object. Finally, we'll use the pointer to extract the quote from the JSON and return it.

The  @Metered  annotation is part of MicroProfile Metrics and will measure the throughput of our endpoint using one-, five- and fifteen-minute rates. This might be a useful metric to monitor to see how our application is performing.

Fetch Data From a Remote API

As we are lazy developers, we don't want to implement boilerplate code to make an HTTP call. With MicroProfile Rest Client we solely define an interface and declare the remote endpoints using well-known annotations from JAX-RS. If you have experience with Feign, this might seem familiar:

Java
 




xxxxxxxxxx
1
19


 
1
@RegisterRestClient(baseUri = "https://quotes.rest/")
2
public interface QuotesClient {
3
4
  
5
  @GET
6
  @Path("qod")
7
  @Retry(maxRetries = 3, maxDuration = 3000L)
8
  @Fallback(fallbackMethod = "getFallbackQuote")
9
  JsonObject getQuoteOfTheDay();
10
11
  
12
  default JsonObject getFallbackQuote() {
13
    return Json.createObjectBuilder()
14
      .add("contents",
15
        Json.createObjectBuilder()
16
          .add("quotes", Json.createArrayBuilder()
17
            .add(Json.createObjectBuilder().add("quote", "Lorem ipsum dolor sit amet"))))
18
      .build();
19
  }
20
21
}



On top of this, we can make our remote calls more resilient using MicroProfile Fault Tolerance. This specification provides a set of strategies to build resilient and fault-tolerant services. For our example, I'm using @Retry and @Fallback. This ensures that we'll try to access the API three times (in case the first and second call failed) and will return a fallback in case this is not successful we are exceeding 3 seconds for this operation.

Eclipse MicroProfile Fault Tolerance has an integration with the Metrics specification that allows us to get statistics about our retry out-of-the-box.

Inspect Application Metrics

The application is now ready to be deployed. Given the Open Liberty Maven plugin, we can start the server and deploy the application in the foreground with mvn liberty:run. You should see something similar like this to ensure the application is up and running:

Shell
 




xxxxxxxxxx
1
10
9


 
1
[INFO] [AUDIT   ] CWWKT0017I: Web application removed (default_host): http://localhost:9080/jwt/
2
[INFO] [AUDIT   ] CWWKT0017I: Web application removed (default_host): http://localhost:9080/openapi/ui/
3
[INFO] [AUDIT   ] CWWKT0017I: Web application removed (default_host): http://localhost:9080/metrics/
4
[INFO] [AUDIT   ] CWWKT0017I: Web application removed (default_host): http://localhost:9080/health/
5
[INFO] [AUDIT   ] CWWKT0017I: Web application removed (default_host): http://localhost:9080/
6
[INFO] [AUDIT   ] CWWKT0017I: Web application removed (default_host): http://localhost:9080/openapi/
7
[INFO] [AUDIT   ] CWWKT0017I: Web application removed (default_host): http://localhost:9080/ibm/api/
8
[INFO] [AUDIT   ] CWWKF0012I: The server installed the following features: [appClientSupport-1.0, appSecurity-2.0, appSecurity-3.0, batch-1.0, beanValidation-2.0, cdi-2.0, concurrent-1.0, distributedMap-1.0, ejb-3.2, ejbHome-3.2, ejbLite-3.2, ejbPersistentTimer-3.2, ejbRemote-3.2, el-3.0, j2eeManagement-1.1, jacc-1.5, jaspic-1.1, javaMail-1.6, javaee-8.0, jaxb-2.2, jaxrs-2.1, jaxrsClient-2.1, jaxws-2.2, jca-1.7, jcaInboundSecurity-1.0, jdbc-4.2, jms-2.0, jndi-1.0, jpa-2.2, jpaContainer-2.2, jsf-2.3, json-1.0, jsonb-1.0, jsonp-1.1, jsp-2.3, jwt-1.0, managedBeans-1.0, mdb-3.2, microProfile-3.2, mpConfig-1.3, mpFaultTolerance-2.0, mpHealth-2.1, mpJwt-1.1, mpMetrics-2.2, mpOpenAPI-1.1, mpOpenTracing-1.3, mpRestClient-1.3, opentracing-1.3, servlet-4.0, ssl-1.0, wasJmsClient-2.0, wasJmsSecurity-1.0, wasJmsServer-1.0, webProfile-8.0, websocket-1.1].
9
[INFO] [AUDIT   ] CWWKF0011I: The defaultServer server is ready to run a smarter planet. The defaultServer server started in 9.941 seconds.



Now you can start accessing the public endpoint to fetch the quote of the day:

Shell
 




x


 
1
 curl -v http://localhost:9080/resources/quotes
2
< HTTP/1.1 200 OK
3
< X-Powered-By: Servlet/4.0
4
< Content-Type: application/json
5
< Date: Mon, 30 Mar 2020 07:47:26 GMT
6
< Content-Language: en-US
7
< Content-Length: 127
8
< 
9
{"quoteOfTheDay":"From now on we live in a world where man has walked on the Moon. It's not a miracle; we just decided to go."}
10
 
          



In case of an error (more than three retries/seconds for calling the remote API), our application responds with a fallback:

Shell
x
 
1
{"quoteOfTheDay":"Lorem ipsum dolor sit amet"}


To get insights for the retry statistics, the throughput of the endpoint and further technical metrics like CPU usage and memory, you can access  http://localhost:9080/metrics :

Plain Text
 




x


 
1
# TYPE application_ft_de_rieckpil_learning_QuotesClient_getQuoteOfTheDay_invocations_total counter
2
application_ft_de_rieckpil_learning_QuotesClient_getQuoteOfTheDay_invocations_total 5
3
# TYPE application_ft_de_rieckpil_learning_QuotesClient_getQuoteOfTheDay_retry_callsSucceededRetried_total counter
4
application_ft_de_rieckpil_learning_QuotesClient_getQuoteOfTheDay_retry_callsSucceededRetried_total 0
5
# TYPE application_ft_de_rieckpil_learning_QuotesClient_getQuoteOfTheDay_retry_callsFailed_total counter
6
application_ft_de_rieckpil_learning_QuotesClient_getQuoteOfTheDay_retry_callsFailed_total 0
7
# TYPE application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_total counter
8
application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_total 5
9
# TYPE application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_rate_per_second gauge
10
application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_rate_per_second 0.008803184637398256
11
# TYPE application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_one_min_rate_per_second gauge
12
application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_one_min_rate_per_second 0.03532066437518856
13
# TYPE application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_five_min_rate_per_second gauge
14
application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_five_min_rate_per_second 0.04249155859338982
15
# TYPE application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_fifteen_min_rate_per_second gauge
16
application_de_rieckpil_learning_QuotesResource_getQuoteOfTheDay_fifteen_min_rate_per_second 0.11158522807330326
17
# TYPE application_ft_de_rieckpil_learning_QuotesClient_getQuoteOfTheDay_retry_callsSucceededNotRetried_total counter
18
application_ft_de_rieckpil_learning_QuotesClient_getQuoteOfTheDay_retry_callsSucceededNotRetried_total 5



Summary

This was a first (and rather quick) guide to both Jakarta EE and Eclipse MicroProfile. Given the following specifications: CDI, JAX-RS, JSON-P, JSON-B, MicroProfile Metrics/Fault Tolerance/Rest Client we were able to build a small resilient and monitorable application. 

There are way more specifications to explore. JWT Auth, OpenAPI support, distributed tracing, configuration, health checks, just to name some for Eclipse MicroProfile. Major parts of Jakarta EE, that were not covered with the demo, are JPA for persistence and JMS for messaging.

The source code for this application with instructions to run it is available on GitHub.

If you want to learn more about Eclipse MicroProfile, consider signing up for my free course bundle (E-Book and Video Course). For more hands-on examples on Jakarta EE, I'm frequently uploading videos on YouTube to make you familiar with this technology.

Further resources on both topics are available at the official homepages:

  • https://microprofile.io
  • https://jakarta.ee

Have fun being productive with Jakarta EE and Eclipse MicroProfile,

Phil

Eclipse application Application server Java (programming language) Java EE

Opinions expressed by DZone contributors are their own.

Related

  • Managed Scheduled Executor Service vs EJB Timer
  • A Systematic Approach for Java Software Upgrades
  • From J2EE to Jakarta EE
  • A Maven Story

Partner Resources

×

Comments
Oops! Something Went Wrong

The likes didn't load as expected. Please refresh the page and try again.

ABOUT US

  • About DZone
  • Support and feedback
  • Community research
  • Sitemap

ADVERTISE

  • Advertise with DZone

CONTRIBUTE ON DZONE

  • Article Submission Guidelines
  • Become a Contributor
  • Core Program
  • Visit the Writers' Zone

LEGAL

  • Terms of Service
  • Privacy Policy

CONTACT US

  • 3343 Perimeter Hill Drive
  • Suite 100
  • Nashville, TN 37211
  • support@dzone.com

Let's be friends:

Likes
There are no likes...yet! 👀
Be the first to like this post!
It looks like you're not logged in.
Sign in to see who liked this post!