转换Observable onError并发出一个项目

时间:2018-07-05 19:03:40

标签: java android rx-java rx-android reactive

在我的Android应用程序中,我有一个生成的(Swagger)ApiClient,它使用令牌进行初始化以标识用户。

此令牌(来自服务器)可以过期。

我可以通过myApiClient.myServiceMethod(params)从Web服务中获取带数据的Observable

当我从服务器获得401时,这意味着我的令牌已过期,并且我必须再次初始化myApiClient(以使用未过期的令牌获取它)。

如何初始化myApiClient。它由getMyApiClient()

返回
 //getOkHttpClientWithToken() first issues another Server call to get an
 //unexpired Token and then returns an OkHttpClient with that Token set.

 myApiClient = new ApiClient()
              .getAdapterBuilder()
              .baseUrl(url)
              .client(getOkHttpClientWithToken())
              .build()
              .create(MyApiClient.class);

我从myApiClient调用中获得了Observable

Observable<Result> getResultObservable(Type param1, Type param2) {
    return Observable.just(getMyApiClient())
                     .flatMap(myApiClient ->
                        myApiClient.getResult(param1, param2).cache()
                      );
}

我尝试过的是

//How to recreate myApiClient and Retry call on new myApiClient when an Error occurs
getResultObservable(param1, param2)
  .take(1)
  .subscribe(result -> {
     doSomethingWithResult();
   }
});

可以运行,但可以处理所有错误,而不仅仅是401,并且永远不会结束

我需要类似的东西

getResultObservable(param1, param2)
  .take(1)
  .subscribe(result -> {
     doSomethingWithResult();
   }, e -> {
     if(e.getMessage.equals("HTTP 401")) {
         "Transform this Observable to getResultObservable with new myApiClient and emit Result in onNext"
     } else {
       "Other error inform User and stop."
     }      
   }
});

1 个答案:

答案 0 :(得分:1)

您可以使用Observable#onErrorResumeNext提供后备可观察的。如果满足给定条件,则可以检查异常的类型,并返回一个fallback-observable,或者将异常包装为Observable#error。

请查看测试,如何使用#onErrorResumNext在发生某些异常时提供可回退的观察。

  @Test
  void nameX() {
    Observable<String> stringObservable = get();

    Observable<String> fallback$ =
        stringObservable
            .doOnError(s -> System.out.println("fail -> " + s.getMessage()))
            .onErrorResumeNext(
                throwable -> {
                  if (throwable instanceof MyException) {
                    return fallBack().doOnNext(s -> System.out.println("use fallback value " + s));
                  } else {
                    return Observable.error(throwable);
                  }
                });

    fallback$
        .test()
        .assertNotComplete()
        .assertValueCount(1)
        .assertValueAt(0, s -> "Wurst".equals(s));
  }

  private Observable<String> get() {
    return Observable.error(new MyException("Fail"));
  }

  private Observable<String> fallBack() {
    return Observable.just("Wurst").mergeWith(Observable.never());
  }

  private static final class MyException extends Exception {
    MyException(String message) {
      super(message);
    }
  }