RxJava:仅当第一个观察者抛出错误并从第一个观察者重复时才执行第二个观察者

时间:2017-08-17 13:55:40

标签: android rx-java retrofit2 rx-java2 rx-kotlin

我正在使用retorift来点击getAricle api并获取与该用户相关的文章列表。如果传递的令牌已过期getArticle api将抛出错误,那么我必须调用refreshToken api获取新令牌然后我再次调用getArticle api

 ApiController.createRx().getArticle(token)
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe({ response -> toast(response.body().url) }, { e ->
                println(e.printStackTrace())
                if(e is HttpException && e.code() in  arrayOf(401,403)){                      
                   //Here I want to call refresh tolken api
                   toast("Auth error")
                }
                else
                   toast(R.string.something_went_wrong)
            })

修改

尽管给出的答案显示了一些方向,但这些并不是我问题的直接答案。这是如何解决它,但我觉得这可以重构为更好的代码

ApiController.createRx().getArticle(Preference.getToken())
            .flatMap { value ->
                if (value.code() in arrayOf(403, 401)) {
                    ApiController.refreshToken()
                    ApiController.createRx().getArticle(Preference.getToken())
                } else Observable.just(value)
            }
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe({ response -> println("Success") }, { e ->
                e.printStackTrace()
                toast(R.string.something_went_wrong)
            })



fun refreshToken() {
        val token:String?=ApiController.createRx().refreshToken(Preferences.getRefreshToken()).blockingFirst()?.body()?.token
        if (token != null) Preferences.setAuthToken(token)
    }

修改

我将我的代码重构为更简洁的版本

Observable.defer { ApiController.createRx().getArticle(Preferences.getToken()) }
            .flatMap {
                if (it.code() in arrayOf(401, 403)) {
                    ApiController.refreshToken()
                    Observable.error(Throwable())
                } else Observable.just(it)
            }
            .retry(1)
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe({println("Success") }, {
              it.printStackTrace()
              toast(R.string.something_went_wrong)
            })



 fun refreshToken() {
        var token: String? = null
        try {
            token = createRx().refreshToken(Preferences.getRefreshToken()).blockingFirst().body()!!.token
        } catch (e: Exception) {
            throw e
        }
        println("saving token")
        if (token != null) Preferences.setAuthToken(token)
    }

修改

请查看我的答案,了解最终重构的代码

4 个答案:

答案 0 :(得分:0)

我已经实现了这个确切的事情。以下是该代码的略微修改版本:

private Observable<Object> refreshTokenIfNotAuthorized(Observable<? extends Throwable> errors) {
    final AtomicBoolean alreadyRetried = new AtomicBoolean(false);

    return errors.flatMap(error -> {

        boolean isAuthorizationError = /* some logic analyzing each error*/ ;

        if (isAuthorizationError && !alreadyRetried.get()) {
            try {
                alreadyRetried.set(true);
                String newToken = federatedTokenRefresher.refreshToken()
                                                         .toBlocking()
                                                         .first();

                setLogin(newToken);
                return Observable.just(null);

            } catch (Exception e) {
                return Observable.error(error);
            }

        }
        return Observable.error(error);
    });
}

您可以像这样使用此方法:

doSomethingRequiringAuth().retryWhen(this::refreshTokenIfNotAuthorized);

答案 1 :(得分:0)

你会收到什么样的错误?您似乎可以使用/运算符。

这个运算符一旦接收到throwable,就允许你在onError中返回一个Observable而不是throwable

onErrorResumeNext

您可以在此处查看更多示例https://github.com/politrons/reactive/blob/master/src/test/java/rx/observables/errors/ObservableExceptions.java

答案 2 :(得分:0)

我将使用groupBy运算符

为您提供另一个选项
/**
 * In this example we create a response code group.
 */
@Test
public void testGroupByCode() {
    Observable.from(Arrays.asList(401,403, 200))
            .groupBy(code -> code)
            .subscribe(groupByCode -> {
                switch (groupByCode.getKey()) {
                    case 401: {
                        System.out.println("refresh token");
                        processResponse(groupByCode);
                        break;
                    }
                    case 403: {
                        System.out.println("refresh token");
                        processResponse(groupByCode);
                        break;
                    }
                    default: {
                        System.out.println("Do the toast");
                        processResponse(groupByCode);
                    }
                }
            });
}

private void processResponse(GroupedObservable<Integer, Integer> groupByCode) {
    groupByCode.asObservable().subscribe(value -> System.out.println("Response code:" + value));
}

答案 3 :(得分:0)

我在阅读了更多关于RxJava的内容后解决了我的问题,这就是我实现它的方法。 首先retrofit会向onErroronNext\onSuccess投掷4xx错误取决于我们如何定义它。 例如:

@GET("content") fun getArticle(@Header("Authorization") token: String):Single<Article>

这会将所有4xx错误抛给onError而不是Single<Article>如果您将其定义为Single<Response<Article>>,则服务器(包括4xx)的所有响应都会转到onNext\onSuccess

Single.defer { ApiController.createRx().getArticle(Preferences.getAuthToken())}
                .doOnError {
                    if (it is HttpException && it.code() == 401)
                        ApiController.refreshToken()
                }
                .retry { attempts, error -> attempts < 3 && error is HttpException && error.code() == 401 }
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe({println("Success") }, {
                  it.printStackTrace()
                  toast(R.string.something_went_wrong)
                })

我正在使用defer作为实际Observable的包装器,因为我想在令牌刷新后重新创建文章提取observable,因为我想再次调用Preferences.getAuthToken()作为我的刷新令牌代码优先存储新获取的令牌。

如果retry为401且未尝试重试超过2次

HttpException,则返回true