Java-агент, использующий Byte-Buddy, не работает
Мне нужна ваша помощь в поиске, что не так с моей реализацией...
Я пытаюсь реализовать простой профилировщик времени выполнения JVM, используя byte-buddy. В общем, мне нужно, чтобы каждый вызов метода регистрировался в стеке, которым я управляю в отдельном объекте.
Прочитав несколько постов, я понял, что лучше использовать подход "Advise" вместо "MethodDelegation", и вот что я получил:
Agent.java:
package com.panaya.java.agent;
import net.bytebuddy.agent.builder.AgentBuilder;
import net.bytebuddy.asm.Advice;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.matcher.ElementMatchers;
import net.bytebuddy.utility.JavaModule;
import java.lang.instrument.Instrumentation;
import java.security.ProtectionDomain;
import java.util.ArrayList;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.regex.PatternSyntaxException;
public class Agent {
private static List<Pattern> pkgIncl = new ArrayList<Pattern>();
private static List<Pattern> pkgExcl = new ArrayList<Pattern>();
private static void initMatcherPatterns(String argument) {
if (argument == null) {
System.out.println("Missing configuration argument");
return;
}
System.out.println("Argument is: " + argument);
String[] tokens = argument.split(";");
if (tokens.length < 1) {
System.out.println("Missing delimeter ;");
return;
}
for (String token : tokens) {
String[] args = token.split("=");
if (args.length < 2) {
System.out.println("Missing argument delimeter =:" + token);
return;
}
String argtype = args[0];
if (!argtype.equals("incl") && !argtype.equals("excl")) {
System.out.println("Wrong argument: " + argtype);
return;
}
String[] patterns = args[1].split(",");
for (String pattern : patterns) {
Pattern p = null;
System.out.println("Compiling " + argtype + " pattern:" + pattern + "$");
try {
p = Pattern.compile(pattern + "$");
} catch (PatternSyntaxException pse) {
System.out.println("pattern: " + pattern + " not valid, ignoring");
}
if (argtype.equals("incl"))
pkgIncl.add(p);
else
pkgExcl.add(p);
}
}
}
private static boolean isShouldInstrumentClass(String className) {
System.out.println("Testing " + className + " for match.");
boolean match = false;
String name = className.replace("/", ".");
for (Pattern p : pkgIncl) {
Matcher m = p.matcher(name);
if (m.matches()) {
match = true;
break;
}
}
for (Pattern p : pkgExcl) {
Matcher m = p.matcher(name);
if (m.matches()) {
match = false;
break;
}
}
if (match) {
System.out.println("Class " + name + "should be instrumented.");
} else {
System.out.println("Skipping class: " + name);
}
return match;
}
public static void premain(String agentArgument, Instrumentation instrumentation) {
System.out.println("Premain started");
try {
initMatcherPatterns(agentArgument);
new AgentBuilder.Default()
.with(AgentBuilder.TypeStrategy.Default.REBASE)
.type(new AgentBuilder.RawMatcher() {
public boolean matches(TypeDescription typeDescription, ClassLoader classLoader, JavaModule javaModule, Class<?> aClass, ProtectionDomain protectionDomain) {
return isShouldInstrumentClass(typeDescription.getActualName());
}
})
.transform((builder, typeDescription, classLoader) -> builder
.visit(Advice.to(ProfilingAdvice.class)
.on(ElementMatchers.any()))).installOn(instrumentation);
} catch (RuntimeException e) {
System.out.println("Exception instrumenting code : " + e);
e.printStackTrace();
}
}
}
И ProfilingAdvice.java:
package com.panaya.java.agent;
import com.panaya.java.profiler.MethodStackCollector;
import net.bytebuddy.asm.Advice;
public class ProfilingAdvice {
@Advice.OnMethodEnter
public static void enter(@Advice.Origin("#t.#m") String signature) {
System.out.println("OnEnter :" + signature);
try {
MethodStackCollector.getInstance().push(signature);
} catch (Exception e) {
e.printStackTrace();
}
}
@Advice.OnMethodExit
public static void exit(@Advice.Return long value) {
System.out.println("OnExit - value = " + value);
try {
MethodStackCollector.getInstance().pop();
} catch (Exception e) {
e.printStackTrace();
}
}
}
По какой-то причине методы "enter" и "exit" в классе ProfilingAdvice вообще не вызываются.
Что я делаю неправильно?
Спасибо, Элад.
2 ответа
Я попробовал ваш пример и после того, как свел рекомендации по выходу к командам печати и вашему совпадению, чтобы соответствовать только некоторому классу foo.Bar
:
package foo;
public class Bar {
public long qux() { return 0L; }
}
приборы работают без проблем. Я нахожу немного подозрительным, что ваш Advice
совпадение указывает any()
в то время как ваш ProfilingAdvice
требует long
тип возврата. Вы пробовали печатать только на консоли без каких-либо комментариев в вашем совете?
Вы можете отладить такие проблемы, решив:
new AgentBuilder.Default()
.with(AgentBuilder.Listener.StreamWriting.toSystemOut());
где потенциальные ошибки во время контрольно-измерительной аппаратуры выводятся на консоль.
@Advice.Return
обозначает возвращаемое значение монитора меторд.
@Advice.Return
это не возвращаемое значение enter(@Advice.Origin("#t.#m") String signature)
метод.
Итак, следующий код в порядке:
public class ProfilerInterceptor {
public static ThreadLocal<Long> threadLocal = new ThreadLocal();
@Advice.OnMethodEnter
public static void enter(@Advice.Origin("#t.#m") String signature) {
System.out.printf("Enter: %s\n", signature);
long start = System.currentTimeMillis();
threadLocal.set(start);
try {
MethodStackCollector.getInstance().push(signature);
} catch (Exception e) {
e.printStackTrace();
}
}
@Advice.OnMethodExit
public static void exit(@Advice.Origin("#t.#m") String signature) {
long value = System.currentTimeMillis() - threadLocal.get();
System.out.printf("Exit: %s\nTime: %d\n\n", signature, value);
System.out.println("OnExit - value = " + value);
try {
MethodStackCollector.getInstance().pop();
} catch (Exception e) {
e.printStackTrace();
}
}
}