Micrometer and OpenTelemetry extension
This extension provides support for both Micrometer and OpenTelemetry in Quarkus applications. It streamlines integration by incorporating both extensions along with a bridge that enables sending Micrometer metrics via OpenTelemetry.
This technology is considered preview. In preview, backward compatibility and presence in the ecosystem is not guaranteed. Specific improvements might require changing configuration or APIs, and plans to become stable are under way. Feedback is welcome on our mailing list or as issues in our GitHub issue tracker. For a full list of possible statuses, check our FAQ entry. |
This document is part of the Observability in Quarkus reference guide which features this and other observability related components.
|
The extension allows the normal use of the Micrometer API, but have the metrics handled by the OpenTelemetry extension.
As an example, the @Timed
annotation from Micrometer is used to measure the execution time of a method:
import io.micrometer.core.annotation.Timed;
//...
@Timed(name = "timer_metric")
public String timer() {
return "OK";
}
The output telemetry data is handled by the OpenTelemetry SDK and sent by the quarkus-opentelemetry
extension exporter using the OTLP protocol.
This reduces the overhead of having an independent Micrometer registry plus the OpenTelemetry SDK in memory for the same application when both quarkus-micrometer
and quarkus-opentelemetry
extensions are used independently.
The OpenTelemetry SDK will handle all metrics. Either Micrometer metrics (manual or automatic) and OpenTelemetry Metrics can be used. All are available with this single extension.
All the configurations from the OpenTelemetry and Micrometer extensions are available with quarkus-micrometer-opentelemetry
.
The bridge is more than the simple OTLP registry found in Quarkiverse. In this extension, the OpenTelemetry SDK provides a Micrometer registry implementation based on the micrometer/micrometer-1.5
OpenTelemetry instrumentation library.
Usage
If you already have your Quarkus project configured, you can add the quarkus-micrometer-opentelemetry
extension to your project by running the following command in your project base directory:
quarkus extension add micrometer-opentelemetry
./mvnw quarkus:add-extension -Dextensions='micrometer-opentelemetry'
./gradlew addExtension --extensions='micrometer-opentelemetry'
This will add the following to your build file:
<dependency>
<groupId>io.quarkus</groupId>
<artifactId>quarkus-micrometer-opentelemetry</artifactId>
</dependency>
implementation("io.quarkus:quarkus-micrometer-opentelemetry")
Configuration
When the extension is present, Micrometer is enabled by default as are OpenTelemetry tracing, metrics and logs.
OpenTelemetry metrics auto-instrumentation for HTTP server and JVM metrics are disabled by default because those metrics can be collected by Micrometer.
Specific automatic Micrometer metrics are all disabled by default and can be enabled by setting, for example in the case of JVM metrics:
quarkus.micrometer.binder.jvm=true
in the application.properties
file.
For this and other properties you can use with the extension, Please refer to:
Metric differences between Micrometer and OpenTelemetry
API differences
The metrics produced with each framework follow different APIs and the mapping is not 1:1.
One fundamental API difference is that Micrometer uses a Timer and OpenTelemetry uses a Histogram to record latency (execution time) metrics and the frequency of the events.
When using the @Timed
annotation with Micrometer, 2 different metrics are created on the OpenTelemetry side, one Gauge
for the max
value and one Histogram
.
The DistributionSummary
from Micrometer is transformed into a Histogram
and a DoubleGauge
for the max
value. If service level objectives (slo) are set to true
when creating a DistributionSummary
, an additional histogram is created for them.
This table shows the differences between the two frameworks:
Micrometer | OpenTelemetry |
---|---|
DistributionSummary |
|
DistributionSummary with SLOs |
|
LongTaskTimer |
|
Timer |
|
Semantic convention differences
The 2 frameworks follow different semantic conventions. The OpenTelemetry Metrics are based on the OpenTelemetry Semantic Conventions and are still under active development (early 2025). Micrometer metrics convention format is around for a long time and has not changed much.
When these 2 configurations are set in the application.properties
file:
quarkus.micrometer.binder.jvm=true
quarkus.micrometer.binder.http-server.enabled=true
The JVM and HTTP server metrics are collected by Micrometer.
Next, are examples of the metrics collected by Micrometer and a comparison of what would be the quarkus-micrometer-registry-prometheus
output vs the one on this bridge. A link to the equivalent OpenTelemetry Semantic Convention is also provided for reference and is not currently used in the bridge.
Micrometer Meter | Quarkus Micrometer Prometheus output | This bridge OpenTelemetry output name | Related OpenTelemetry Semantic Convention (not applied) |
---|---|---|---|
Using the @Timed interceptor. |
method.timed (Histogram), method.timed.max (DoubleGauge) |
NA |
|
Using the @Counted interceptor. |
method.counted (DoubleSum) |
NA |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
N/A |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
N/A |
|
|
|
|
Some metrics might be missing from the output if they contain no data. |
See the output
Grafana-OTel-LGTM Dev Service
You can use the Grafana-OTel-LGTM Dev Service.
This Dev Service includes Grafana for visualizing data, Loki to store logs, Tempo to store traces and Prometheus to store metrics. It also provides an OTel collector to receive the data
Logging exporter
You can output all metrics to the console by setting the exporter to logging
in the application.properties
file:
quarkus.otel.metrics.exporter=logging (1)
quarkus.otel.metric.export.interval=10000ms (2)
1 | Set the exporter to logging .
Normally you don’t need to set this.
The default is cdi . |
2 | Set the interval to export the metrics.
The default is 1m , which is too long for debugging. |
Also add this dependency to your project:
<dependency>
<groupId>io.opentelemetry</groupId>
<artifactId>opentelemetry-exporter-logging</artifactId>
</dependency>