OpenTelemetry extension for generating spans via profiling instead of instrumentation. This extension enhances traces by running async-profiler in wall-clock profiling mode whenever there is an active sampled OpenTelemetry span.
The resulting profiling data is analyzed afterward and spans are "inferred". This means there is a delay between the regular and the inferred spans being visible in your OpenTelemetry backend/UI.
Only platform threads are supported. Virtual threads are not supported and will not be profiled.
This section describes the usage of this extension outside of an agent. Add the following dependency to your project:
<dependency>
<groupId>io.opentelemetry.contrib</groupId>
<artifactId>opentelemetry-inferred-spans</artifactId>
<version>{latest version}</version>
</dependency>
This extension supports autoconfiguration.
So if you are using an autoconfigured OpenTelemetry SDK, you'll only need to add this extension to your class path and configure it via system properties or environment variables:
Property Name / Environment Variable Name | Default | Description |
---|---|---|
otel.inferred.spans.enabled OTEL_INFERRED_SPANS_ENABLED |
false |
Enables the inferred spans feature. |
otel.inferred.spans.logging.enabled OTEL_INFERRED_SPANS_LOGGING_ENABLED |
true |
By default, async profiler prints warning messages about missing JVM symbols to standard output. Set this option to false to suppress such messages |
otel.inferred.spans.backup.diagnostic.files OTEL_INFERRED_SPANS_BACKUP_DIAGNOSTIC_FILES |
false |
Do not delete the temporary profiling files, can be used later to reproduce in case of issues. |
otel.inferred.spans.safe.mode OTEL_INFERRED_SPANS_SAFE_MODE |
0 |
Can be used for analysis: the Async Profiler's area that deals with recovering stack trace frames is known to be sensitive in some systems. It is used as a bit mask using values are between 0 and 31, where 0 enables all recovery attempts and 31 disables all five (corresponding 1, 2, 4, 8 and 16). |
otel.inferred.spans.post.processing.enabled OTEL_INFERRED_SPANS_POST_PROCESSING_ENABLED |
true |
Can be used to test the effect of the async-profiler in isolation from the agent's post-processing. |
otel.inferred.spans.sampling.interval OTEL_INFERRED_SPANS_SAMPLING_INTERVAL |
50ms |
he frequency at which stack traces are gathered within a profiling session. The lower you set it, the more accurate the durations will be. This comes at the expense of higher overhead and more spans for potentially irrelevant operations. The minimal duration of a profiling-inferred span is the same as the value of this setting. |
otel.inferred.spans.min.duration OTEL_INFERRED_SPANS_MIN_DURATION |
0ms |
The minimum duration of an inferred span. Note that the min duration is also implicitly set by the sampling interval. However, increasing the sampling interval also decreases the accuracy of the duration of inferred spans. |
otel.inferred.spans.included.classes OTEL_INFERRED_SPANS_INCLUDED_CLASSES |
* |
If set, the agent will only create inferred spans for methods which match this list. Setting a value may slightly reduce overhead and can reduce clutter by only creating spans for the classes you are interested in. Example: org.example.myapp.* |
otel.inferred.spans.excluded.classes OTEL_INFERRED_SPANS_EXCLUDED_CLASSES |
java.* javax.* sun.* com.sun.* jdk.* org.apache.tomcat.* org.apache.catalina.* org.apache.coyote.* org.jboss.as.* org.glassfish.* org.eclipse.jetty.* com.ibm.websphere.* io.undertow.* |
Excludes classes for which no profiler-inferred spans should be created. |
otel.inferred.spans.interval OTEL_INFERRED_SPANS_INTERVAL |
5s |
The interval at which profiling sessions should be started. |
otel.inferred.spans.duration OTEL_INFERRED_SPANS_DURATION |
5s |
The duration of a profiling session. For sampled transactions which fall within a profiling session (they start after and end before the session), so-called inferred spans will be created. They appear in the trace waterfall view like regular spans. NOTE: It is not recommended to set much higher durations as it may fill the activation events file and async-profiler's frame buffer. Warnings will be logged if the activation events file is full. If you want to have more profiling coverage, try decreasing profiling_inferred_spans_interval |
otel.inferred.spans.lib.directory OTEL_INFERRED_SPANS_LIB_DIRECTORY |
Defaults to the value of java.io.tmpdir |
Profiling requires that the async-profiler shared library is exported to a temporary location and loaded by the JVM. The partition backing this location must be executable, however in some server-hardened environments, noexec may be set on the standard /tmp partition, leading to java.lang.UnsatisfiedLinkError errors. Set this property to an alternative directory (e.g. /var/tmp ) to resolve this. |
otel.inferred.spans.duration OTEL_INFERRED_SPANS_DURATION |
5s |
The duration of a profiling session. For sampled transactions which fall within a profiling session (they start after and end before the session), so-called inferred spans will be created. They appear in the trace waterfall view like regular spans. NOTE: It is not recommended to set much higher durations as it may fill the activation events file and async-profiler's frame buffer. Warnings will be logged if the activation events file is full. If you want to have more profiling coverage, try decreasing profiling_inferred_spans_interval |
otel.inferred.spans.parent.override.handler OTEL_INFERRED_SPANS_PARENT_OVERRIDE_HANDLER |
Defaults to a handler adding span-links to the inferred span | Inferred spans sometimes need to be inserted as the new parent of a normal span, which is not directly possible because that span has already been sent. For this reason, this relationship needs to be represented differently, which normally is done by adding a span-link to the inferred span. This configuration can be used to override that behaviour by providing the fully qualified name of a class implementing BiConsumer<SpanBuilder, SpanContext> : The biconsumer will be invoked with the inferred span as first argument and the span for which the inferred one was detected as new parent as second argument |
If you manually set-up your OpenTelemetrySDK
, you need to create and register an InferredSpansProcessor
with your TracerProvider
:
InferredSpansProcessor inferredSpans = InferredSpansProcessor.builder()
//.samplingInterval(Duration.ofMillis(10))
.build();
SdkTracerProvider tracerProvider = SdkTracerProvider.builder()
.addSpanProcessor(inferredSpans)
.addSpanProcessor(BatchSpanProcessor.builder(OtlpGrpcSpanExporter.builder()
.setEndpoint("http://localhost:4317")
.build()).build())
.build();
inferredSpans.setTracerProvider(tracerProvider);
The setTracerProvider(..)
call shown at the end may be omitted, in that case GlobalOpenTelemetry
will be used for generating the inferred spans.
-
After each profiling session, while the stack traces and activation events are processed, no traces are collected.
-
Under load, processing can take seconds; ~200ms are normal.
-
Log:
DEBUG Processing {} stack traces ... DEBUG Processing traces took {}µs
-
-
While stack traces are processed, activation events are still put into the ring buffer. However, they don't get processed. If, during this period, there are more activation events than the buffer can handle, we're losing activation events.
- Log:
Could not add activation event to ring buffer as no slots are available
- Lost activation events can lead to orpaned call trees (lost end event), missing roots (lost start event) and messed up parent/child relationships (lost span activations/deactivations) Log:
DEBUG Illegal state ...
- Log:
-
Under load, the activation event ring buffer can also get full
-
The actual
otel.inferred.spans.sampling.interval
might be a bit lower. async-profiler aims to keep the interval relatively consistent but if there are too many threads actively running transactions or if there's a traffic spike, the interval can be lower. -
As a result of the above, some transactions don't contain inferred spans, even if their duration is longer than
otel.inferred.spans.sampling.interval
. Log:DEBUG Created no spans for thread {} (count={})
-
There can be a race condition when putting activation events into the queue which leads to older events being in front of newer ones, like
1, 2, 4, 3, 5
. But this is quite infrequent and the consequences are similar to loosing that activation event or event without any consequence. Log:Timestamp of current activation event ({}) is lower than the one from the previous event ({})
Inferred span starts after actual span, even though it should be the parent
---------[inferred ]
[actual]
^ ^ ^
Inferred span ends before actual span, even though it should be the parent
[inferred ]------------
[actual]
^ ^ ^
-------[inferred ]------- [actual ]
[actual ] -> -------[inferred ]-------
^ ^ ^ ^
Two consecutive method invocations are interpreted as one longer execution
[actual] [actual] -> [-------- --------]
^ ^
These are some tricky situations we have managed to find a workaround for.
This is tricky as regular spans are exported right after the event has ended. Inferred spans are sent later - after the profiling session ends.
This is how the situation looks like without a workaround:
[transaction ] [transaction ]
└─[inferred ] -> ├─[inferred ]
└─[actual] └───[actual]
There are situations where the ordering is off as a result of that.
The workaround is that inferred spans have span-links with a special is_child
attribute,
pointing to the regular spans they are the parent of.
Workaround: set end timestamp of inferred span to end timestamp of actual span.
[inferred ]-------- [inferred -----]--
[actual] -> [actual]
^ ^ ^
Workaround: set start timestamp of inferred span to start timestamp of actual span.
--------[inferred ] --[------inferred ]
[actual ] -> [actual ]
^ ^ ^
In this screenshot, we can see several problems at once
- Jack Shirazi, Elastic
- Jonas Kunz, Elastic
- Sylvain Juge, Elastic
Learn more about component owners in component_owners.yml.