Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Introduce LocalRootSpan (replacing ServerSpan) #5896

Merged
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
import io.opentelemetry.context.ContextKey;
import io.opentelemetry.instrumentation.api.instrumenter.ContextCustomizer;
import io.opentelemetry.instrumentation.api.instrumenter.Instrumenter;
import io.opentelemetry.instrumentation.api.server.ServerSpan;
import io.opentelemetry.instrumentation.api.instrumenter.LocalRootSpan;
import io.opentelemetry.semconv.trace.attributes.SemanticAttributes;
import javax.annotation.Nullable;

Expand Down Expand Up @@ -101,7 +101,7 @@ public static <T, U> void updateHttpRoute(
HttpRouteBiGetter<T, U> httpRouteGetter,
T arg1,
U arg2) {
Span serverSpan = ServerSpan.fromContextOrNull(context);
Span serverSpan = LocalRootSpan.fromContextOrNull(context);
// even if the server span is not sampled, we have to continue - we need to compute the
// http.route properly so that it can be captured by the server metrics
if (serverSpan == null) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,13 +8,16 @@
import io.opentelemetry.api.trace.Span;
import io.opentelemetry.api.trace.SpanKind;
import io.opentelemetry.context.Context;
import io.opentelemetry.instrumentation.api.internal.SpanKey;
import io.opentelemetry.instrumentation.api.instrumenter.LocalRootSpan;
import javax.annotation.Nullable;

/**
* This class encapsulates the context key for storing the current {@link SpanKind#SERVER} span in
* the {@link Context}.
*
* @deprecated Use {@link LocalRootSpan} instead.
*/
@Deprecated
public final class ServerSpan {

/**
Expand All @@ -23,13 +26,7 @@ public final class ServerSpan {
*/
@Nullable
public static Span fromContextOrNull(Context context) {
mateuszrzeszutek marked this conversation as resolved.
Show resolved Hide resolved
// try the HTTP semconv span first
Span span = SpanKey.HTTP_SERVER.fromContextOrNull(context);
// if it's absent then try the SERVER one - perhaps suppression by span kind is enabled
if (span == null) {
span = SpanKey.KIND_SERVER.fromContextOrNull(context);
}
return span;
return LocalRootSpan.fromContextOrNull(context);
}

private ServerSpan() {}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -162,6 +162,10 @@ public Context start(Context parentContext, REQUEST request) {
}
}

if (LocalRootSpan.isLocalRoot(parentContext)) {
context = LocalRootSpan.store(context, span);
}

return spanSuppressor.storeInContext(context, spanKind, span);
}

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
/*
* Copyright The OpenTelemetry Authors
* SPDX-License-Identifier: Apache-2.0
*/

package io.opentelemetry.instrumentation.api.instrumenter;

import io.opentelemetry.api.trace.Span;
import io.opentelemetry.api.trace.SpanContext;
import io.opentelemetry.context.Context;
import io.opentelemetry.context.ContextKey;
import javax.annotation.Nullable;

/**
* A local root span is a span that either does not have a parent span (it is the root span of a
* trace), or its parent span is a remote span (context was propagated from another application).
*/
public final class LocalRootSpan {

private static final ContextKey<Span> KEY =
ContextKey.named("opentelemetry-traces-local-root-span");

/**
* Returns the local root span from the current context or {@linkplain Span#getInvalid() invalid
* span} if there is no local root span.
*/
public static Span current() {
return fromContext(Context.current());
}

/**
* Returns the local root span from the given context or {@linkplain Span#getInvalid() invalid
* span} if there is no local root span in the context.
*/
public static Span fromContext(Context context) {
Span span = fromContextOrNull(context);
return span == null ? Span.getInvalid() : span;
}

/**
* Returns the local root span from the given context or {@code null} if there is no local root
* span in the context.
*/
@Nullable
public static Span fromContextOrNull(Context context) {
return context.get(KEY);
}

static boolean isLocalRoot(Context parentContext) {
SpanContext spanContext = Span.fromContext(parentContext).getSpanContext();
return !spanContext.isValid() || spanContext.isRemote();
}

static Context store(Context context, Span span) {
return context.with(KEY, span);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,6 @@
import io.opentelemetry.instrumentation.api.config.Config;
import io.opentelemetry.instrumentation.api.instrumenter.SpanSuppressor.BySpanKey;
import io.opentelemetry.instrumentation.api.instrumenter.SpanSuppressor.DelegateBySpanKind;
import io.opentelemetry.instrumentation.api.instrumenter.SpanSuppressor.JustStoreServer;
import io.opentelemetry.instrumentation.api.instrumenter.SpanSuppressor.Noop;
import io.opentelemetry.instrumentation.api.internal.SpanKey;
import io.opentelemetry.instrumentation.api.internal.SpanKeyProvider;
Expand All @@ -21,16 +20,11 @@
import java.util.Set;

enum SpanSuppressionStrategy {
/**
* Do not suppress spans at all.
*
* <p>This strategy will mark spans of {@link SpanKind#SERVER SERVER} kind in the context, so that
* they can be accessed using {@code ServerSpan}, but will not suppress server spans.
*/
/** Do not suppress spans at all. */
NONE {
@Override
SpanSuppressor create(Set<SpanKey> spanKeys) {
return JustStoreServer.INSTANCE;
return Noop.INSTANCE;
}
},
/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -250,6 +250,7 @@ void server_parent() {

Context context = instrumenter.start(Context.root(), request);
assertThat(Span.fromContext(context).getSpanContext().isValid()).isTrue();
assertThat(LocalRootSpan.fromContext(context)).isSameAs(Span.fromContext(context));

instrumenter.end(context, request, RESPONSE, null);

Expand Down Expand Up @@ -358,6 +359,7 @@ void client_parent() {

assertThat(spanContext.isValid()).isTrue();
assertThat(request).containsKey("traceparent");
assertThat(LocalRootSpan.fromContextOrNull(context)).isNull();

instrumenter.end(context, request, RESPONSE, new IllegalStateException("test"));

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
/*
* Copyright The OpenTelemetry Authors
* SPDX-License-Identifier: Apache-2.0
*/

package io.opentelemetry.instrumentation.api.instrumenter;

import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertNull;
import static org.junit.jupiter.api.Assertions.assertSame;
import static org.junit.jupiter.api.Assertions.assertTrue;

import io.opentelemetry.api.trace.Span;
import io.opentelemetry.api.trace.SpanContext;
import io.opentelemetry.api.trace.TraceFlags;
import io.opentelemetry.api.trace.TraceState;
import io.opentelemetry.context.Context;
import io.opentelemetry.context.Scope;
import org.junit.jupiter.api.Test;

class LocalRootSpanTest {

@Test
void spanWithoutParentIsLocalRoot() {
Context parentContext = Context.root();
assertTrue(LocalRootSpan.isLocalRoot(parentContext));
}

@Test
void spanWithInvalidParentIsLocalRoot() {
Context parentContext = Context.root().with(Span.getInvalid());
assertTrue(LocalRootSpan.isLocalRoot(parentContext));
}

@Test
void spanWithRemoteParentIsLocalRoot() {
Context parentContext =
Context.root()
.with(
Span.wrap(
SpanContext.createFromRemoteParent(
"00000000000000000000000000000001",
"0000000000000002",
TraceFlags.getSampled(),
TraceState.getDefault())));
assertTrue(LocalRootSpan.isLocalRoot(parentContext));
}

@Test
void spanWithValidLocalParentIsNotLocalRoot() {
Context parentContext =
Context.root()
.with(
Span.wrap(
SpanContext.create(
"00000000000000000000000000000001",
"0000000000000002",
TraceFlags.getSampled(),
TraceState.getDefault())));
assertFalse(LocalRootSpan.isLocalRoot(parentContext));
}

@Test
void shouldGetLocalRootSpan() {
Span span = Span.getInvalid();
Context context = LocalRootSpan.store(Context.root(), span);

try (Scope ignored = context.makeCurrent()) {
assertSame(span, LocalRootSpan.current());
}
assertSame(span, LocalRootSpan.fromContext(context));
assertSame(span, LocalRootSpan.fromContextOrNull(context));
}

@Test
void shouldNotGetLocalRootSpanIfThereIsNone() {
Context context = Context.root();

try (Scope ignored = context.makeCurrent()) {
assertFalse(LocalRootSpan.current().getSpanContext().isValid());
}
assertFalse(LocalRootSpan.fromContext(context).getSpanContext().isValid());
assertNull(LocalRootSpan.fromContextOrNull(context));
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -69,20 +69,9 @@ void none_shouldNotSuppressAnything(SpanKind spanKind, SpanKey spanKey) {
assertFalse(suppressor.shouldSuppress(context, spanKind));
}

@Test
void none_shouldStoreServerSpanInContext() {
SpanSuppressor suppressor = SpanSuppressionStrategy.NONE.create(emptySet());
Context context = Context.root();

Context newContext = suppressor.storeInContext(context, SpanKind.SERVER, span);

assertNotSame(newContext, context);
assertSame(span, SpanKey.KIND_SERVER.fromContextOrNull(newContext));
}

@ParameterizedTest
@EnumSource(value = SpanKind.class, mode = EnumSource.Mode.EXCLUDE, names = "SERVER")
void none_shouldNotStoreNonServerSpansInContext(SpanKind spanKind) {
@EnumSource(value = SpanKind.class)
void none_shouldNotStoreSpansInContext(SpanKind spanKind) {
SpanSuppressor suppressor = SpanSuppressionStrategy.NONE.create(emptySet());
Context context = Context.root();

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,9 +9,9 @@

import io.opentelemetry.api.trace.Span;
import io.opentelemetry.context.Context;
import io.opentelemetry.instrumentation.api.instrumenter.LocalRootSpan;
import io.opentelemetry.instrumentation.api.instrumenter.http.HttpRouteHolder;
import io.opentelemetry.instrumentation.api.instrumenter.http.HttpRouteSource;
import io.opentelemetry.instrumentation.api.server.ServerSpan;
import io.opentelemetry.javaagent.instrumentation.api.Java8BytecodeBridge;
import javax.ws.rs.container.ContainerRequestContext;

Expand All @@ -25,7 +25,7 @@ public static Context createOrUpdateAbortSpan(

requestContext.setProperty(JaxrsSingletons.ABORT_HANDLED, true);
Context parentContext = Java8BytecodeBridge.currentContext();
Span serverSpan = ServerSpan.fromContextOrNull(parentContext);
Span serverSpan = LocalRootSpan.fromContextOrNull(parentContext);
Span currentSpan = Java8BytecodeBridge.spanFromContext(parentContext);

HttpRouteHolder.updateHttpRoute(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,15 +7,15 @@

import io.opentelemetry.api.trace.Span;
import io.opentelemetry.context.Context;
import io.opentelemetry.instrumentation.api.server.ServerSpan;
import io.opentelemetry.instrumentation.api.instrumenter.LocalRootSpan;
import io.opentelemetry.javaagent.bootstrap.servlet.ServletContextPath;
import javax.servlet.http.HttpServletRequest;
import org.apache.axis2.jaxws.core.MessageContext;

public final class Axis2ServerSpanNaming {

public static void updateServerSpan(Context context, Axis2Request axis2Request) {
Span serverSpan = ServerSpan.fromContextOrNull(context);
Span serverSpan = LocalRootSpan.fromContextOrNull(context);
if (serverSpan == null) {
return;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@

import io.opentelemetry.api.trace.Span;
import io.opentelemetry.context.Context;
import io.opentelemetry.instrumentation.api.server.ServerSpan;
import io.opentelemetry.instrumentation.api.instrumenter.LocalRootSpan;
import io.opentelemetry.javaagent.bootstrap.servlet.ServletContextPath;
import javax.servlet.http.HttpServletRequest;

Expand All @@ -19,7 +19,7 @@ public static void updateServerSpanName(Context context, CxfRequest cxfRequest)
return;
}

Span serverSpan = ServerSpan.fromContextOrNull(context);
Span serverSpan = LocalRootSpan.fromContextOrNull(context);
if (serverSpan == null) {
return;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
import com.sun.xml.ws.api.message.Packet;
import io.opentelemetry.api.trace.Span;
import io.opentelemetry.context.Context;
import io.opentelemetry.instrumentation.api.server.ServerSpan;
import io.opentelemetry.instrumentation.api.instrumenter.LocalRootSpan;
import io.opentelemetry.javaagent.bootstrap.servlet.ServletContextPath;
import javax.servlet.http.HttpServletRequest;
import javax.xml.ws.handler.MessageContext;
Expand All @@ -21,7 +21,7 @@ public static void updateServerSpanName(Context context, MetroRequest metroReque
return;
}

Span serverSpan = ServerSpan.fromContextOrNull(context);
Span serverSpan = LocalRootSpan.fromContextOrNull(context);
if (serverSpan == null) {
return;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@

import io.opentelemetry.api.trace.Span;
import io.opentelemetry.context.Context;
import io.opentelemetry.instrumentation.api.server.ServerSpan;
import io.opentelemetry.instrumentation.api.instrumenter.LocalRootSpan;
import io.opentelemetry.javaagent.bootstrap.servlet.ServletContextPath;
import javax.faces.component.UIViewRoot;
import javax.faces.context.FacesContext;
Expand All @@ -16,7 +16,7 @@ public final class JsfServerSpanNaming {

public static void updateViewName(Context context, FacesContext facesContext) {
// just update the server span name, without touching the http.route
Span serverSpan = ServerSpan.fromContextOrNull(context);
Span serverSpan = LocalRootSpan.fromContextOrNull(context);
if (serverSpan == null) {
return;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -157,6 +157,11 @@ public static Context newContextWrapper(
"io.opentelemetry.api.baggage.BaggageContextKey",
BaggageBridging::toApplication,
BaggageBridging::toAgent),
new ContextKeyBridge<Span, io.opentelemetry.api.trace.Span>(
"application.io.opentelemetry.instrumentation.api.instrumenter.LocalRootSpan",
"io.opentelemetry.instrumentation.api.instrumenter.LocalRootSpan",
Bridging::toApplication,
Bridging::toAgentOrNull),
// span kind keys
bridgeSpanKey("KIND_SERVER_KEY"),
bridgeSpanKey("KIND_CLIENT_KEY"),
Expand Down
Loading