From ca3fbac0ef7d2f5a2f39f5f378fee9b5c98d7d6d Mon Sep 17 00:00:00 2001 From: Steve Rao Date: Tue, 7 Jan 2025 04:49:56 +0800 Subject: [PATCH] Fix wrong dubbo trace caused by using rpcContext.isProviderSide() (#12930) Co-authored-by: Lauri Tulmin Co-authored-by: Lauri Tulmin --- .../v2_7/DubboInstrumentationModule.java | 6 +- ...emetryFilter.java => DubboSingletons.java} | 25 +++----- .../v2_7/OpenTelemetryClientFilter.java | 27 +++++++++ .../v2_7/OpenTelemetryServerFilter.java | 27 +++++++++ .../META-INF/org.apache.dubbo.rpc.Filter | 3 +- .../apachedubbo/v2_7/DubboTelemetry.java | 25 ++++++-- ...er.java => OpenTelemetryClientFilter.java} | 8 +-- .../v2_7/OpenTelemetryServerFilter.java | 28 +++++++++ .../apachedubbo/v2_7/TracingFilter.java | 60 ++++++++++++++++--- .../services/org.apache.dubbo.rpc.Filter | 3 +- 10 files changed, 175 insertions(+), 37 deletions(-) rename instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/{OpenTelemetryFilter.java => DubboSingletons.java} (62%) create mode 100644 instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryClientFilter.java create mode 100644 instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryServerFilter.java rename instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/{OpenTelemetryFilter.java => OpenTelemetryClientFilter.java} (80%) create mode 100644 instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryServerFilter.java diff --git a/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/DubboInstrumentationModule.java b/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/DubboInstrumentationModule.java index c5f0b5641e94..b353997ea93b 100644 --- a/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/DubboInstrumentationModule.java +++ b/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/DubboInstrumentationModule.java @@ -44,7 +44,11 @@ public ElementMatcher.Junction classLoaderMatcher() { public void injectClasses(ClassInjector injector) { injector .proxyBuilder( - "io.opentelemetry.javaagent.instrumentation.apachedubbo.v2_7.OpenTelemetryFilter") + "io.opentelemetry.javaagent.instrumentation.apachedubbo.v2_7.OpenTelemetryClientFilter") + .inject(InjectionMode.CLASS_ONLY); + injector + .proxyBuilder( + "io.opentelemetry.javaagent.instrumentation.apachedubbo.v2_7.OpenTelemetryServerFilter") .inject(InjectionMode.CLASS_ONLY); } diff --git a/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryFilter.java b/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/DubboSingletons.java similarity index 62% rename from instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryFilter.java rename to instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/DubboSingletons.java index 550f469ad9df..034d739dc82c 100644 --- a/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryFilter.java +++ b/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/DubboSingletons.java @@ -10,30 +10,23 @@ import io.opentelemetry.instrumentation.apachedubbo.v2_7.internal.DubboClientNetworkAttributesGetter; import io.opentelemetry.instrumentation.api.incubator.semconv.net.PeerServiceAttributesExtractor; import io.opentelemetry.javaagent.bootstrap.internal.AgentCommonConfig; -import org.apache.dubbo.common.extension.Activate; import org.apache.dubbo.rpc.Filter; -import org.apache.dubbo.rpc.Invocation; -import org.apache.dubbo.rpc.Invoker; -import org.apache.dubbo.rpc.Result; -@Activate(group = {"consumer", "provider"}) -public class OpenTelemetryFilter implements Filter { +public final class DubboSingletons { + public static final Filter CLIENT_FILTER; + public static final Filter SERVER_FILTER; - private final Filter delegate; - - public OpenTelemetryFilter() { - delegate = + static { + DubboTelemetry telemetry = DubboTelemetry.builder(GlobalOpenTelemetry.get()) .addAttributesExtractor( PeerServiceAttributesExtractor.create( new DubboClientNetworkAttributesGetter(), AgentCommonConfig.get().getPeerServiceResolver())) - .build() - .newFilter(); + .build(); + CLIENT_FILTER = telemetry.newClientFilter(); + SERVER_FILTER = telemetry.newServerFilter(); } - @Override - public Result invoke(Invoker invoker, Invocation invocation) { - return delegate.invoke(invoker, invocation); - } + private DubboSingletons() {} } diff --git a/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryClientFilter.java b/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryClientFilter.java new file mode 100644 index 000000000000..03e5ce6a2183 --- /dev/null +++ b/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryClientFilter.java @@ -0,0 +1,27 @@ +/* + * Copyright The OpenTelemetry Authors + * SPDX-License-Identifier: Apache-2.0 + */ + +package io.opentelemetry.javaagent.instrumentation.apachedubbo.v2_7; + +import org.apache.dubbo.common.extension.Activate; +import org.apache.dubbo.rpc.Filter; +import org.apache.dubbo.rpc.Invocation; +import org.apache.dubbo.rpc.Invoker; +import org.apache.dubbo.rpc.Result; + +@Activate(group = {"consumer"}) +public final class OpenTelemetryClientFilter implements Filter { + + private final Filter delegate; + + public OpenTelemetryClientFilter() { + delegate = DubboSingletons.CLIENT_FILTER; + } + + @Override + public Result invoke(Invoker invoker, Invocation invocation) { + return delegate.invoke(invoker, invocation); + } +} diff --git a/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryServerFilter.java b/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryServerFilter.java new file mode 100644 index 000000000000..f9d7065815c5 --- /dev/null +++ b/instrumentation/apache-dubbo-2.7/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/apachedubbo/v2_7/OpenTelemetryServerFilter.java @@ -0,0 +1,27 @@ +/* + * Copyright The OpenTelemetry Authors + * SPDX-License-Identifier: Apache-2.0 + */ + +package io.opentelemetry.javaagent.instrumentation.apachedubbo.v2_7; + +import org.apache.dubbo.common.extension.Activate; +import org.apache.dubbo.rpc.Filter; +import org.apache.dubbo.rpc.Invocation; +import org.apache.dubbo.rpc.Invoker; +import org.apache.dubbo.rpc.Result; + +@Activate(group = {"provider"}) +public final class OpenTelemetryServerFilter implements Filter { + + private final Filter delegate; + + public OpenTelemetryServerFilter() { + delegate = DubboSingletons.SERVER_FILTER; + } + + @Override + public Result invoke(Invoker invoker, Invocation invocation) { + return delegate.invoke(invoker, invocation); + } +} diff --git a/instrumentation/apache-dubbo-2.7/javaagent/src/main/resources/apache-dubbo-2.7/META-INF/org.apache.dubbo.rpc.Filter b/instrumentation/apache-dubbo-2.7/javaagent/src/main/resources/apache-dubbo-2.7/META-INF/org.apache.dubbo.rpc.Filter index c4d39b0b344f..f76248783a71 100644 --- a/instrumentation/apache-dubbo-2.7/javaagent/src/main/resources/apache-dubbo-2.7/META-INF/org.apache.dubbo.rpc.Filter +++ b/instrumentation/apache-dubbo-2.7/javaagent/src/main/resources/apache-dubbo-2.7/META-INF/org.apache.dubbo.rpc.Filter @@ -1 +1,2 @@ -io.opentelemetry.javaagent.instrumentation.apachedubbo.v2_7.OpenTelemetryFilter \ No newline at end of file +io.opentelemetry.javaagent.instrumentation.apachedubbo.v2_7.OpenTelemetryClientFilter +io.opentelemetry.javaagent.instrumentation.apachedubbo.v2_7.OpenTelemetryServerFilter diff --git a/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/DubboTelemetry.java b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/DubboTelemetry.java index 982ab442d27f..a44489af3463 100644 --- a/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/DubboTelemetry.java +++ b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/DubboTelemetry.java @@ -13,6 +13,9 @@ /** Entrypoint for instrumenting Apache Dubbo servers and clients. */ public final class DubboTelemetry { + private final Instrumenter serverInstrumenter; + private final Instrumenter clientInstrumenter; + /** Returns a new {@link DubboTelemetry} configured with the given {@link OpenTelemetry}. */ public static DubboTelemetry create(OpenTelemetry openTelemetry) { return builder(openTelemetry).build(); @@ -25,9 +28,6 @@ public static DubboTelemetryBuilder builder(OpenTelemetry openTelemetry) { return new DubboTelemetryBuilder(openTelemetry); } - private final Instrumenter serverInstrumenter; - private final Instrumenter clientInstrumenter; - DubboTelemetry( Instrumenter serverInstrumenter, Instrumenter clientInstrumenter) { @@ -35,8 +35,23 @@ public static DubboTelemetryBuilder builder(OpenTelemetry openTelemetry) { this.clientInstrumenter = clientInstrumenter; } - /** Returns a new Dubbo {@link Filter} that traces Dubbo RPC invocations. */ + /** + * Returns a new Dubbo {@link Filter} that traces Dubbo RPC invocations. + * + * @deprecated Use {@link #newClientFilter} and {@link #newServerFilter} instead. + */ + @Deprecated public Filter newFilter() { - return new TracingFilter(serverInstrumenter, clientInstrumenter); + return TracingFilter.newFilter(serverInstrumenter, clientInstrumenter); + } + + /** Returns a new Dubbo client {@link Filter} that traces Dubbo RPC invocations. */ + public Filter newClientFilter() { + return TracingFilter.newClientFilter(clientInstrumenter); + } + + /** Returns a new Dubbo server {@link Filter} that traces Dubbo RPC invocations. */ + public Filter newServerFilter() { + return TracingFilter.newServerFilter(serverInstrumenter); } } diff --git a/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryFilter.java b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryClientFilter.java similarity index 80% rename from instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryFilter.java rename to instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryClientFilter.java index 8cf99f56d89a..49f480429c88 100644 --- a/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryFilter.java +++ b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryClientFilter.java @@ -12,13 +12,13 @@ import org.apache.dubbo.rpc.Invoker; import org.apache.dubbo.rpc.Result; -@Activate(group = {"consumer", "provider"}) -public final class OpenTelemetryFilter implements Filter { +@Activate(group = {"consumer"}) +public final class OpenTelemetryClientFilter implements Filter { private final Filter delegate; - public OpenTelemetryFilter() { - delegate = DubboTelemetry.create(GlobalOpenTelemetry.get()).newFilter(); + public OpenTelemetryClientFilter() { + delegate = DubboTelemetry.create(GlobalOpenTelemetry.get()).newClientFilter(); } @Override diff --git a/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryServerFilter.java b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryServerFilter.java new file mode 100644 index 000000000000..06e8fd402395 --- /dev/null +++ b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/OpenTelemetryServerFilter.java @@ -0,0 +1,28 @@ +/* + * Copyright The OpenTelemetry Authors + * SPDX-License-Identifier: Apache-2.0 + */ + +package io.opentelemetry.instrumentation.apachedubbo.v2_7; + +import io.opentelemetry.api.GlobalOpenTelemetry; +import org.apache.dubbo.common.extension.Activate; +import org.apache.dubbo.rpc.Filter; +import org.apache.dubbo.rpc.Invocation; +import org.apache.dubbo.rpc.Invoker; +import org.apache.dubbo.rpc.Result; + +@Activate(group = {"provider"}) +public final class OpenTelemetryServerFilter implements Filter { + + private final Filter delegate; + + public OpenTelemetryServerFilter() { + delegate = DubboTelemetry.create(GlobalOpenTelemetry.get()).newServerFilter(); + } + + @Override + public Result invoke(Invoker invoker, Invocation invocation) { + return delegate.invoke(invoker, invocation); + } +} diff --git a/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/TracingFilter.java b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/TracingFilter.java index d3d239096bb1..feced63c0bf0 100644 --- a/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/TracingFilter.java +++ b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/java/io/opentelemetry/instrumentation/apachedubbo/v2_7/TracingFilter.java @@ -18,14 +18,52 @@ final class TracingFilter implements Filter { - private final Instrumenter serverInstrumenter; - private final Instrumenter clientInstrumenter; + private final InstrumenterSupplier instrumenterSupplier; - TracingFilter( + private TracingFilter(InstrumenterSupplier instrumenterSupplier) { + this.instrumenterSupplier = instrumenterSupplier; + } + + static TracingFilter newClientFilter(Instrumenter clientInstrumenter) { + return newFilter(clientInstrumenter, true); + } + + static TracingFilter newServerFilter(Instrumenter serverInstrumenter) { + return newFilter(serverInstrumenter, false); + } + + private static TracingFilter newFilter( + Instrumenter instrumenter, boolean isClientSide) { + return new TracingFilter( + new InstrumenterSupplier() { + + @Override + public Instrumenter get(RpcContext rpcContext) { + return instrumenter; + } + + @Override + public boolean isClientSide(RpcContext rpcContext) { + return isClientSide; + } + }); + } + + static TracingFilter newFilter( Instrumenter serverInstrumenter, Instrumenter clientInstrumenter) { - this.serverInstrumenter = serverInstrumenter; - this.clientInstrumenter = clientInstrumenter; + return new TracingFilter( + new InstrumenterSupplier() { + @Override + public Instrumenter get(RpcContext rpcContext) { + return rpcContext.isConsumerSide() ? clientInstrumenter : serverInstrumenter; + } + + @Override + public boolean isClientSide(RpcContext rpcContext) { + return rpcContext.isConsumerSide(); + } + }); } @Override @@ -40,9 +78,7 @@ public Result invoke(Invoker invoker, Invocation invocation) { return invoker.invoke(invocation); } - boolean isServer = rpcContext.isProviderSide(); - Instrumenter instrumenter = - isServer ? serverInstrumenter : clientInstrumenter; + Instrumenter instrumenter = instrumenterSupplier.get(rpcContext); Context parentContext = Context.current(); DubboRequest request = DubboRequest.create((RpcInvocation) invocation, rpcContext); @@ -55,7 +91,7 @@ public Result invoke(Invoker invoker, Invocation invocation) { boolean isSynchronous = true; try (Scope ignored = context.makeCurrent()) { result = invoker.invoke(invocation); - if (!isServer) { + if (instrumenterSupplier.isClientSide(rpcContext)) { CompletableFuture future = rpcContext.getCompletableFuture(); if (future != null) { isSynchronous = false; @@ -72,4 +108,10 @@ public Result invoke(Invoker invoker, Invocation invocation) { } return result; } + + private interface InstrumenterSupplier { + Instrumenter get(RpcContext rpcContext); + + boolean isClientSide(RpcContext rpcContext); + } } diff --git a/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/resources/META-INF/services/org.apache.dubbo.rpc.Filter b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/resources/META-INF/services/org.apache.dubbo.rpc.Filter index 42297d5caab4..250516c0dc0b 100644 --- a/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/resources/META-INF/services/org.apache.dubbo.rpc.Filter +++ b/instrumentation/apache-dubbo-2.7/library-autoconfigure/src/main/resources/META-INF/services/org.apache.dubbo.rpc.Filter @@ -1 +1,2 @@ -io.opentelemetry.instrumentation.apachedubbo.v2_7.OpenTelemetryFilter \ No newline at end of file +io.opentelemetry.instrumentation.apachedubbo.v2_7.OpenTelemetryClientFilter +io.opentelemetry.instrumentation.apachedubbo.v2_7.OpenTelemetryServerFilter