Skip to content

Commit

Permalink
Use env-entry to add tags per webapp deployment
Browse files Browse the repository at this point in the history
  • Loading branch information
amarziali committed Dec 31, 2024
1 parent da26e28 commit 74641ba
Show file tree
Hide file tree
Showing 27 changed files with 773 additions and 118 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
import com.google.auto.service.AutoService;
import datadog.trace.agent.tooling.Instrumenter;
import datadog.trace.agent.tooling.InstrumenterModule;
import datadog.trace.api.naming.ClassloaderServiceNames;
import datadog.trace.api.ClassloaderConfigurationOverrides;
import datadog.trace.bootstrap.AgentClassLoading;
import java.io.IOException;
import java.io.InputStream;
Expand Down Expand Up @@ -164,7 +164,8 @@ public static class CaptureModuleNameAdvice {
public static void afterConstruct(@Advice.This final Module module) {
final String name = ModuleNameHelper.extractDeploymentName(module.getClassLoader());
if (name != null && !name.isEmpty()) {
ClassloaderServiceNames.addServiceName(module.getClassLoader(), name);
ClassloaderConfigurationOverrides.addContextualInfo(
module.getClassLoader(), new ClassloaderConfigurationOverrides.ContextualInfo(name));
}
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,11 +16,11 @@
import com.ibm.wsspi.webcontainer.webapp.IWebAppDispatcherContext;
import datadog.trace.agent.tooling.Instrumenter;
import datadog.trace.agent.tooling.InstrumenterModule;
import datadog.trace.api.ClassloaderConfigurationOverrides;
import datadog.trace.api.Config;
import datadog.trace.api.CorrelationIdentifier;
import datadog.trace.api.GlobalTracer;
import datadog.trace.api.gateway.Flow;
import datadog.trace.api.naming.ClassloaderServiceNames;
import datadog.trace.bootstrap.ActiveSubsystems;
import datadog.trace.bootstrap.ContextStore;
import datadog.trace.bootstrap.InstrumentationContext;
Expand Down Expand Up @@ -115,7 +115,7 @@ public static class HandleRequestAdvice {
if (webapp != null) {
final ClassLoader cl = webapp.getClassLoader();
if (cl != null) {
ClassloaderServiceNames.maybeSetToSpan(span, cl);
ClassloaderConfigurationOverrides.maybeEnrichSpan(span, cl);
}
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
import com.ibm.ws.classloading.internal.ThreadContextClassLoader;
import datadog.trace.agent.tooling.Instrumenter;
import datadog.trace.agent.tooling.InstrumenterModule;
import datadog.trace.api.naming.ClassloaderServiceNames;
import datadog.trace.api.ClassloaderConfigurationOverrides;
import net.bytebuddy.asm.Advice;

@AutoService(InstrumenterModule.class)
Expand Down Expand Up @@ -40,7 +40,8 @@ public static class ThreadContextClassloaderAdvice {
public static void afterConstruct(@Advice.This ThreadContextClassLoader self) {
final String name = BundleNameHelper.extractDeploymentName(self);
if (name != null && !name.isEmpty()) {
ClassloaderServiceNames.addServiceName(self, name);
ClassloaderConfigurationOverrides.addContextualInfo(
self, new ClassloaderConfigurationOverrides.ContextualInfo(name));
}
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,11 +16,11 @@
import com.ibm.wsspi.webcontainer.webapp.IWebAppDispatcherContext;
import datadog.trace.agent.tooling.Instrumenter;
import datadog.trace.agent.tooling.InstrumenterModule;
import datadog.trace.api.ClassloaderConfigurationOverrides;
import datadog.trace.api.Config;
import datadog.trace.api.CorrelationIdentifier;
import datadog.trace.api.GlobalTracer;
import datadog.trace.api.gateway.Flow;
import datadog.trace.api.naming.ClassloaderServiceNames;
import datadog.trace.bootstrap.ActiveSubsystems;
import datadog.trace.bootstrap.ContextStore;
import datadog.trace.bootstrap.InstrumentationContext;
Expand Down Expand Up @@ -117,7 +117,7 @@ public static class HandleRequestAdvice {
if (webapp != null) {
final ClassLoader cl = webapp.getClassLoader();
if (cl != null) {
ClassloaderServiceNames.maybeSetToSpan(span, cl);
ClassloaderConfigurationOverrides.maybeEnrichSpan(span, cl);
}
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,12 +4,12 @@
import static datadog.trace.bootstrap.instrumentation.decorator.HttpServerDecorator.DD_SPAN_ATTRIBUTE;
import static datadog.trace.instrumentation.servlet2.Servlet2Decorator.DECORATE;

import datadog.trace.api.ClassloaderConfigurationOverrides;
import datadog.trace.api.Config;
import datadog.trace.api.CorrelationIdentifier;
import datadog.trace.api.DDTags;
import datadog.trace.api.GlobalTracer;
import datadog.trace.api.gateway.Flow;
import datadog.trace.api.naming.ClassloaderServiceNames;
import datadog.trace.bootstrap.InstrumentationContext;
import datadog.trace.bootstrap.instrumentation.api.AgentScope;
import datadog.trace.bootstrap.instrumentation.api.AgentSpan;
Expand Down Expand Up @@ -41,7 +41,7 @@ public static boolean onEnter(
final boolean hasServletTrace = spanAttr instanceof AgentSpan;
if (hasServletTrace) {
final AgentSpan span = (AgentSpan) spanAttr;
ClassloaderServiceNames.maybeSetToSpan(span);
ClassloaderConfigurationOverrides.maybeEnrichSpan(span);
// Tracing might already be applied by the FilterChain or a parent request (forward/include).
return false;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,12 +7,12 @@
import static datadog.trace.bootstrap.instrumentation.decorator.HttpServerDecorator.DD_SPAN_ATTRIBUTE;
import static datadog.trace.instrumentation.servlet3.Servlet3Decorator.DECORATE;

import datadog.trace.api.ClassloaderConfigurationOverrides;
import datadog.trace.api.Config;
import datadog.trace.api.CorrelationIdentifier;
import datadog.trace.api.DDTags;
import datadog.trace.api.GlobalTracer;
import datadog.trace.api.gateway.Flow;
import datadog.trace.api.naming.ClassloaderServiceNames;
import datadog.trace.bootstrap.instrumentation.api.AgentScope;
import datadog.trace.bootstrap.instrumentation.api.AgentSpan;
import datadog.trace.instrumentation.servlet.ServletBlockingHelper;
Expand Down Expand Up @@ -62,7 +62,7 @@ public static boolean onEnter(
final boolean hasServletTrace = spanAttrValue instanceof AgentSpan;
if (hasServletTrace) {
final AgentSpan span = (AgentSpan) spanAttrValue;
ClassloaderServiceNames.maybeSetToSpan(span);
ClassloaderConfigurationOverrides.maybeEnrichSpan(span);
// Tracing might already be applied by other instrumentation,
// the FilterChain or a parent request (forward/include).
return false;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
package datadog.trace.instrumentation.servlet3;

import datadog.trace.api.naming.ClassloaderServiceNames;
import datadog.trace.api.ClassloaderConfigurationOverrides;
import datadog.trace.bootstrap.instrumentation.api.AgentPropagation;
import datadog.trace.bootstrap.instrumentation.api.AgentSpan;
import datadog.trace.bootstrap.instrumentation.api.URIDataAdapter;
Expand Down Expand Up @@ -83,7 +83,7 @@ public AgentSpan onRequest(
final HttpServletRequest request,
AgentSpan.Context.Extracted context) {
assert span != null;
ClassloaderServiceNames.maybeSetToSpan(span);
ClassloaderConfigurationOverrides.maybeEnrichSpan(span);
if (request != null) {
String contextPath = request.getContextPath();
String servletPath = request.getServletPath();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,9 @@
import com.google.auto.service.AutoService;
import datadog.trace.agent.tooling.Instrumenter;
import datadog.trace.agent.tooling.InstrumenterModule;
import datadog.trace.api.ClassloaderConfigurationOverrides;
import datadog.trace.api.Config;
import datadog.trace.api.DDTags;
import datadog.trace.api.naming.ClassloaderServiceNames;
import datadog.trace.bootstrap.CallDepthThreadLocalMap;
import datadog.trace.bootstrap.instrumentation.api.AgentSpan;
import jakarta.servlet.ServletRequest;
Expand Down Expand Up @@ -61,7 +61,7 @@ public static AgentSpan before(@Advice.Argument(0) final ServletRequest request)
if (span instanceof AgentSpan
&& CallDepthThreadLocalMap.incrementCallDepth(HttpServletRequest.class) == 0) {
final AgentSpan agentSpan = (AgentSpan) span;
ClassloaderServiceNames.maybeSetToSpan(agentSpan);
ClassloaderConfigurationOverrides.maybeEnrichSpan(agentSpan);
return agentSpan;
}
return null;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import org.apache.catalina.core.StandardHost
import org.apache.catalina.startup.Tomcat
import org.apache.tomcat.JarScanFilter
import org.apache.tomcat.JarScanType
import org.apache.tomcat.util.descriptor.web.ContextEnvironment

class TomcatServer implements HttpServer {
def port = 0
Expand Down Expand Up @@ -38,7 +39,6 @@ class TomcatServer implements HttpServer {
return false
}
}

setupServlets(servletContext)

(server.host as StandardHost).errorReportValveClass = TomcatServletTest.ErrorHandlerValve.name
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import org.apache.catalina.connector.Request
import org.apache.catalina.connector.Response
import org.apache.catalina.startup.Tomcat
import org.apache.catalina.valves.ErrorReportValve
import org.apache.tomcat.util.descriptor.web.ContextEnvironment
import org.apache.tomcat.util.descriptor.web.FilterDef
import org.apache.tomcat.util.descriptor.web.FilterMap

Expand Down Expand Up @@ -213,4 +214,32 @@ class TomcatServletClassloaderNamingForkedTest extends TomcatServletTest {
}
}

class TomcatServletEnvEntriesTagTest extends TomcatServletTest {
def addEntry (context, name, value) {
def envEntry = new ContextEnvironment()
envEntry.setName(name)
envEntry.setValue(value)
envEntry.setType("java.lang.String")
context.getNamingResources().addEnvironment(envEntry)
}
@Override
protected void setupServlets(Context context) {
super.setupServlets(context)
addEntry(context, "datadog/tags/custom-tag", "custom-value")
addEntry(context, "java:comp/env/datadog/tags/service", "custom-service")
}

@Override
String expectedServiceName() {
"custom-service"
}

@Override
Map<String, Serializable> expectedExtraServerTags(ServerEndpoint endpoint) {
super.expectedExtraServerTags(endpoint) + ["custom-tag": "custom-value"] as Map<String, Serializable>
}
}




Original file line number Diff line number Diff line change
@@ -1,3 +1,11 @@
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.CUSTOM_EXCEPTION
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.ERROR
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.EXCEPTION
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.NOT_FOUND
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.REDIRECT
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.TIMEOUT_ERROR
import static org.junit.Assume.assumeTrue

import com.google.common.io.Files
import datadog.trace.agent.test.asserts.TraceAssert
import datadog.trace.agent.test.base.HttpServer
Expand All @@ -18,14 +26,6 @@ import org.apache.coyote.http11.Http11BaseProtocol
import javax.servlet.Servlet
import javax.servlet.ServletException

import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.CUSTOM_EXCEPTION
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.ERROR
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.EXCEPTION
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.NOT_FOUND
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.REDIRECT
import static datadog.trace.agent.test.base.HttpServerTest.ServerEndpoint.TIMEOUT_ERROR
import static org.junit.Assume.assumeTrue

abstract class TomcatServletTest extends AbstractServletTest<Embedded, Context> {

class TomcatServer implements HttpServer {
Expand Down Expand Up @@ -187,7 +187,7 @@ abstract class TomcatServletTest extends AbstractServletTest<Embedded, Context>
childOfPrevious()
tags {
"component" "java-web-servlet-response"
if ({isDataStreamsEnabled()}) {
if ({ isDataStreamsEnabled() }) {
"$DDTags.PATHWAY_HASH" { String }
}
defaultTags()
Expand Down Expand Up @@ -227,8 +227,8 @@ abstract class TomcatServletTest extends AbstractServletTest<Embedded, Context>
if (endpoint.throwsException) {
// Exception classes get wrapped in ServletException
["error.message": { endpoint == EXCEPTION ? "Servlet execution threw an exception" : it == endpoint.body },
"error.type": { it == ServletException.name || it == InputMismatchException.name },
"error.stack": String]
"error.type" : { it == ServletException.name || it == InputMismatchException.name },
"error.stack" : String]
} else {
Collections.emptyMap()
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,16 +1,19 @@
package datadog.trace.instrumentation.tomcat9;

import static datadog.trace.agent.tooling.bytebuddy.matcher.NameMatchers.named;
import static datadog.trace.api.ClassloaderConfigurationOverrides.DATADOG_TAGS_JNDI_PREFIX;
import static datadog.trace.api.ClassloaderConfigurationOverrides.DATADOG_TAGS_PREFIX;
import static net.bytebuddy.matcher.ElementMatchers.isMethod;

import com.google.auto.service.AutoService;
import datadog.trace.agent.tooling.Instrumenter;
import datadog.trace.agent.tooling.InstrumenterModule;
import datadog.trace.api.naming.ClassloaderServiceNames;
import datadog.trace.api.ClassloaderConfigurationOverrides;
import net.bytebuddy.asm.Advice;
import org.apache.catalina.Context;
import org.apache.catalina.WebResourceRoot;
import org.apache.catalina.loader.WebappClassLoaderBase;
import org.apache.tomcat.util.descriptor.web.ContextEnvironment;

@AutoService(InstrumenterModule.class)
public class WebappClassLoaderInstrumentation extends InstrumenterModule.Tracing
Expand All @@ -37,12 +40,44 @@ public static void onContextAvailable(
@Advice.Argument(0) final WebResourceRoot webResourceRoot) {
// at this moment we have the context set in this classloader, hence its name
final Context context = webResourceRoot.getContext();
if (context != null) {
final String contextName = context.getBaseName();
if (contextName != null && !contextName.isEmpty()) {
ClassloaderServiceNames.addServiceName(classLoader, contextName);
if (context == null) {
return;
}
ClassloaderConfigurationOverrides.ContextualInfo info = null;

final String contextName = context.getBaseName();
if (contextName != null && !contextName.isEmpty()) {
info = new ClassloaderConfigurationOverrides.ContextualInfo(contextName);
}
if (context.getNamingResources() != null) {
final ContextEnvironment[] envs = context.getNamingResources().findEnvironments();
if (envs != null) {
if (info == null) {
info = new ClassloaderConfigurationOverrides.ContextualInfo(null);
}
for (final ContextEnvironment env : envs) {
// as a limitation here we simplify a lot the logic and we do not try to resolve the
// typed value but we just take the string representation. It avoids implementing the
// logic to convert to other types
// (i.e. long, double) or instrument more deeply tomcat naming
if (env.getValue() == null || env.getValue().isEmpty()) {
continue;
}
String name = null;
if (env.getName().startsWith(DATADOG_TAGS_PREFIX)) {
name = env.getName().substring(DATADOG_TAGS_PREFIX.length());
} else if (env.getName().startsWith(DATADOG_TAGS_JNDI_PREFIX)) {
name = env.getName().substring(DATADOG_TAGS_JNDI_PREFIX.length());
}
if (name != null && !name.isEmpty()) {
info.addTag(name, env.getValue());
}
}
}
}
if (info != null) {
ClassloaderConfigurationOverrides.addContextualInfo(classLoader, info);
}
}
}
}
Loading

0 comments on commit 74641ba

Please sign in to comment.