Retrofit自定义CallAdapter完成动态mock数据Mock

上一篇文章讲述了如何搭建一个Mock服务,并在最后留下了一个还没有实现的功能,就是通过Retrofit + 自定义CallAdapter + 注解 + 反射 完成一个可以在真实服务和测Mock服务之间动态切换的组件

关于如何搭建一个Mock Server,可以查看上一篇文章:PostMan mock数据,这里就不做过多的介绍了。

实现的效果

Retrofit通过接口和抽象方法来管理网络请求,希望在不改变这个接口中抽象方法的返回值(继续使用Observable返回值)的情况下(如果返回值改变了,代码需要大改,这并不是希望看到的结果)实现Mock的功能,并且能通过开关配置Mock的开关,在Release环境强制的关掉mock功能。

分析思路

主要的实现思路可以划分为两部分,

  • 第一部分:标识方法,并携带一定的参数信息;(自定义注解)
  • 第二部分:拦截到网络请求发起的地方,把Mock的信息塞进去。
实现第一部分

实现对方法的标识,并携带Mock网络请求的url,优秀的程序员马上想到了自定义注解,通过注解的方式很容易的就实现了这个需求;

下面是自定义注解的代码,因为我这里设计的mock和真实的网络请求的差别仅在相对的url上,就像下面这样

Mock网络请求的url : mock/user/info

真实网络请求的url : user/info

所以只需要携带mock后的url参数即可完成后面参数值的替换;

// 设置RUNTIME级别配合后面的反射,并要求只能在Method上使用该注解
@Retention(RetentionPolicy.RUNTIME)
public @Target({ElementType.METHOD})
@interface Mock {
    String mockUrl();
}
复制代码

以上,第一部分的工作就完成了,需求明确,实现简单;

实现第二个部分

第一个部分完成了自定义直接,第二个部分的需求是这个样子的

  1. 需要配置开关,在release环境关闭mock功能,避免忘记删除注解导致生产事故;
  2. 能够拿到方法上的注解信息
  3. 要保留之前的返回值类型,这样可以减少代码的改动量;
  4. 需要拦截到真实的网络请求,把mockUrl塞进去,替换掉原来的值。

如何才能实现上面的要求呢,这里是这样设计的(不要问我为什么这样设计,是真的想不起来是如何分析的);

  • 自定义一个CallAdapter.Factory的实现类,这个实现类配置了mock开关和生产模式开关,只有在非生产模式并打开了mock开关的情况下才工作,完成第1点;

  • CallAdapter.Factory的get()函数中,将方法上的注解通过数组传递了过来;完成第2点;

  • 自定义CallAdapter中持有一个RxJava2CallAdapterFactory在非mock的情况下,保证返回值不修改的情况下发起真实的网络请求,在mock的情况下,将mockurl保存到自定义CallAdapter里面并且也模仿RxJavaCallAdapter的写法实现一次,完成第3点;

  • 在CallAdapter里面有一个adapt()方法,会传入一个Call类型的OKHttpCall对象,这个对象是负责组装网络请求的对象,里面有一个requestFactory对象,这个对象存储了网络请求的一些参数,通过反射的方式把mockUrl替换掉requestFactory对象的relativeUrl即可完成mock,完成第4点。

按照上面的分析思路,一步步的实现代码

配置开关

自定义一个CallAdapter.Factory,并配置好开关配置 isRelease开关标识当前的开发环境是否为生产,mockEnable开关标识当前是否打开mock开关

public class MockRxJava2CallAdapterFactory extends CallAdapter.Factory {
    private boolean mockEnable;
    private boolean isRelease;
    private RxJava2CallAdapterFactory rxJava2CallAdapterFactory;

    private MockRxJava2CallAdapterFactory(boolean mockEnable, boolean isRelease) {
        this.mockEnable = mockEnable;
        this.isRelease = isRelease;
        rxJava2CallAdapterFactory = RxJava2CallAdapterFactory.create();
    }

    public static MockRxJava2CallAdapterFactory create(boolean mockEnable, boolean isRelease) {
        return new MockRxJava2CallAdapterFactory(mockEnable, isRelease);
    }
}
复制代码
重写get()方法,拿到Mock注解中的mockUrl

通过annotations数组,拿到Mock注解,并拿到mockUrl字符串返回

@Nullable
@Override
public CallAdapter<?, ?> get(Type returnType, Annotation[] annotations, Retrofit retrofit) {
    String mockUrl = MockUtil.getMockUrl(annotations);
    return null;
}


public static String getMockUrl(Annotation[] annotations) {
    for (Annotation ann : annotations) {
        if (ann.annotationType() == Mock.class) {
            return ((Mock) ann).mockUrl();
        }
    }
    return null;
}

复制代码
实现返回CallAdapter的逻辑

当mock开关打开,并且是非生产环境,且方法上的mockUrl有值的情况下,返回自定义的CallAdapter,否则交给RxJava2CallAdapterFactory处理。

@Nullable
@Override
public CallAdapter<?, ?> get(Type returnType, Annotation[] annotations, Retrofit retrofit) {
    String mockUrl = MockUtil.getMockUrl(annotations);
    if (mockEnable && !isRelease && mockUrl != null) {
        return getCallAdapter(returnType, mockUrl);
    } else {
        return rxJava2CallAdapterFactory.get(returnType, annotations, retrofit);
    }
}
复制代码
实现getCallAdapter()方法

这里是模仿RxJava2CallAdapter的写法,做了一些微调,主要是返回的自定义CallAdapter类

public MockCallAdapter getCallAdapter(Type returnType, String mockUrl) {
    Class<?> rawType = MockUtil.getRawType(returnType);
    if (rawType == Completable.class) {
        // Completable is not parameterized (which is what the rest of this method deals with) so it
        // can only be created with a single configuration.
        return new MockCallAdapter(Void.class, false, true, false, false,
                false, true, mockUrl);
    }
    boolean isFlowable = rawType == Flowable.class;
    boolean isSingle = rawType == Single.class;
    boolean isMaybe = rawType == Maybe.class;
    if (rawType != Observable.class && !isFlowable && !isSingle && !isMaybe) {
        return null;
    }

    boolean isResult = false;
    boolean isBody = false;
    Type responseType;
    if (!(returnType instanceof ParameterizedType)) {
        String name = isFlowable ? "Flowable"
                : isSingle ? "Single"
                : isMaybe ? "Maybe" : "Observable";
        throw new IllegalStateException(name + " return type must be parameterized"
                + " as " + name + "<Foo> or " + name + "<? extends Foo>");
    }

    Type observableType = getParameterUpperBound(0, (ParameterizedType) returnType);
    Class<?> rawObservableType = getRawType(observableType);
    if (rawObservableType == Response.class) {
        if (!(observableType instanceof ParameterizedType)) {
            throw new IllegalStateException("Response must be parameterized"
                    + " as Response<Foo> or Response<? extends Foo>");
        }
        responseType = getParameterUpperBound(0, (ParameterizedType) observableType);
    } else if (rawObservableType == Result.class) {
        if (!(observableType instanceof ParameterizedType)) {
            throw new IllegalStateException("Result must be parameterized"
                    + " as Result<Foo> or Result<? extends Foo>");
        }
        responseType = getParameterUpperBound(0, (ParameterizedType) observableType);
        isResult = true;
    } else {
        responseType = observableType;
        isBody = true;
    }

    return new MockCallAdapter(responseType, isResult, isBody, isFlowable,
            isSingle, isMaybe, false, mockUrl);
}
复制代码
自定义CallAdapter

自定义CallAdapter也同样的模仿RxJava2CallAdapter类(直接CV,然后微调) 主要是把拿到的mockUrl替换掉原来的网络请求地址,所以下面里面的关键是MockUtil.mockUrl()方法的实现; 当然模仿RxJava2CallAdapter的时候会出现找不到类的情况,原因是因为有几个类是没有访问权限的,还得模仿一次,主要有:BodyObservable、CallExecuteObservable、ResultObservable;(模仿的代码就不贴了)

// 模仿的RxJava2CallAdapter类
public final class MockCallAdapter<R> implements CallAdapter<R, Object> {
    private final Type responseType;
    private final boolean isResult;
    private final boolean isBody;
    private final boolean isFlowable;
    private final boolean isSingle;
    private final boolean isMaybe;
    private final boolean isCompletable;

    private final String mockUrl;

    MockCallAdapter(Type responseType, boolean isResult, boolean isBody, boolean isFlowable,
                    boolean isSingle, boolean isMaybe, boolean isCompletable, String mockUrl) {
        this.responseType = responseType;
        this.isResult = isResult;
        this.isBody = isBody;
        this.isFlowable = isFlowable;
        this.isSingle = isSingle;
        this.isMaybe = isMaybe;
        this.isCompletable = isCompletable;
        this.mockUrl = mockUrl;
    }

    @NonNull
    @Override
    public Type responseType() {
        return responseType;
    }

    @NonNull
    @Override
    public Object adapt(@NonNull Call<R> call) {
        // 将mockUrl替换掉原来的relativeUrl
        call = MockUtil.mockUrl(call, mockUrl);


        Observable<Response<R>> responseObservable = new CallExecuteObservable<>(call);

        Observable<?> observable;
        if (isResult) {
            observable = new ResultObservable<>(responseObservable);
        } else if (isBody) {
            observable = new BodyObservable<>(responseObservable);
        } else {
            observable = responseObservable;
        }

        if (isFlowable) {
            return observable.toFlowable(BackpressureStrategy.LATEST);
        }
        if (isSingle) {
            return observable.singleOrError();
        }
        if (isMaybe) {
            return observable.singleElement();
        }
        if (isCompletable) {
            return observable.ignoreElements();
        }
        return RxJavaPlugins.onAssembly(observable);
    }
}
复制代码
mockUrl()方法

拿到OKHttpCall对象中的requestFactory属性,再次反射去修改relativeUrl的值,并返回call对象。

public static <R> Call<R> mockUrl(Call<R> call, String mockUrl) {
    Class<? extends Call> clz = call.getClass();
    try {
        Field requestFactory = clz.getDeclaredField("requestFactory");
        requestFactory.setAccessible(true);
        Object o1 = requestFactory.get(call);
        Class<?> factoryClz = o1.getClass();
        Field relativeUrl = factoryClz.getDeclaredField("relativeUrl");
        relativeUrl.setAccessible(true);
        relativeUrl.set(o1, mockUrl);
    } catch (NoSuchFieldException | IllegalAccessException e) {
        e.printStackTrace();
    }

    return call;
}
复制代码

至此,完整的功能就实现了。

总结

其实这一块的内容,从想要实现什么样的效果,如何去保证不修改返回值,如何去找到反射的切入点都花费了我一天多的时间,但是奈何文化有效,不知道怎么把自己查看源码、分析源码的过程描述出来,所以写这篇文章的时候直接把最终实现的思路分析了一下,大致的实现贴了一下,当然这个代码的局限性还是比较大的,也还有值得优化的点留给我慢慢的思考,不过实现这个功能,对于注解、反射、Retrofit组装网络请求的理解还是有很大的帮助的。

猜你喜欢

转载自juejin.im/post/7104846835902054413