Skip to content

Commit 4b03e98

Browse files
make internal-reflection indy compatible (#12136)
Co-authored-by: Jonas Kunz <[email protected]>
1 parent 182e938 commit 4b03e98

File tree

4 files changed

+44
-16
lines changed

4 files changed

+44
-16
lines changed

instrumentation/internal/internal-reflection/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/internal/reflection/ReflectionInstrumentation.java

+14-8
Original file line numberDiff line numberDiff line change
@@ -53,21 +53,27 @@ public void transform(TypeTransformer transformer) {
5353

5454
@SuppressWarnings("unused")
5555
public static class FilterFieldsAdvice {
56+
57+
// using AsScalar is needed to return the array itself instead of "advice Object[] return"
58+
@Advice.AssignReturned.ToReturned
59+
@Advice.AssignReturned.AsScalar
5660
@Advice.OnMethodExit(suppress = Throwable.class)
57-
public static void filter(
58-
@Advice.Argument(0) Class<?> containingClass,
59-
@Advice.Return(readOnly = false) Field[] fields) {
60-
fields = ReflectionHelper.filterFields(containingClass, fields);
61+
public static Field[] filter(
62+
@Advice.Argument(0) Class<?> containingClass, @Advice.Return Field[] fields) {
63+
return ReflectionHelper.filterFields(containingClass, fields);
6164
}
6265
}
6366

6467
@SuppressWarnings("unused")
6568
public static class FilterMethodsAdvice {
69+
70+
// using AsScalar is needed to return the array itself instead of "advice Object[] return"
71+
@Advice.AssignReturned.ToReturned
72+
@Advice.AssignReturned.AsScalar
6673
@Advice.OnMethodExit(suppress = Throwable.class)
67-
public static void filter(
68-
@Advice.Argument(0) Class<?> containingClass,
69-
@Advice.Return(readOnly = false) Method[] methods) {
70-
methods = ReflectionHelper.filterMethods(containingClass, methods);
74+
public static Method[] filter(
75+
@Advice.Argument(0) Class<?> containingClass, @Advice.Return Method[] methods) {
76+
return ReflectionHelper.filterMethods(containingClass, methods);
7177
}
7278
}
7379
}

instrumentation/internal/internal-reflection/javaagent/src/main/java/io/opentelemetry/javaagent/instrumentation/internal/reflection/ReflectionInstrumentationModule.java

+15-5
Original file line numberDiff line numberDiff line change
@@ -10,11 +10,15 @@
1010
import com.google.auto.service.AutoService;
1111
import io.opentelemetry.javaagent.extension.instrumentation.InstrumentationModule;
1212
import io.opentelemetry.javaagent.extension.instrumentation.TypeInstrumentation;
13+
import io.opentelemetry.javaagent.extension.instrumentation.internal.ExperimentalInstrumentationModule;
14+
import io.opentelemetry.javaagent.extension.instrumentation.internal.injection.ClassInjector;
15+
import io.opentelemetry.javaagent.extension.instrumentation.internal.injection.InjectionMode;
1316
import io.opentelemetry.sdk.autoconfigure.spi.ConfigProperties;
1417
import java.util.List;
1518

1619
@AutoService(InstrumentationModule.class)
17-
public class ReflectionInstrumentationModule extends InstrumentationModule {
20+
public class ReflectionInstrumentationModule extends InstrumentationModule
21+
implements ExperimentalInstrumentationModule {
1822
public ReflectionInstrumentationModule() {
1923
super("internal-reflection");
2024
}
@@ -26,12 +30,18 @@ public boolean defaultEnabled(ConfigProperties config) {
2630
}
2731

2832
@Override
29-
public boolean isIndyModule() {
30-
return false;
33+
public List<TypeInstrumentation> typeInstrumentations() {
34+
return asList(new ClassInstrumentation(), new ReflectionInstrumentation());
3135
}
3236

3337
@Override
34-
public List<TypeInstrumentation> typeInstrumentations() {
35-
return asList(new ClassInstrumentation(), new ReflectionInstrumentation());
38+
public void injectClasses(ClassInjector injector) {
39+
// we do not use ByteBuddy Advice dispatching in this instrumentation
40+
// Instead, we manually call ReflectionHelper via ASM
41+
// Easiest solution to work with indy is to inject an indy-proxy to be invoked
42+
injector
43+
.proxyBuilder(
44+
"io.opentelemetry.javaagent.instrumentation.internal.reflection.ReflectionHelper")
45+
.inject(InjectionMode.CLASS_ONLY);
3646
}
3747
}

javaagent-tooling/src/main/java/io/opentelemetry/javaagent/tooling/instrumentation/indy/InstrumentationModuleClassLoader.java

+9-2
Original file line numberDiff line numberDiff line change
@@ -109,10 +109,17 @@ public MethodHandles.Lookup getLookup() {
109109
if (cachedLookup == null) {
110110
// Load the injected copy of LookupExposer and invoke it
111111
try {
112+
MethodType getLookupType = MethodType.methodType(MethodHandles.Lookup.class);
112113
// we don't mind the race condition causing the initialization to run multiple times here
113114
Class<?> lookupExposer = loadClass(LookupExposer.class.getName());
114-
cachedLookup = (MethodHandles.Lookup) lookupExposer.getMethod("getLookup").invoke(null);
115-
} catch (Exception e) {
115+
// Note: we must use MethodHandles instead of reflection here to avoid a recursion
116+
// for our internal ReflectionInstrumentationModule which instruments reflection methods
117+
cachedLookup =
118+
(MethodHandles.Lookup)
119+
MethodHandles.publicLookup()
120+
.findStatic(lookupExposer, "getLookup", getLookupType)
121+
.invoke();
122+
} catch (Throwable e) {
116123
throw new IllegalStateException(e);
117124
}
118125
}

muzzle/src/main/java/io/opentelemetry/javaagent/tooling/HelperInjector.java

+6-1
Original file line numberDiff line numberDiff line change
@@ -172,7 +172,8 @@ public DynamicType.Builder<?> transform(
172172
injectedClassLoaders.computeIfAbsent(
173173
maskNullClassLoader(classLoader),
174174
cl -> {
175-
List<HelperClassDefinition> helpers = helperClassesGenerator.apply(cl);
175+
List<HelperClassDefinition> helpers =
176+
helperClassesGenerator.apply(unmaskNullClassLoader(cl));
176177

177178
LinkedHashMap<String, Supplier<byte[]>> classesToInject =
178179
helpers.stream()
@@ -355,6 +356,10 @@ private static ClassLoader maskNullClassLoader(ClassLoader classLoader) {
355356
return classLoader != null ? classLoader : BOOTSTRAP_CLASSLOADER_PLACEHOLDER;
356357
}
357358

359+
private static ClassLoader unmaskNullClassLoader(ClassLoader classLoader) {
360+
return isBootClassLoader(classLoader) ? null : classLoader;
361+
}
362+
358363
private static boolean isBootClassLoader(ClassLoader classLoader) {
359364
return classLoader == BOOTSTRAP_CLASSLOADER_PLACEHOLDER;
360365
}

0 commit comments

Comments
 (0)