Fluent Mock 入门二

556 阅读5分钟

使用配置介绍

MockUp具体使用介绍

为了讲述如何使用MockUp,我们先给出业务类,这个类有static,final,private,public等方法。

/**
 * 一个普通的业务类, 有各式限定方法
 */
@Getter
public class AnNormalService {
    /**
     * 静态变量
     */
    private static int staticInt;
    /**
     * 成员变量
     */
    private int anInt;

    /**
     * 静态代码块
     */
    static {
        setStaticInt(200);
    }

    /**
     * 构造函数
     *
     * @param anInt
     */
    public AnNormalService(int anInt) {
        this.anInt = anInt;
    }

    /**
     * 静态方法
     *
     * @param num
     */
    public static void setStaticInt(int num) {
        staticInt = num;
    }

    public static int getStaticInt() {
        return staticInt;
    }

    /**
     * 普通方法
     *
     * @return
     */
    public int publicMethod() {
        return this.privateMethod();
    }

    /**
     * private方法
     *
     * @return
     */
    private int privateMethod() {
        return anInt;
    }

    /**
     * final方法
     *
     * @return
     */
    public final int finalMethod() {
        return this.protectedMethod();
    }

    /**
     * protected方法
     *
     * @return
     */
    private int protectedMethod() {
        return anInt;
    }
}

各种限定类型方法mock演示

对静态代码块的mock

静态代码块对应的mock方法名为 $clinit(), 无参数

public class AnNormalServiceMockUpTest {
    @BeforeAll
    static void setup() {
        new MockUp<AnNormalService>() {
            /**
             * 对静态代码块的mock
             */
            @Mock
            void $clinit() {
                AnNormalService.setStaticInt(300);
            }
        };

    }

    @DisplayName("验证静态代码块mock")
    @Test
    void test1() {
        int anInt = AnNormalService.getStaticInt();
        Assertions.assertEquals(300, anInt);
    }
}

对构造函数进行mock

要mock构造函数,使用方法名 $init, 具体的入参同构造函数, 为了引用被mock的实例对象, 可以在入参列表前增加入参Invocation。

public class AnNormalServiceMockUpTest {
    
    @DisplayName("对构造函数进行mock")
    @Test
    void test2() {
        new MockUp<AnNormalService>() {
            @Mock
            void $init(Invocation inv, int anInt) {
                AnNormalService self = inv.getTarget();
                /**
                 * 成员变量anInt = 输入值 + 200
                 */
                self.setAnInt(anInt + 200);
            }
        };
        int anInt = new AnNormalService(200).publicMethod();
        Assertions.assertEquals(400, anInt);
    }
}

mock public和final方法

对任意方法的mock很简单,在MockUp类里面定义同名函数,入参类型一致,就可以改变方法的行为。

public class AnNormalServiceMockUpTest {
    
    @DisplayName("演示对public和final方法mock")
    @Test
    void test3() {
        new MockUp<AnNormalService>() {
            @Mock
            int publicMethod() {
                return 5;
            }

            @Mock
            int finalMethod() {
                return 6;
            }
        };
        AnNormalService service = new AnNormalService(400);
        Assertions.assertEquals(5, service.publicMethod());
        Assertions.assertEquals(6, service.finalMethod());
    }


    @DisplayName("演示对private和protected方法mock")
    @Test
    void test4() {
        new MockUp<AnNormalService>() {
            @Mock
            int privateMethod() {
                return 9;
            }

            @Mock
            int protectedMethod() {
                return 11;
            }
        };
        AnNormalService service = new AnNormalService(400);
        Assertions.assertEquals(9, service.publicMethod());
        Assertions.assertEquals(11, service.finalMethod());
    }   
 
    @DisplayName("演示对静态方法的mock")
    @Test
    void test5() {
        new MockUp<AnNormalService>() {
            @Mock
            int getStaticInt() {
                return 178;
            }
        };
        Assertions.assertEquals(178, AnNormalService.getStaticInt());
    }
}

使用Invocation来指向mock实例对象

FluentMock提供了Invocation类型,可以作为mock方法的第一个参数。 Invocation提供了下面方法供mock使用:

/**
 * Invocation类型可以作为mock方法的第一个参数
 */
public abstract class Invocation {
    /**
     * 执行方法原有的逻辑
     *
     * @param args 传递给方法的参数(可以是原始入参, 也可以是mock逻辑中篡改过的参数)
     */
    public <T> T proceed(Object... args);

    /**
     * 执行方法mock前的逻辑, 入参是原始入参
     */
    public <T> T proceed();

    /**
     * 当前执行实例(即相当于Java内置的变量this), 静态方法返回null
     */
    public abstract <T> T getTarget();

    /**
     * 获得方法的入参列表
     */
    public abstract Object[] getArgs();

    /**
     * 获得第index个参数, index从0开始
     */
    public Object arg(int index);

    /**
     * 获得第index个参数, index从0开始; 并强制转换为类型ARG
     *
     * @param index
     * @param clazz 强转类型
     */
    public <ARG> ARG arg(int index, Class<ARG> clazz);

    /**
     * 返回方法是被第几次调用
     */
    public abstract int getInvokedTimes();
}

Invocation在mock中作用很大,可以做很多事情

执行原逻辑

比如有业务类如下:

public class Service {
    private String value = "origin string";

    public String getString() {
        return value;
    }

    public void setString(String input) {
        this.value = input;
    }
}

mock方法getString,我们希望在原有的返回值上面追加信息, 可以像下面这样mock。

public class InvocationDemo {
    @DisplayName("演示Mock方法执行原方法逻辑")
    @Test
    void test1() {
        new MockUp<Service>() {
            @Mock
            String getString(Invocation inv) {
                String origin = inv.proceed();
                return origin + ", plus mock info.";
            }
        };
        String result = new Service().getString();
        Assertions.assertEquals("origin string, plus mock info.", result);
    }
}

按调用时序进行mock

有时候,我们对同一方法调用时,需要根据调用时序返回不同的结果。 比如,还是对上面的getString就行mock,希望前3次返回不同的值,其余的返回原方法返回值。

public class InvocationDemo {
    @DisplayName("演示按调用时序进行mock")
    @Test
    void test2() {
        new MockUp<Service>() {
            @Mock
            String getString(Invocation inv) {
                switch (inv.getInvokedTimes()) {
                    case 1: return "mock 1";
                    case 2: return "mock 2";
                    case 3: return "mock 3";
                    default: return inv.proceed();
                }
            }
        };
        Service service = new Service();
        assertEquals("mock 1", service.getString());
        assertEquals("mock 2", service.getString());
        assertEquals("mock 3", service.getString());
        assertEquals("origin string", service.getString());
        assertEquals("origin string", service.getString());
    }
}

对入参或出参进行断言

在测试过程中,除了需要对外部接口进行mock外。有时候还需要对入参和返回结果值进行判断,来验证我们的程序执行是否正常。

对入参进行断言

public class InvocationDemo {
    @DisplayName("对方法入参进行断言演示")
    @Test
    void test3() {
        new MockUp<Service>() {
            @Mock
            void setString(Invocation inv, String input) {
                Assertions.assertEquals("期望值", input);
                inv.proceed();
            }
        };
        Service service = new Service();
        // 正常通过
        service.setString("期望值");
        // 设置其它值, 应该是抛出一个断言异常
        Assertions.assertThrows(AssertionError.class, () -> service.setString("其它值"));
    }
}

对方法出参进行断言

public class InvocationDemo {
    @DisplayName("对方法出参进行断言演示")
    @Test
    void test4() {
        new MockUp<Service>() {
            @Mock
            String getString(Invocation inv) {
                String result = inv.proceed();
                Assertions.assertEquals("origin string", result);
                return result;
            }
        };
        Service service = new Service();
        /**
         * 方法调用正常通过
         */
        service.getString();
        /**
         * 将返回值改为其它值, 方法调用应该抛出断言异常
         */
        service.setString("other value");
        Assertions.assertThrows(AssertionError.class, () -> service.getString());
    }
}

对指定的对象实例进行Mock

上面我们演示的普适性的Mock, 及Mock的方法对所有实例生效(因为fluent mock修改的是类字节码实现),任何执行到对应方法的调用,都会进入mock后的逻辑。 但如果我们只想对特定的对象实例进行mock如何处理呢? 比如我们有个User类, User有2个成员变量:配偶和小孩列表.

@Data
@Accessors(chain = true)
public class User {
    private String name;

    private User spouse;

    private List<User> children = new ArrayList<>();

    public User(String name) {
        this.name = name;
    }

    public User addChild(User child) {
        children.add(child);
        return this;
    }

    /**
     * 自我介绍
     *
     * @return
     */
    public String sayHi() {
        StringBuilder buff = new StringBuilder();
        buff.append("I'm ").append(this.getName());
        if (this.spouse != null) {
            buff.append(" and my spouse is ").append(this.spouse.getName()).append(".");
        }
        if (!this.children.isEmpty()) {
            buff.append("I have ").append(children.size()).append(" children, ")
                .append(this.children.stream().map(User::getName).collect(Collectors.joining(" and ")))
                .append(".");
        }
        return buff.toString();
    }
}

现在我们演示一下如何对指定对象进行mock

public class TargetObjectMockDemo {

    @DisplayName("对指定实例进行mock演示")
    @Test
    void test1() {
        /** 有一个配偶, 2个小孩 **/
        User user = new User("tom")
            .setSpouse(new User("mary"))
            .addChild(new User("mike"))
            .addChild(new User("jack"));
        String hi = user.sayHi();
        /** mock前的自我介绍 **/
        assertEquals("I'm tom and my spouse is mary.I have 2 children, mike and jack.", hi);
        /**
         * 对配偶进行mock
         */
        new MockUp<User>(user.getSpouse()) {
            @Mock
            String getName(Invocation inv) {
                return "virtual " + inv.proceed();
            }
        };
        /**
         * 对小孩进行mock
         */
        new MockUp<User>(user.getChildren()) {
            @Mock
            String getName(Invocation inv) {
                return "fictitious " + inv.proceed();
            }
        };
        hi = user.sayHi();
        /** mock后的自我介绍 **/
        assertEquals("I'm tom and my spouse is virtual mary.I have 2 children, fictitious mike and fictitious jack.", hi);
    }
}

通过示例,我们看到"virtual"修饰只对配偶实例getName()方法起作用, "fictitious"修饰只对小孩的getName()起作用, 而user本尊的getName()没有受到影响。

链接

Fluent Mock开源地址

Fluent Mybatis开源地址