From 004c45e6c31fe6ab381b26bcd37b1626bf438d37 Mon Sep 17 00:00:00 2001 From: mateuszrzeszutek Date: Tue, 15 Sep 2020 12:34:07 +0200 Subject: [PATCH] Implement MDC auto-instrumentation for log4j2 --- .../instrumentation/api/log/MdcConstants.java | 25 +++++++++ ...ion.java => Log4j2MdcInstrumentation.java} | 51 +++++-------------- .../library/log4j-2.13.2-library.gradle | 3 -- .../OpenTelemetryContextDataProvider.java | 12 +++-- ...ogging.log4j.core.util.ContextDataProvider | 1 + ...on.java => Log4j27MdcInstrumentation.java} | 21 +++++--- .../logback/v1_0_0/OpenTelemetryAppender.java | 12 +++-- 7 files changed, 66 insertions(+), 59 deletions(-) create mode 100644 instrumentation-api/src/main/java/io/opentelemetry/instrumentation/api/log/MdcConstants.java rename instrumentation/log4j/log4j-2.13.2/auto/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_13_2/{Log4j2Instrumentation.java => Log4j2MdcInstrumentation.java} (51%) create mode 100644 instrumentation/log4j/log4j-2.13.2/library/src/main/resources/META-INF/services/org.apache.logging.log4j.core.util.ContextDataProvider rename instrumentation/log4j/log4j-2.7/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_7/{Log4j27Instrumentation.java => Log4j27MdcInstrumentation.java} (79%) diff --git a/instrumentation-api/src/main/java/io/opentelemetry/instrumentation/api/log/MdcConstants.java b/instrumentation-api/src/main/java/io/opentelemetry/instrumentation/api/log/MdcConstants.java new file mode 100644 index 000000000000..5f6b335f8a49 --- /dev/null +++ b/instrumentation-api/src/main/java/io/opentelemetry/instrumentation/api/log/MdcConstants.java @@ -0,0 +1,25 @@ +/* + * Copyright The OpenTelemetry 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.opentelemetry.instrumentation.api.log; + +public final class MdcConstants { + public static final String TRACE_ID = "traceId"; + public static final String SPAN_ID = "spanId"; + public static final String SAMPLED = "sampled"; + + private MdcConstants() {} +} diff --git a/instrumentation/log4j/log4j-2.13.2/auto/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_13_2/Log4j2Instrumentation.java b/instrumentation/log4j/log4j-2.13.2/auto/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_13_2/Log4j2MdcInstrumentation.java similarity index 51% rename from instrumentation/log4j/log4j-2.13.2/auto/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_13_2/Log4j2Instrumentation.java rename to instrumentation/log4j/log4j-2.13.2/auto/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_13_2/Log4j2MdcInstrumentation.java index 611d13ef463c..ca5656f419f7 100644 --- a/instrumentation/log4j/log4j-2.13.2/auto/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_13_2/Log4j2Instrumentation.java +++ b/instrumentation/log4j/log4j-2.13.2/auto/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_13_2/Log4j2MdcInstrumentation.java @@ -17,30 +17,20 @@ package io.opentelemetry.instrumentation.auto.log4j.v2_13_2; import static io.opentelemetry.javaagent.tooling.ClassLoaderMatcher.hasClassesNamed; -import static net.bytebuddy.matcher.ElementMatchers.isMethod; -import static net.bytebuddy.matcher.ElementMatchers.isPrivate; -import static net.bytebuddy.matcher.ElementMatchers.isStatic; import static net.bytebuddy.matcher.ElementMatchers.named; -import static net.bytebuddy.matcher.ElementMatchers.takesArguments; import com.google.auto.service.AutoService; -import io.opentelemetry.instrumentation.log4j.v2_13_2.OpenTelemetryContextDataProvider; import io.opentelemetry.javaagent.tooling.Instrumenter; -import java.util.ArrayList; import java.util.Collections; -import java.util.List; import java.util.Map; -import net.bytebuddy.asm.Advice; import net.bytebuddy.description.method.MethodDescription; import net.bytebuddy.description.type.TypeDescription; -import net.bytebuddy.implementation.bytecode.assign.Assigner.Typing; import net.bytebuddy.matcher.ElementMatcher; -import org.apache.logging.log4j.core.util.ContextDataProvider; @AutoService(Instrumenter.class) -public final class Log4j2Instrumentation extends Instrumenter.Default { - public Log4j2Instrumentation() { - super("log4j2", "log4j", "log4j2.13.2"); +public final class Log4j2MdcInstrumentation extends Instrumenter.Default { + public Log4j2MdcInstrumentation() { + super("log4j2", "log4j", "log4j-2.13.2"); } @Override @@ -53,6 +43,13 @@ public ElementMatcher typeMatcher() { return named("org.apache.logging.log4j.core.impl.ThreadContextDataInjector"); } + @Override + public String[] helperResourceNames() { + return new String[] { + "META-INF/services/org.apache.logging.log4j.core.util.ContextDataProvider", + }; + } + @Override public String[] helperClassNames() { return new String[] { @@ -62,31 +59,7 @@ public String[] helperClassNames() { @Override public Map, String> transformers() { - return Collections.singletonMap( - isMethod() - .and(isPrivate()) - .and(isStatic()) - .and(named("getProviders")) - .and(takesArguments(0)), - Log4j2Instrumentation.class.getName() + "$GetProvidersAdvice"); - } - - public static class GetProvidersAdvice { - @Advice.OnMethodExit(suppress = Throwable.class) - public static void onExit( - @Advice.Return(typing = Typing.DYNAMIC, readOnly = false) - List providers) { - // check if already instrumented - for (ContextDataProvider provider : providers) { - if (provider instanceof OpenTelemetryContextDataProvider) { - return; - } - } - - List instrumentedProviders = new ArrayList<>(providers.size() + 1); - instrumentedProviders.addAll(providers); - instrumentedProviders.add(new OpenTelemetryContextDataProvider()); - providers = instrumentedProviders; - } + // Nothing to instrument, injecting helper resource & class is enough + return Collections.emptyMap(); } } diff --git a/instrumentation/log4j/log4j-2.13.2/library/log4j-2.13.2-library.gradle b/instrumentation/log4j/log4j-2.13.2/library/log4j-2.13.2-library.gradle index 2945fa9dd8c8..ec0fa655eebc 100644 --- a/instrumentation/log4j/log4j-2.13.2/library/log4j-2.13.2-library.gradle +++ b/instrumentation/log4j/log4j-2.13.2/library/log4j-2.13.2-library.gradle @@ -7,8 +7,5 @@ apply from: "$rootDir/gradle/instrumentation-library.gradle" dependencies { library group: 'org.apache.logging.log4j', name: 'log4j-core', version: '2.13.2' - annotationProcessor deps.autoservice - compileOnly deps.autoservice - testImplementation project(':instrumentation:log4j:log4j-2-testing') } diff --git a/instrumentation/log4j/log4j-2.13.2/library/src/main/java/io/opentelemetry/instrumentation/log4j/v2_13_2/OpenTelemetryContextDataProvider.java b/instrumentation/log4j/log4j-2.13.2/library/src/main/java/io/opentelemetry/instrumentation/log4j/v2_13_2/OpenTelemetryContextDataProvider.java index f9648cf0f157..32bc978e8117 100644 --- a/instrumentation/log4j/log4j-2.13.2/library/src/main/java/io/opentelemetry/instrumentation/log4j/v2_13_2/OpenTelemetryContextDataProvider.java +++ b/instrumentation/log4j/log4j-2.13.2/library/src/main/java/io/opentelemetry/instrumentation/log4j/v2_13_2/OpenTelemetryContextDataProvider.java @@ -16,7 +16,10 @@ package io.opentelemetry.instrumentation.log4j.v2_13_2; -import com.google.auto.service.AutoService; +import static io.opentelemetry.instrumentation.api.log.MdcConstants.SAMPLED; +import static io.opentelemetry.instrumentation.api.log.MdcConstants.SPAN_ID; +import static io.opentelemetry.instrumentation.api.log.MdcConstants.TRACE_ID; + import io.opentelemetry.trace.Span; import io.opentelemetry.trace.SpanContext; import io.opentelemetry.trace.TracingContextUtils; @@ -29,7 +32,6 @@ * Implementation of Log4j 2's {@link ContextDataProvider} which is loaded via SPI. {@link * #supplyContextData()} is called when a log entry is created. */ -@AutoService(ContextDataProvider.class) public class OpenTelemetryContextDataProvider implements ContextDataProvider { /** @@ -47,10 +49,10 @@ public Map supplyContextData() { Map contextData = new HashMap<>(); SpanContext spanContext = currentSpan.getContext(); - contextData.put("traceId", spanContext.getTraceIdAsHexString()); - contextData.put("spanId", spanContext.getSpanIdAsHexString()); + contextData.put(TRACE_ID, spanContext.getTraceIdAsHexString()); + contextData.put(SPAN_ID, spanContext.getSpanIdAsHexString()); if (spanContext.isSampled()) { - contextData.put("sampled", "true"); + contextData.put(SAMPLED, "true"); } return contextData; } diff --git a/instrumentation/log4j/log4j-2.13.2/library/src/main/resources/META-INF/services/org.apache.logging.log4j.core.util.ContextDataProvider b/instrumentation/log4j/log4j-2.13.2/library/src/main/resources/META-INF/services/org.apache.logging.log4j.core.util.ContextDataProvider new file mode 100644 index 000000000000..e4eb8009d0b4 --- /dev/null +++ b/instrumentation/log4j/log4j-2.13.2/library/src/main/resources/META-INF/services/org.apache.logging.log4j.core.util.ContextDataProvider @@ -0,0 +1 @@ +io.opentelemetry.instrumentation.log4j.v2_13_2.OpenTelemetryContextDataProvider diff --git a/instrumentation/log4j/log4j-2.7/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_7/Log4j27Instrumentation.java b/instrumentation/log4j/log4j-2.7/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_7/Log4j27MdcInstrumentation.java similarity index 79% rename from instrumentation/log4j/log4j-2.7/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_7/Log4j27Instrumentation.java rename to instrumentation/log4j/log4j-2.7/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_7/Log4j27MdcInstrumentation.java index 8c4b435d3b68..849e0ab99e7a 100644 --- a/instrumentation/log4j/log4j-2.7/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_7/Log4j27Instrumentation.java +++ b/instrumentation/log4j/log4j-2.7/src/main/java/io/opentelemetry/instrumentation/auto/log4j/v2_7/Log4j27MdcInstrumentation.java @@ -16,6 +16,9 @@ package io.opentelemetry.instrumentation.auto.log4j.v2_7; +import static io.opentelemetry.instrumentation.api.log.MdcConstants.SAMPLED; +import static io.opentelemetry.instrumentation.api.log.MdcConstants.SPAN_ID; +import static io.opentelemetry.instrumentation.api.log.MdcConstants.TRACE_ID; import static io.opentelemetry.javaagent.tooling.ClassLoaderMatcher.hasClassesNamed; import static io.opentelemetry.javaagent.tooling.bytebuddy.matcher.AgentElementMatchers.implementsInterface; import static net.bytebuddy.matcher.ElementMatchers.isMethod; @@ -38,9 +41,9 @@ import org.apache.logging.log4j.util.StringMap; @AutoService(Instrumenter.class) -public class Log4j27Instrumentation extends Instrumenter.Default { - public Log4j27Instrumentation() { - super("log4j2", "log4j", "log4j2.7"); +public class Log4j27MdcInstrumentation extends Instrumenter.Default { + public Log4j27MdcInstrumentation() { + super("log4j2", "log4j", "log4j-2.7"); } @Override @@ -59,7 +62,7 @@ public Map, String> transfor isMethod() .and(named("injectContextData")) .and(returns(named("org.apache.logging.log4j.util.StringMap"))), - Log4j27Instrumentation.class.getName() + "$InjectContextDataAdvice"); + Log4j27MdcInstrumentation.class.getName() + "$InjectContextDataAdvice"); } public static class InjectContextDataAdvice { @@ -71,15 +74,17 @@ public static void onExit( return; } - if (contextData.containsKey("traceId")) { + if (contextData.containsKey(TRACE_ID)) { // Assume already instrumented event if traceId is present. return; } StringMap newContextData = new SortedArrayStringMap(contextData); - newContextData.putValue("traceId", currentContext.getTraceId().toLowerBase16()); - newContextData.putValue("spanId", currentContext.getSpanId().toLowerBase16()); - newContextData.putValue("traceFlags", currentContext.getTraceFlags().toLowerBase16()); + newContextData.putValue(TRACE_ID, currentContext.getTraceIdAsHexString()); + newContextData.putValue(SPAN_ID, currentContext.getSpanIdAsHexString()); + if (currentContext.isSampled()) { + newContextData.putValue(SAMPLED, "true"); + } contextData = newContextData; } } diff --git a/instrumentation/logback/logback-1.0.0/library/src/main/java/io/opentelemetry/instrumentation/logback/v1_0_0/OpenTelemetryAppender.java b/instrumentation/logback/logback-1.0.0/library/src/main/java/io/opentelemetry/instrumentation/logback/v1_0_0/OpenTelemetryAppender.java index e83defcf96ff..877d00e5ac31 100644 --- a/instrumentation/logback/logback-1.0.0/library/src/main/java/io/opentelemetry/instrumentation/logback/v1_0_0/OpenTelemetryAppender.java +++ b/instrumentation/logback/logback-1.0.0/library/src/main/java/io/opentelemetry/instrumentation/logback/v1_0_0/OpenTelemetryAppender.java @@ -16,6 +16,10 @@ package io.opentelemetry.instrumentation.logback.v1_0_0; +import static io.opentelemetry.instrumentation.api.log.MdcConstants.SAMPLED; +import static io.opentelemetry.instrumentation.api.log.MdcConstants.SPAN_ID; +import static io.opentelemetry.instrumentation.api.log.MdcConstants.TRACE_ID; + import ch.qos.logback.classic.spi.ILoggingEvent; import ch.qos.logback.core.Appender; import ch.qos.logback.core.UnsynchronizedAppenderBase; @@ -40,17 +44,17 @@ public static ILoggingEvent wrapEvent(ILoggingEvent event) { } Map eventContext = event.getMDCPropertyMap(); - if (eventContext != null && eventContext.containsKey("traceId")) { + if (eventContext != null && eventContext.containsKey(TRACE_ID)) { // Assume already instrumented event if traceId is present. return event; } Map contextData = new HashMap<>(); SpanContext spanContext = currentSpan.getContext(); - contextData.put("traceId", spanContext.getTraceIdAsHexString()); - contextData.put("spanId", spanContext.getSpanIdAsHexString()); + contextData.put(TRACE_ID, spanContext.getTraceIdAsHexString()); + contextData.put(SPAN_ID, spanContext.getSpanIdAsHexString()); if (spanContext.isSampled()) { - contextData.put("sampled", "true"); + contextData.put(SAMPLED, "true"); } if (eventContext == null) {