【发布时间】:2014-12-27 21:01:13
【问题描述】:
我正在使用 Spring 的 RestTemplate 和 Jackson 使用 RESTful JSON API。在某些情况下,我们可能会收到带有自定义 JSON 正文的 Status 401(未经授权)响应,该响应由 API 制造商定义,如下所示:
{
"code": 123,
"message": "Reason for the error"
}
我们需要解析正文,并在我们的业务逻辑中使用code 属性。
这是我们需要解析的错误响应 Java 对象:
public class CustomError {
@JsonProperty
private Integer code;
@JsonProperty
private String message;
public Integer getCode() {
return code;
}
public String getMessage() {
return message;
}
}
还有一个自定义错误处理程序来执行此操作:
public class CustomErrorHandler extends DefaultResponseErrorHandler {
private RestTemplate restTemplate;
private ObjectMapper objectMapper;
private MappingJacksonHttpMessageConverter messageConverter;
@Override
public boolean hasError(ClientHttpResponse response) throws IOException {
return super.hasError(response);
}
@Override
public void handleError(final ClientHttpResponse response) throws IOException {
try {
CustomError error =
(CustomError) messageConverter.read(CustomError.class, response);
throw new CustomErrorIOException(error, error.getMessage());
} catch (Exception e) {
// parsing failed, resort to default behavior
super.handleError(response);
}
}
}
错误处理程序失败,在 try 块中出现 HttpMessageNotReadableException:
“无法读取 JSON:由于服务器身份验证,无法重试,处于流模式”
这就是我发送请求的方式:
restTemplate.postForObject(url, pojoInstance, responseClass);
如果使用普通的旧客户端程序(如 Postman)执行相同的请求,则会收到预期的 JSON 响应。因此,我认为问题可能在于 Spring 的 ClientHttpResponse 实现在 401 状态的情况下以某种方式不允许访问响应正文。
真的可以解析响应体吗?
更新
根据我的调查,RestTemplate 类使用ClientHttpResponse,而ClientHttpResponse 又创建了一个提供输入流的sun.net.www.protocol.http.HttpURLConnection。它在那里,输入流被忽略并抛出IOException:
由于服务器身份验证无法重试,在流模式下
所以,HttpURLConnection 的实现导致了这个问题。
是否有可能避免这个问题?也许我们应该使用一种替代实现,在出现错误状态代码的情况下不忽略响应主体?你能推荐任何替代品吗?
【问题讨论】:
标签: java spring api jackson resttemplate