New vendor metrics and MicroProfile Reactive Messaging 3.0 updates available in Open Liberty 23.0.0.11-beta
The Open Liberty 23.0.0.11-beta includes new vendor metrics for MicroProfile Metrics that you can add directly to your dashboards. This release also introduces new capabilities for MicroProfile Reactive Messaging and MicroProfile Stream Operators, including negative acknowledgement.
In addition, Jakarta Data is still in beta; find out more in the Open Liberty 23.0.0.10-beta blog post.
The Open Liberty 23.0.0.11-beta includes the following beta features (along with all GA features):
See also previous Open Liberty beta blog posts.
New vendor metrics for MicroProfile Metrics 5.0
This update to MicroProfile Metrics 5.0 (mpMetrics-5.0) on Open Liberty includes some new vendor metrics at the /metrics endpoint.
Previously, you could calculate the new metrics for yourself from the Time and Total counts that were already provided for various monitoring components. For example, to obtain a "response time per request" metric, you would calculate it using the array of time series data provided by the MicroProfile Metrics feature. However, not all monitoring tools support such complex time series expressions.
With the MicroProfile Metrics 5.0 feature, you can use the new metrics directly in the dashboards of various monitoring tools, without any additional computation.
The following table lists the new vendor metrics:
| Metric | Endpoint output (Prometheus format) | 
|---|---|
| Process CPU Utilization Percent | 
 | 
| Heap Utilization Percent | 
 | 
| GC Time per Cycle | 
 | 
| Connection Pool in Use Time per Used Connection | 
 | 
| Connection Pool Wait Time per Queued Request | 
 | 
| Servlet Elapsed Time per Request | 
 | 
| REST Elapsed Time per Request | 
 | 
The Heap Utilization and CPU Utilization metrics are available when the server is started. The Connection Pool, REST, and Servlet metrics are available if the application contains any of the relevant data sources, REST APIs, or servlets, as is the case with the existing vendor metrics.
The new vendor metrics are available in the /metrics output, by enabling the following Microprofile Metrics 5.0 feature in your server.xml:
<featureManager>
   <feature>mpMetrics-5.0</feature>
</featureManager>For more information, see:
Negative acknowledgement and more for MicroProfile Reactive Messaging 3.0 and MicroProfile Stream Operators 3.0
MicroProfile Reactive Messaging 3.0 has a number of new features and changes over MicroProfile Reactive Messaging 1.0, including negative acknowledgements, emitters, and backpressure support.
An application using MicroProfile Reactive Messaging or MicroProfile Stream Operators is typically composed of CDI beans that consume, produce, and process messages that are passing along reactive streams. These messages can be internal to the application, or they can be sent and received through external message brokers. MicroProfile Reactive Messaging uses MicroProfile Stream Operators to pass messages through channels between methods or to messaging solutions, such as Kafka, to provide resilient storage of messages.
MicroProfile Reactive Messaging and Reactive Stream Operators 3.0 provide a range of new features and is compatible with Jakarta EE 9 onwards.
To use both features together, add the Microprofile Reactive Messaging 3.0 feature to the server.xml:
<feature>mpReactiveMessaging-3.0</feature>Enabling MicroProfile Reactive Messaging 3.0 automatically enables the MicroProfile Reactive Stream Operators 3.0 feature too.
To use MicroProfile Reactive Stream Operators 3.0 without MicroProfile Reactive Messaging:
<feature>mpReactiveStreams-3.0</feature>Negative acknowledgements
In MicroProfile Reactive Messaging 1.0, Messages could only be positively acknowledged. If there was a problem with the payload or if exceptional behavior occurred, there was no mechanism to indicate the problem or to handle the problem if it occurred within the stream. The new nack support can send or handle these events.
To explicitly negatively acknowledge an incoming Message:
@Incoming("data")
@Outgoing("out")
public Message<String> process(Message<String> m) {
    String s = m.getPayload();
    if (s.equalsIgnoreCase("b")) {
        // we cannot fail, we must nack explicitly.
        m.nack(new IllegalArgumentException("b"));
        return null;
    }
    return m.withPayload(s.toUpperCase());
}The method signature of accepting a Message without defining an acknowledgement strategy defaults the strategy to MANUAL. It is the responsibility of your code to ack() or nack() the message. In the previous example, the message can be acknowledged downstream of the out channel.
To throw an exception that causes a negative acknowledgement:
@Incoming("data")
@Outgoing("out")
public String process(String s) {
    if (s.equalsIgnoreCase("b")) {
        throw new IllegalArgumentException("b");
    }
    return s.toUpperCase();
}The method signature of accepting a payload without defining an acknowledgement strategy defaults the strategy to POST_PROCESSING. The implementation handles ack() and nack() calls on the message after the method or chain completes. The upstream data receives the negative acknowledgement with the reason of IllegalArgumentException. In the case of an exception being thrown, the implementation invokes the 'nack()' on the message.
Emitters
MicroProfile Reactive Messaging 1.0 did not offer a clear way to integrate imperative code, such as RESTful resources and beans, because Reactive Messaging put and took messages onto a channel according to the Outgoing or Incoming annotations. In version 3.0, emitters provide a bridge across the two models.
To inject emitters into a RESTful resources by using CDI to put messages onto a given channel:
@Inject
@Channel(CHANNEL_NAME)
Emitter<String> emitter;
@POST
@Path("/payload")
public CompletionStage<Void> emitPayload(String payload){
    CompletionStage<Void> cs = emitter.send(payload);
    return cs;
}
@POST
@Path("/message")
public CompletionStage<Void> emitPayload(String payload){
    CompletableFuture<Void> ackCf = new CompletableFuture<>();
    emitter.send(Message.of(payload,
        () -> {
            ackCf.complete(null);
            return CompletableFuture.completedFuture(null);
        },
        t -> {
            ackCf.completeExceptionally(t);
            return CompletableFuture.completedFuture(null);
        }));
    return ackCf;
}When defining emitters, you define the type of Object that will be sent as either the payload or the contents of the Message.
If an emitter sends a payload, MicroProfile Reactive Messaging automatically handles the invocation of ack() and nack() on the message. If, however, the emitter sends a message, the sending code must handle the message being either acked or nacked downstream.
Backpressure support
Backpressure support handles messages or payloads that are emitted faster than they are consumed. A backpressure strategy defines application behaviour in this circumstance. In the following example, the buffer holds up to 300 messages and throws an exception if it is full when a new message is emitted:
@Inject @Channel("myChannel")
@OnOverflow(value=OnOverflow.Strategy.BUFFER, bufferSize=300)
private Emitter<String> emitter;
public void publishMessage() {
  emitter.send("a");
  emitter.send("b");
  emitter.complete();
}You can define the following backpressure strategies:
- 
BUFFER- Use a buffer, with a size determined by the value of bufferSize, if set. Otherwise, the size is the value of themp.messaging.emitter.default-buffer-sizeMicroProfile Config property, if it exists. If neither of these values is defined, the default size is 128. If the buffer is full, an exception is thrown from the send method. This is the default strategy if no other strategy is defined.
- 
DROP- Drops the most recent value if the downstream can’t keep up. Any new values emitted by the emitter are ignored.
- 
FAIL- Propagates a failure in case the downstream can’t keep up. No more values will be emitted.
- 
LATEST- Keeps only the latest value, dropping any previous value if the downstream can’t keep up.
- 
NONE- Ignores the backpressure signals and leave it to the downstream consumer to implement a strategy.
- 
THROW_EXCEPTION- Throws an exception from the send method if the downstream can’t keep up.
- 
UNBOUNDED_BUFFER- Use an unbounded buffer. The application might run out of memory if values are continually added faster than they are consumed.
For more information, see:
Try it now
To try out these features, update your build tools to pull the Open Liberty All Beta Features package instead of the main release. The beta works with Java SE 21, Java SE 17, Java SE 11, and Java SE 8.
If you’re using Maven, you can install the All Beta Features package using:
<plugin>
    <groupId>io.openliberty.tools</groupId>
    <artifactId>liberty-maven-plugin</artifactId>
    <version>3.9</version>
    <configuration>
        <runtimeArtifact>
          <groupId>io.openliberty.beta</groupId>
          <artifactId>openliberty-runtime</artifactId>
          <version>23.0.0.11-beta</version>
          <type>zip</type>
        </runtimeArtifact>
    </configuration>
</plugin>You must also add dependencies to your pom.xml file for the beta version of the APIs that are associated with the beta features that you want to try.  For example, for Jakarta EE 10 and MicroProfile 6, you would include:
<dependency>
    <groupId>org.eclipse.microprofile</groupId>
    <artifactId>microprofile</artifactId>
    <version>6.0-RC3</version>
    <type>pom</type>
    <scope>provided</scope>
</dependency>
<dependency>
    <groupId>jakarta.platform</groupId>
    <artifactId>jakarta.jakartaee-api</artifactId>
    <version>10.0.0</version>
    <scope>provided</scope>
</dependency>Or for Gradle:
buildscript {
    repositories {
        mavenCentral()
    }
    dependencies {
        classpath 'io.openliberty.tools:liberty-gradle-plugin:3.7'
    }
}
apply plugin: 'liberty'
dependencies {
    libertyRuntime group: 'io.openliberty.beta', name: 'openliberty-runtime', version: '[23.0.0.11-beta,)'
}Or if you’re using container images:
FROM icr.io/appcafe/open-liberty:betaOr take a look at our Downloads page.
If you’re using IntelliJ IDEA, Visual Studio Code, or Eclipse IDE, try our open source Liberty developer tools for efficient development, testing, debugging, and application management, all within your IDE.
For more information on using a beta release, refer to the Installing Open Liberty beta releases documentation.
We welcome your feedback
Let us know what you think on our mailing list. If you hit a problem, post a question on StackOverflow. If you hit a bug, please raise an issue.