我正在使用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)
}
修改
请查看我的答案,了解最终重构的代码
答案 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
答案 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
会向onError
或onNext\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