如何捕获FeignClient异常

时间:2016-11-01 14:38:41

标签: spring-mvc netflix-feign

我尝试捕获从FeignClient连接的另一个微服务中收到的异常。我已经定制了ErrorDecoder,并且

public class CustomErrorDecoder implements ErrorDecoder {

    private final Logger log = LoggerFactory.getLogger(getClass());

    private final ErrorDecoder defaultErrorDecoder = new Default();

    @Override
    public Exception decode(String methodKey, Response response) {

        if (response.status() >= 400 && response.status() <= 499) {
            log.info("----------------- "+methodKey+" , "+response.status()+" , "+ response.reason());
            return new RestApiException(99,99,"");
        }
        return defaultErrorDecoder.decode(methodKey, response);
    }
} 

RestApiException扩展Exception。

@ControllerAdvice
public class GlobalControllerAdvice {

    private final Logger log = LoggerFactory.getLogger(getClass());

    @ExceptionHandler(RestApiException.class)
    public ResponseEntity<RestApiException> handleException(RestApiException exception, HttpServletRequest req) {
        log.error("Sending error to client ( "+req.getUserPrincipal().getName()+" ) \"{}\"", exception.getErrMsg());
        return new ResponseEntity<RestApiException>(exception, exception.getStatus());
    }

    @ExceptionHandler(Throwable.class)
    public ResponseEntity<RestApiException> handleException(Throwable throwable, HttpServletRequest req) {
        RestApiException exception=new RestApiException(HttpStatus.INTERNAL_SERVER_ERROR, 100, 100,
                throwable.getMessage());
        return handleException(exception, req);
    }

结果,当我得到&lt; --- HTTP / 1.1 400 Bad Request(5380ms) 我有默认的错误消息

HttpStatus.INTERNAL_SERVER_ERROR,100,100,                     throwable.getMessage());

但没有考虑自定义异常,我尝试在CustomErrorDecoder中设置。

我做错了什么,为什么我不能调用RetAppiException并将错误答案返回给rest客户端。

感谢。

2 个答案:

答案 0 :(得分:6)

您无法使用@ControllerAdvice捕获FeignClient的例外情况。异常处理程序不会捕获假装客户端生成的异常,错误解码器..

一个简单的解决方案是抓住你的假装调用,然后抛出你想要的异常。

try{
   feignClient.method();
} catch(Exception ex){
  //throw exceptions you want
  throw new YourException();
}

然后你就可以处理它了:

@ControllerAdvice
public class GlobalControllerAdvice {

    private final Logger log = LoggerFactory.getLogger(getClass());

    @ExceptionHandler(YourException.class)
    public ResponseEntity<RestApiException> handleException(RestApiException exception, HttpServletRequest req) {
        //impl
    }

}

答案 1 :(得分:0)

您可以通过捕获HystrixRuntimeException并将getCause()强制转换为FeignClientException来捕获伪装客户端异常。

示例:

@ExceptionHandler
public ResponseEntity<Problem> handleFeignException(HystrixRuntimeException ex, NativeWebRequest request) {
    final FeignException.FeignClientException feignClientException = (FeignException.FeignClientException) ex.getCause();
    return handleException(feignClientException, request);
}