io.opentracing.Tracer Maven / Gradle / Ivy
/*
* Copyright 2016-2019 The OpenTracing Authors
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
* in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License
* is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
* or implied. See the License for the specific language governing permissions and limitations under
* the License.
*/
package io.opentracing;
import java.io.Closeable;
import io.opentracing.propagation.Format;
import io.opentracing.tag.Tag;
/**
* Tracer is a simple, thin interface for Span creation and propagation across arbitrary transports.
*/
public interface Tracer extends Closeable {
/**
* @return the current {@link ScopeManager}, which may be a noop but may not be null.
*/
ScopeManager scopeManager();
/**
* @return the active {@link Span}. This is a shorthand for {@code Tracer.scopeManager().activeSpan()}.
*/
Span activeSpan();
/**
* Make a {@link Span} instance active for the current context (usually a thread).
* This is a shorthand for {@code Tracer.scopeManager().activate(span)}.
*
* @return a {@link Scope} instance to control the end of the active period for the {@link Span}. It is a
* programming error to neglect to call {@link Scope#close()} on the returned instance,
* and it may lead to memory leaks as the {@link Scope} may remain in the thread-local stack.
*/
Scope activateSpan(Span span);
/**
* Return a new SpanBuilder for a Span with the given `operationName`.
*
* You can override the operationName later via {@link Span#setOperationName(String)}.
*
*
A contrived example:
*
* Tracer tracer = ...
*
* // Note: if there is a `tracer.activeSpan()` instance, it will be used as the target
* // of an implicit CHILD_OF Reference when `start()` is invoked,
* // unless another Span reference is explicitly provided to the builder.
* Span span = tracer.buildSpan("HandleHTTPRequest")
* .asChildOf(rpcSpanContext) // an explicit parent
* .withTag("user_agent", req.UserAgent)
* .withTag("lucky_number", 42)
* .start();
* span.setTag("...", "...");
*
* // It is possible to set the Span as the active instance for the current context
* // (usually a thread).
* try (Scope scope = tracer.activateSpan(span)) {
* ...
* }
*
*/
SpanBuilder buildSpan(String operationName);
/**
* Inject a SpanContext into a `carrier` of a given type, presumably for propagation across process boundaries.
*
* Example:
*
* Tracer tracer = ...
* Span clientSpan = ...
* TextMap httpHeadersCarrier = new AnHttpHeaderCarrier(httpRequest);
* tracer.inject(span.context(), Format.Builtin.HTTP_HEADERS, httpHeadersCarrier);
*
*
* @param the carrier type, which also parametrizes the Format.
* @param spanContext the SpanContext instance to inject into the carrier
* @param format the Format of the carrier
* @param carrier the carrier for the SpanContext state. All Tracer.inject() implementations must support
* io.opentracing.propagation.TextMap and java.nio.ByteBuffer.
*
* @see io.opentracing.propagation.Format
* @see io.opentracing.propagation.Format.Builtin
*/
void inject(SpanContext spanContext, Format format, C carrier);
/**
* Extract a SpanContext from a `carrier` of a given type, presumably after propagation across a process boundary.
*
* Example:
*
* Tracer tracer = ...
* TextMap httpHeadersCarrier = new AnHttpHeaderCarrier(httpRequest);
* SpanContext spanCtx = tracer.extract(Format.Builtin.HTTP_HEADERS, httpHeadersCarrier);
* ... = tracer.buildSpan('...').asChildOf(spanCtx).start();
*
*
* If the span serialized state is invalid (corrupt, wrong version, etc) inside the carrier this will result in an
* IllegalArgumentException. If the span serialized state is missing the method returns null.
*
* @param the carrier type, which also parametrizes the Format.
* @param format the Format of the carrier
* @param carrier the carrier for the SpanContext state. All Tracer.extract() implementations must support
* io.opentracing.propagation.TextMap and java.nio.ByteBuffer.
*
* @return the SpanContext instance holding context to create a Span, null otherwise.
*
* @see io.opentracing.propagation.Format
* @see io.opentracing.propagation.Format.Builtin
*/
SpanContext extract(Format format, C carrier);
/**
* Closes the Tracer, and tries to flush the in-memory collection to the configured persistance store.
*
*
* The close method should be considered idempotent; closing an already closed Tracer should not raise an error.
* Spans that are created or finished after a Tracer has been closed may or may not be flushed.
* Calling the close method should be considered a synchronous operation. Observe this call may block for
* a relatively long period of time, depending on the internal shutdown.
*
* For stateless tracers, this can be a no-op.
*/
@Override
void close();
interface SpanBuilder {
/**
* A shorthand for addReference(References.CHILD_OF, parent).
*
*
* If parent==null, this is a noop.
*/
SpanBuilder asChildOf(SpanContext parent);
/**
* A shorthand for addReference(References.CHILD_OF, parent.context()).
*
*
* If parent==null, this is a noop.
*/
SpanBuilder asChildOf(Span parent);
/**
* Add a reference from the Span being built to a distinct (usually parent) Span. May be called multiple times
* to represent multiple such References.
*
*
* If
*
* - the {@link Tracer}'s {@link ScopeManager#activeSpan()} is not null, and
*
- no explicit references are added via {@link SpanBuilder#addReference}, and
*
- {@link SpanBuilder#ignoreActiveSpan()} is not invoked,
*
* ... then an inferred {@link References#CHILD_OF} reference is created to the
* {@link ScopeManager#activeSpan()} {@link SpanContext} when either {@link SpanBuilder#startActive(boolean)} or
* {@link SpanBuilder#start} is invoked.
*
* @param referenceType the reference type, typically one of the constants defined in References
* @param referencedContext the SpanContext being referenced; e.g., for a References.CHILD_OF referenceType, the
* referencedContext is the parent. If referencedContext==null, the call to
* {@link #addReference} is a noop.
*
* @see io.opentracing.References
*/
SpanBuilder addReference(String referenceType, SpanContext referencedContext);
/**
* Do not create an implicit {@link References#CHILD_OF} reference to the {@link ScopeManager#activeSpan()}).
*/
SpanBuilder ignoreActiveSpan();
/** Same as {@link Span#setTag(String, String)}, but for the span being built. */
SpanBuilder withTag(String key, String value);
/** Same as {@link Span#setTag(String, boolean)}, but for the span being built. */
SpanBuilder withTag(String key, boolean value);
/** Same as {@link Span#setTag(String, Number)}, but for the span being built. */
SpanBuilder withTag(String key, Number value);
/** Same as {@link AbstractTag#set(Span, T)}, but for the span being built. */
SpanBuilder withTag(Tag tag, T value);
/** Specify a timestamp of when the Span was started, represented in microseconds since epoch. */
SpanBuilder withStartTimestamp(long microseconds);
/**
* @deprecated use {@link #start} instead.
*/
@Deprecated
Span startManual();
/**
* Returns a newly-started {@link Span}.
*
*
* If
*
* - the {@link Tracer}'s {@link ScopeManager#activeSpan()} is not null, and
*
- no explicit references are added via {@link SpanBuilder#addReference}, and
*
- {@link SpanBuilder#ignoreActiveSpan()} is not invoked,
*
* ... then an inferred {@link References#CHILD_OF} reference is created to the
* {@link ScopeManager#activeSpan()}'s {@link SpanContext} when either
* {@link SpanBuilder#start()} or {@link SpanBuilder#startActive} is invoked.
* @return the newly-started Span instance, which has *not* been automatically registered
* via the {@link ScopeManager}
*/
Span start();
/**
* @deprecated use {@link #start()} and {@link ScopeManager#activate(Span span)} instead.
* Returns a newly started and activated {@link Scope}.
*
*
* {@link SpanBuilder#startActive()} is a shorthand for
* {@code tracer.scopeManager().activate(spanBuilder.start())}.
* The returned {@link Scope} supports try-with-resources, but using this method is
* discouraged as the {@link Span} reference could be easily lost, and reporting
* errors on {@link Span} through this method becomes impossible:
*
* try (Scope scope = tracer.buildSpan("...").startActive(true)) {
* // (Do work)
* scope.span().setTag( ... ); // etc, etc
* } catch (Exception e) {
* // Not possible to report errors, as
* // the span reference has been lost,
* // and span has been already finished too.
* }
*
*
*
* It is recommended to use {@link #start()} with a subsequent call to
* {@link ScopeManager#activate(Span)}
*
* Span span = tracer.buildSpan("...").start();
* try (Scope scope = tracer.activateSpan(span)) {
* } catch (Exception e) {
* span.log(...); // Report any errors properly.
* } finally {
* span.finish(); // Optionally close the Span.
* }
*
*
* @return a {@link Scope}, already registered via the {@link ScopeManager}
*
* @see ScopeManager
* @see Scope
*/
@Deprecated
Scope startActive(boolean finishSpanOnClose);
}
}