如何使用byte-buddy覆盖运行时对象的值

时间:2018-01-05 13:45:28

标签: java javaagents byte-buddy

这是我的对象类:

package com.example;

public class Car {

private String name;
private String model;

public String getName() {
    return name;
}
public void setName(String name) {
    this.name = name;
}
public String getModel() {
    return model;
}
public void setModel(String model) {
    this.model = model;
}

这是操作类:

package com.example;

public class Call1 {

public static String callMethod1(){
    return "Hello from callMethod1";
}

public static Car callingCall2(){
    Call2 call = new Call2();
    Car args= call.callMethod2();

    return args;
}
}

这是另一个类:

package com.example;

public class Call2 {

public  Car callMethod2(){

    Car car = new Car();
    car.setModel("2009");
    car.setName("mustang");
    return car;
}
}

这是我的测试类:

package com.example;

import net.bytebuddy.ByteBuddy;
import net.bytebuddy.agent.ByteBuddyAgent;
import net.bytebuddy.agent.builder.AgentBuilder;
import net.bytebuddy.agent.builder.AgentBuilder.RedefinitionStrategy;
import net.bytebuddy.asm.Advice;
import net.bytebuddy.asm.Advice.OnMethodEnter;
import net.bytebuddy.asm.AsmVisitorWrapper;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.dynamic.DynamicType;
import net.bytebuddy.dynamic.DynamicType.Builder;
import net.bytebuddy.dynamic.loading.ClassLoadingStrategy;
import net.bytebuddy.dynamic.loading.ClassReloadingStrategy;
import net.bytebuddy.implementation.FixedValue;
import net.bytebuddy.implementation.MethodDelegation;
import net.bytebuddy.matcher.ElementMatcher;
import net.bytebuddy.matcher.ElementMatchers;
import net.bytebuddy.utility.JavaModule;

import org.junit.Test;

import java.lang.instrument.Instrumentation;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;

import static net.bytebuddy.matcher.ElementMatchers.*;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;

import org.junit.Test;


public class CallingTest {

@Test
public void givenCall1_whenRedefined_thenReturnFooRedefined() throws 
Exception {

     premain(null, ByteBuddyAgent.install());
    /* Car carTest = new Car();
        carTest.setModel("2011");
        carTest.setName("Maruti");
   ByteBuddyAgent.install();
    new ByteBuddy()
        .redefine(Call1.class)
        .method(named("callingCall2"))
        .intercept(FixedValue.value(carTest))
        .make()
        .load(Car.class.getClassLoader(), 
ClassReloadingStrategy.fromInstalledAgent());*/
     Call1 f = new Call1();
    assertEquals(f.callingCall2().getModel(), "2011");
    assertEquals(f.callingCall2().getName(), "Maruti");
}

 public static void premain(String arguments, Instrumentation instrumentation) {
    new AgentBuilder.Default()
        .disableClassFormatChanges()
        .with(RedefinitionStrategy.RETRANSFORMATION)
        .type(is(Call1.class))
        .transform(new AgentBuilder.Transformer() {
/*          @Override
      public DynamicType.Builder transform(DynamicType.Builder builder,
                                              TypeDescription typeDescription,
                                              ClassLoader classloader) {
        return builder.method(named("toString"))
                      .intercept(FixedValue.value("transformed"));
      }*/

    @Override
    public Builder<?> transform(Builder<?> builder, TypeDescription typeDescription,
            ClassLoader classLoader, JavaModule arg3) {
        Car carTest = new Car();
        carTest.setModel("2011");
        carTest.setName("Maruti");

        return builder.method(named("callingCall2")).intercept(MethodDelegation.to(MyAdvice.class));
//      return builder.visit((AsmVisitorWrapper) Advice.to(Advice.class).on(ElementMatchers.named("callingCall2")));

    }
    }).installOn(instrumentation);
  }
 class MyAdvice {
  @OnMethodEnter 
  Car foo() {
        Car carTest = new Car();
        carTest.setModel("2011");
        carTest.setName("Maruti");
        return carTest;
  }

  }

}

要求是在此场景中覆盖对象Car的值,以便测试类应该通过。我尝试使用byte buddy和AgentBuilder这样做。当我尝试使用重新定义时,它会抛出错误 &#39; java.lang.UnsupportedOperationException:类重定义失败:尝试更改架构(添加/删除字段)&#39; 当我尝试使用AgentBuilder时,它不会影响断言,从而导致失败。  我刚接触到byte-buddy所以请帮助达到要求。

1 个答案:

答案 0 :(得分:1)

FixedValue 只能与Strings,Class对象,基元类型及其包装器一起使用。覆盖你的案例你可以像这样介绍一个拦截器:

@Test
public void givenCall1_whenRedefined_thenReturnFooRedefined() throws Exception {
    ByteBuddyAgent.install();

    new ByteBuddy()
        .redefine(Call1.class)
        .method(named("callingCall2"))
        .intercept(to(Interceptor.class))
        .make()
        .load(ByteBuddyTest2.class.getClassLoader(),
              ClassReloadingStrategy.fromInstalledAgent());

    Call1 f = new Call1();
    assertEquals(f.callingCall2().getModel(), "2011");
    assertEquals(f.callingCall2().getName(), "Maruti");
}

public static class Interceptor {
    public static Car callingCall2() {
      Car carTest = new Car();
      carTest.setModel("2011");
      carTest.setName("Maruti");
      return carTest;
    }
}