从 Okhttp 使用 One-Shot ResponseBody 会导致 Retrofit 出现问题
Consuming One-Shot ResponseBody from Okhttp causes issues with Retrofit
我正在使用带有 Okhttp 拦截器的 Retrofit 来检测我的 oauth 令牌是否已过期。如果令牌已过期,我想请求一个新令牌,再次尝试请求,然后将该响应发送给 Retrofit。
这是我的拦截器class:
public class CustomInterceptor implements Interceptor {
@Override
public Response intercept(Chain chain) throws IOException {
Request request = chain.request();
// try the request
Response response = chain.proceed(request);
if (response.body().string().contains(Constants.TOKEN_AUTH_ERROR_MESSAGE)) {
Log.v("retrofit_error", "token expired");
//get current token, create headers
OAuthTokenResponse expiredToken = SharedPreferencesUtil.getOAuthToken();
OAuthTokenResponse newOauthToken = RestClient.getInstance().getTokenService().refreshOauthToken(expiredToken.getRefreshToken());
//store new token, return
SharedPreferencesUtil.saveOAuthToken(newOauthToken);
// create a new request and modify it accordingly using the new token
Request.Builder newRequestBuilder = request.newBuilder()
.removeHeader("Authorization");
Request newRequest = newRequestBuilder.addHeader("Authorization", SharedPreferencesUtil.getOAuthToken().getAccessToken()).build();
// retry the request
return chain.proceed(newRequest);
}
// otherwise just pass the original response on
return response;
}
}
问题是调用 response.body.string() 将消耗 ResponseBody,因为它是 Okhttp 文档状态的一次性值。
这意味着在代码末尾返回的 Response 将不再包含主体,当它被传递到改造时。有什么方法可以在消耗 body 的同时仍将其与响应一起返回吗?
谢谢!
我通过使用 Response.Builder 创建一个新响应来完成此操作。我可以使用 responseBodyString
进行支票;然后我 return newResponse
,它被赋予了我消费的 body。
Request request = chain.request();
Response response = chain.proceed(request);
ResponseBody responseBody = response.body();
String responseBodyString = response.body().string();
Response newResponse = response.newBuilder().body(ResponseBody.create(responseBody.contentType(), responseBodyString.getBytes())).build();
...
return newResponse;
你可以使用日志拦截器
https://github.com/square/okhttp/tree/master/okhttp-logging-interceptor
我正在使用带有 Okhttp 拦截器的 Retrofit 来检测我的 oauth 令牌是否已过期。如果令牌已过期,我想请求一个新令牌,再次尝试请求,然后将该响应发送给 Retrofit。
这是我的拦截器class:
public class CustomInterceptor implements Interceptor {
@Override
public Response intercept(Chain chain) throws IOException {
Request request = chain.request();
// try the request
Response response = chain.proceed(request);
if (response.body().string().contains(Constants.TOKEN_AUTH_ERROR_MESSAGE)) {
Log.v("retrofit_error", "token expired");
//get current token, create headers
OAuthTokenResponse expiredToken = SharedPreferencesUtil.getOAuthToken();
OAuthTokenResponse newOauthToken = RestClient.getInstance().getTokenService().refreshOauthToken(expiredToken.getRefreshToken());
//store new token, return
SharedPreferencesUtil.saveOAuthToken(newOauthToken);
// create a new request and modify it accordingly using the new token
Request.Builder newRequestBuilder = request.newBuilder()
.removeHeader("Authorization");
Request newRequest = newRequestBuilder.addHeader("Authorization", SharedPreferencesUtil.getOAuthToken().getAccessToken()).build();
// retry the request
return chain.proceed(newRequest);
}
// otherwise just pass the original response on
return response;
}
}
问题是调用 response.body.string() 将消耗 ResponseBody,因为它是 Okhttp 文档状态的一次性值。
这意味着在代码末尾返回的 Response 将不再包含主体,当它被传递到改造时。有什么方法可以在消耗 body 的同时仍将其与响应一起返回吗?
谢谢!
我通过使用 Response.Builder 创建一个新响应来完成此操作。我可以使用 responseBodyString
进行支票;然后我 return newResponse
,它被赋予了我消费的 body。
Request request = chain.request();
Response response = chain.proceed(request);
ResponseBody responseBody = response.body();
String responseBodyString = response.body().string();
Response newResponse = response.newBuilder().body(ResponseBody.create(responseBody.contentType(), responseBodyString.getBytes())).build();
...
return newResponse;
你可以使用日志拦截器 https://github.com/square/okhttp/tree/master/okhttp-logging-interceptor