从 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