我正在尝试处理服务器响应,这是GZIP' d。响应带有标题
Content-Type: application/x-gzip
但没有标题
Content-Encoding: gzip
如果我使用代理添加该标头,则响应会被解析得很好。 我无法控制服务器,因此我无法添加标头。
我可以强制将Retrofit视为GZIP内容吗?有没有更好的办法? 服务器的URL是: http://crowdtorch.cms.s3.amazonaws.com/4474/Updates/update-1.xml
答案 0 :(得分:8)
我明白了。我们的想法是添加一个自定义拦截器,它将采用尚未解压缩的响应,并“手动”解压缩 - 执行OkHttp基于Content-Encoding标头自动执行的操作,但不需要该标头。
就像dis:
OkHttpClient.Builder clientBuilder = new OkHttpClient.Builder()
.addInterceptor(new UnzippingInterceptor());
OkHttpClient client = clientBuilder.build();
拦截器就像dis:
private class UnzippingInterceptor implements Interceptor {
@Override
public Response intercept(Chain chain) throws IOException {
Response response = chain.proceed(chain.request());
return unzip(response);
}
}
解压缩功能就像dis:
// copied from okhttp3.internal.http.HttpEngine (because is private)
private Response unzip(final Response response) throws IOException {
if (response.body() == null) {
return response;
}
GzipSource responseBody = new GzipSource(response.body().source());
Headers strippedHeaders = response.headers().newBuilder()
.removeAll("Content-Encoding")
.removeAll("Content-Length")
.build();
return response.newBuilder()
.headers(strippedHeaders)
.body(new RealResponseBody(strippedHeaders, Okio.buffer(responseBody)))
.build();
}
答案 1 :(得分:1)
有一种比重新发明轮子更好的方法。只需自己添加Content-Encoding
标题即可。
.addNetworkInterceptor((Interceptor.Chain chain) -> {
Request req = chain.request();
Headers.Builder headersBuilder = req.headers().newBuilder();
String credential = Credentials.basic(...);
headersBuilder.set("Authorization", credential);
Response res = chain.proceed(req.newBuilder().headers(headersBuilder.build()).build());
return res.newBuilder()
.header("Content-Encoding", "gzip")
.header("Content-Type", ""application/json")
.build();
})
事实上,您的代码是使用内部代码(例如JDK的com.sun
包)的邪恶的典型示例。 RealResponseBody
不再具有该构造函数。