RxWeaver

Project Url: qingmei2/RxWeaver
Introduction: :baby_chick: A lightweight and flexible error handler tools of RxJava2 (一个轻量且灵活的 RxJava2 全局 Error 处理组件).
More: Author   ReportBugs   OfficialWebsite   
Tags:

关于这个 repo 的起源,请参考这篇文章:

不要打破链式调用!一个极低成本的 RxJava 全局 Error 处理方案

通知

  • 关于移除了 Gradle 依赖的方式(2019/3/18)

最新的代码中,我移除了 Gradle 依赖的方式,根本原因是,我认为这个 repo 被称为 业务逻辑的展示 更为确切——每个项目都有属于自己的特定业务,这些业务逻辑千变万化却又不离其宗。

因此我把最基本的逻辑抽了出来,放在了rxweaver这个Module里,然后又把复杂的业务逻辑放在了sample中进行展示。我希望您能够从我的代码中得到灵感,即使它的代码可能对您来说非常抽象。

同样也非常真诚的欢迎您宝贵的建议和讨论~

简介

关于RxJava的全局的 error 处理,通用的方案是通过 继承,将处理的逻辑加入到放入Observer导出类的onError()中,这种方式相对 简单易于上手,但限制性在于:

  • 1.开发者无法在subscribe()中使用 lambda,灵活性降低, 这也是为什么强调尽量考虑 组合 的设计方式而不直接采用 继承 的原因。
  • 2.如果开发者想要移除(或者重构)相关代码,就必须改动既有的业务逻辑,比如每一个网络请求的subscribe()中的代码,这种逻辑改动是产品级的。
  • 3.onError()中对 error 进行处理,这意味着打破了Observable的链式调用。

RxWeaver 是轻量且灵活的 RxJava2 全局 Error 处理中间件 ,类似 AOP 的思想,在既有的代码上 插入删除 一行代码,即可实现全局 Error 处理的需求——而不是破坏RxJava所表达的 响应式编程函数式编程 的思想。

特性

  • 1.轻量:整个工具库只有 4 个类共不到 200 行代码,jar 包体积仅 3kb;
  • 2.无额外依赖:所有代码的实现都依赖于RxJava本身的原生操作符;
  • 3.高扩展性:开发者可以通过接口实现任意复杂的需求实现,详情请参考下文;
  • 4.灵活:不破坏既有的业务代码,而是在原本的流中 插入删除 一行代码——它是 可插拔的

Usage

1.Fork/Clone 项目,并将rxweaver模块的代码复制进自己的项目中,并进行对应的修改:

$ git clone https://github.com/qingmei2/RxWeaver.git

或者直接添加依赖(不建议)

implementation 'com.github.qingmei2.rxweaver:rxweaver:0.3.0'

2.配置 GlobalErrorTransformer

GlobalErrorTransformer 是一个是Transformer<T, R>的实现类,负责把全局的 error 处理逻辑,分发给不同的 响应式类型(Observable、Flowable、Single、Maybe、Completable):

class GlobalErrorTransformer<T> constructor(
        private val globalOnNextInterceptor: (T) -> Observable<T> = { Observable.just(it) },
        private val globalOnErrorResume: (Throwable) -> Observable<T> = { Observable.error(it) },
        private val retryConfigProvider: (Throwable) -> RetryConfig = { RetryConfig() },
        private val globalDoOnErrorConsumer: (Throwable) -> Unit = { }
) : ObservableTransformer<T, T>, FlowableTransformer<T, T>, SingleTransformer<T, T>,  MaybeTransformer<T, T>, CompletableTransformer {
      // ...
}

配置一个函数,保证能够返回GlobalErrorTransformer的实例:

fun <T> handleGlobalError(activity: FragmentActivity): GlobalErrorTransformer<T>{
  return .....
}

点击这里查看 sample 中的配置方式示例。

3.对需要进行全局 error 处理的 RxJava 流中添加这行代码:

private fun requestHttp(observable: Observable<UserInfo>) {
    observable
            .compose(handleGlobalError<UserInfo>(this))  // 将上面的接口配置给 Observable
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe {
                  // ....
            }
}

功能

下面通过几个案例展示 RxWeaver 的功能支持,你也可以运行 sample 并运行查看效果。

1.全局异常配置,弹出 toast:

2.全局异常配置,弹出 Dialog 并是否重试:

3.全局异常配置,token 异常,重新登录后返回界面重试:

4.更多

当然也能够实现更多私有化定制的需求....

原理&如何配置 GlobalErrorTransformer

RxWeaver 是一个轻量且灵活的 RxJava2 全局 Error 处理组件 ,这意味着,它并不是一个设计非常复杂的框架。Weaver 翻译过来叫做 编织鸟, 可以让开发者对 error 处理逻辑 组织,以达到实现全局 Error 的把控。

它的核心原理是依赖 compose() 操作符——这是 RxJava 给我们提供的可以面向 响应式数据类型 (Observable/Flowable/Single 等等)进行 AOP 的接口, 可以对响应式数据类型 加工修饰 ,甚至 替换

它的原理也是非常 简单 的,只要熟悉了 onErrorResumeNextretryWhendoOnError 这几个关键的操作符,你就可以马上上手对应的配置;它也是非常 轻量 的,轻到甚至可以直接把源代码复制粘贴到自己的项目中,通过 jcenter 依赖,它的体积也只有 3kb。

1. globalOnNextInterceptor: (T) -> Observable

将正常数据转换为一个异常 是很常见的需求,有时流中的数据可能会是一个错误的状态(比如,token 失效)。

globalOnNextInterceptor 函数内部直接使用 flatMap() 操作符将数据进行了转换,因此你可以将一个错误的状态转换为 Observable.error() 向下传递:

globalOnNextInterceptor = {
    when (it.statusCode) {
        STATUS_UNAUTHORIZED -> {        // token 失效,将流转换为 error,交由下游处理
            Observable.error(TokenExpiredException())
        }
        else -> Observable.just(it)     // 其他情况,数据向下游正常传递
    }
}

2. globalOnErrorResume: (Throwable) -> Observable

globalOnNextInterceptor 函数很相似,更常见的情况是通过解析不同的 Throwable ,然后根据实际业务做出对应的处理:

globalOnErrorResume = { error ->
    when (error) {
        is ConnectException -> {        // 连接错误,转换为特殊的异常(标记作用),交给下游
            Observable.error<T>(ConnectFailedAlertDialogException())
        }
        else -> Observable.error<T>(error)  // 其他情况,异常向下游正常传递
    }
}

globalOnErrorResume 函数内部是通过 onErrorResumeNext() 操作符实现的。

3.retryConfigProvider: (Throwable) -> RetryConfig

当需要做出是否要重试的决定时,需要根据异常的类型进行判断,并做出对应的行为:

retryConfigProvider = { error ->
    when (error) {
        is ConnectFailedAlertDialogException -> RetryConfig {
            // .....
        }
        is TokenExpiredException -> RetryConfig(delay = 3000) {
            // .....
        }
        else -> RetryConfig() // 其它异常都不重试
    }
}

retryConfigProvider 函数内部是通过 retryWhen() 操作符实现的。

4.globalDoOnErrorConsumer: (Throwable) -> Unit

globalDoOnErrorConsumer 函数并不会拦截或消费上游发射的异常,它的内部实际上就是通过 doOnError() 操作符的调用,做一些 顺势而为 的处理,比如 toast,或者其它。

globalDoOnErrorConsumer = { error ->
    when (error) {
        is JSONException -> {
            Toast.makeText(activity, "全局异常捕获-Json 解析异常!", Toast.LENGTH_SHORT).show()
        }
        else -> {

        }
    }
}

License

The RxWeaver: MIT License

Copyright (c) 2018 qingmei2

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
Apps
About Me
GitHub: Trinea
Facebook: Dev Tools