❗ Warning: This library is still a work in progress!
This library provides a way to manage spans and propagate them to other threads.
Defines current span management.
A SpanManager separates the creation of a Span
from its use later on.
This relieves application developers from passing the current span around through their code.
Only tracing-related code will need access to a SpanManager reference, provided as an ordinary dependency.
SpanManager provides the following methods:
manage(span)
makes the given span the current managed span.
Returns aManagedSpan
containing arelease()
method to later 'unmanage' the span with.currentSpan()
returns the current managed span, or theNoopSpan
if no span is managed.clear()
provides unconditional cleanup of all managed spans for the current process.
A default SpanManager maintaining a Stack
-like ThreadLocal
storage of linked managed spans.
Releasing a linked managed span uses the following algorithm:
- If the released span is not the current span, the current span is left alone.
- Otherwise, the first parent that is not yet released is set as the new current span.
- If no current parents remain, the current span is cleared.
- Consecutive
release()
calls for already-released spans will be ignored.
This ExecutorService
propagates the current span
from the caller into each call that is executed.
The current span of the caller is obtained from the configured SpanManager
.
Please Note: The current span is merely propagated (as-is).
It is explicitly not finished when the calls end,
nor will new spans be automatically related to the propagated span.
Contains factory-methods similar to standard java Executors
:
SpanPropagatingExecutors.newFixedThreadPool(int, SpanManager)
SpanPropagatingExecutors.newSingleThreadExecutor(SpanManager)
SpanPropagatingExecutors.newCachedThreadPool(SpanManager)
- Variants of the above with additional
ThreadFactory
argument.
This convenience Tracer
automates managing the current span:
- It wraps another
Tracer
. Spans
created with this tracer are:- automatically managed when started, and
- automatically released when finished.
To propagate a Span
into a new Thread
, the currentSpan from the caller must be
remembered by the Runnable
:
class ExampleRunnable implements Runnable {
private final SpanManager spanManager;
private final Span currentSpanFromCallingThread;
ExampleRunnable(SpanManager spanManager) {
this(spanManager, NoopSpan.INSTANCE);
}
private ExampleRunnable(SpanManager spanManager, Span currentSpanFromCallingThread) {
this.spanManager = spanManager;
this.currentSpanFromCallingThread = currentSpanFromCallingThread;
}
ExampleRunnable withCurrentSpan() {
return new ExampleRunnable(spanManager, spanManager.currentSpan());
}
@Override
public void run() {
try (ManagedSpan parent = spanManager.manage(currentSpanFromCallingThread)) {
// Any background code that requires tracing
// and may use spanManager.currentSpan().
} // parent.release() restores spanManager.currentSpan() to NoopSpan in new thread.
}
}
Then the application can propagate this currentSpan into background threads:
class App {
public static void main(String... args) throws InterruptedException {
Config config = ...;
Tracer tracer = config.getTracer();
SpanManager spanManager = config.getSpanManager();
ExampleRunnable runnable = new ExampleRunnable(spanManager);
try (Span appSpan = tracer.buildSpan("main").start(); // start appSpan
ManagedSpan managed = spanManager.manage(appSpan)) { // update currentSpan
Thread example = new Thread(runnable.withCurrentSpan());
example.start();
example.join();
} // managed.release() + appSpan.finish()
System.exit(0);
}
}
class TracedCall implements Callable<String> {
SpanManager spanManager = ... // inject or DefaultSpanManager.getInstance();
@Override
public String call() {
Span currentSpan = spanManager.currentSpan(); // Propagated span from caller
// ...
}
}
class Caller {
SpanManager spanManager = ... // inject or DefaultSpanManager.getInstance();
ExecutorService threadpool = new SpanPropagatingExecutorService(anyThreadpool(), spanManager);
void run() {
// ...code that sets the current Span somewhere:
try (ManagedSpan current = spanManager.manage(someSpan)) {
// scheduling the traced call:
Future<String> result = threadpool.submit(new TracedCall());
}
}
}
When starting a new span and making it the currentSpan, the manual example above used:
try (Span span = tracer.buildSpan("main").start(); // start span
ManagedSpan managed = spanManager.manage(span)) { // set currentSpan() to span
// ...traced block of code...
}
The ManagedSpanTracer
automatically makes every started span the current span.
It also releases it again when the span is finished:
class Caller {
SpanManager spanManager = ... // inject or DefaultSpanManager.getInstance();
Tracer tracer = new ManagedSpanTracer(anyTracer(), spanManager);
ExecutorService threadpool = new SpanPropagatingExecutorService(anyThreadpool(), spanManager);
void run() {
try (Span parent = tracer.buildSpan("parentOperation").start()) { // parent == currentSpan
// Scheduling the traced call:
Future<String> result = threadpool.submit(new TracedCall());
} // parent.finish() + ((ManagedSpan) parent).release()
}
}
When asynchronous processing is handled by separate request/response filters,
a try-with-resources
code block is insufficient.
Existing filters that start / finish new spans asynchronously can simply
be supplied with the ManagedSpanTracer
around the existing tracer.
This sets the currentSpan from the request filter
and calls release()
automatically from the response filter
when the existing filter finishes the span.
An example would be using the opentracing jaxrs filters
in combination with the ManagedSpanTracer:
// Add example when opentracing jaxrs library stabilizes
Alternatively, the following hypothetic filter pair could be used on an asynchronous server:
Handling the request:
final SpanManager spanManager = ... // inject or DefaultSpanManager.getInstance();
void onRequest(RequestContext reqCtx) {
Span span = ... // either obtain Span from previous filter or start from the request
ManagedSpan managedSpan = spanManager.manage(span); // span is now currentSpan.
reqCtx.put(SOMEKEY, managedSpan);
}
For the response:
final SpanManager spanManager = ...
void onResponse(RequestContext reqCtx, ResponseContext resCtx) {
spanManager.clear(); // Clear stack containing the currentSpan if this is a boundary-filter
// or:
// ManagedSpan managedSpan = reqCtx.get(SOMEKEY);
// managedSpan.release();
// If the corresponding request filter starts a span, don't forget to call span.finish() here!
}