当前位置:   article > 正文

SpringCloud组件OpenFeign——将服务端详细异常信息返回给客户端_java open fein获取异常信息

java open fein获取异常信息

 一、场景

最近使用单位封装的微服务架构搭建微服务项目,封装的不太多,大部分可以按原生的SpringCloud组件的使用方式使用。涉及到的组件有Nacos、Ribbon、OpenFeign、Hystrix、Sentinel。领导让搭建微服务项目,但是不使用Hystrix和Sentinel做服务熔断和降级,即只需要客户端通过OpenFeign调用服务端服务。如果服务端方法异常,则直接将详细的异常信息返回给客户端,客户端能够捕获并打印出异常信息。在调用处便可知道报错原因,而无需再查看服务端日志。

二、模拟服务端异常

先让服务端产生一个算数异常并抛出,看一下正常情况下客户端会怎么样。

客户端代码

TestErrorController

  1. @Slf4j
  2. @RestController
  3. @RequestMapping("/test")
  4. public class TestErrorController {
  5. @Autowired
  6. private TestErrorFeignService testErrorFeignService;
  7. /**
  8. * 测试异常
  9. */
  10. @RequestMapping("/testError")
  11. public ResultBody testError() {
  12. String result;
  13. try {
  14. result = testErrorFeignService.testError();
  15. } catch (Exception e) {
  16. log.error("TestService testError e :{}.", e);
  17. return ResultBody.fail500(e.getMessage());
  18. }
  19. return ResultBody.success(result);
  20. }
  21. }

TestErrorFeignService

  1. @FeignClient(name = "service-provider",path = "/test")
  2. public interface TestErrorFeignService {
  3. /**
  4. * 测试异常
  5. */
  6. @RequestMapping(value = "/testError")
  7. public String testError();
  8. }

服务端代码

TestErrorController

  1. @Slf4j
  2. @RestController
  3. @RequestMapping("/test")
  4. public class TestErrorController {
  5. /**
  6. * 测试异常
  7. */
  8. @RequestMapping("/testError")
  9. public String testError() {
  10. int i;
  11. try {
  12. i = 16 / 0;
  13. } catch (Exception e) {
  14. log.error("TestErrorService testError 异常,异常信息为:{}.", e);
  15. throw e;
  16. }
  17. return "provider" + i;
  18. }
  19. }

运行后查看日志

服务端日志

 客户端日志

 

 通过上面日志可以看出,服务端日志显示是算数异常并且告诉我们是因为 / by zero。但是查看客户端打印的日志发现,捕获并打印的异常信息是feign.FeignException$InternalServerError: status 500 reading TestErrorFeignService#testError()。通过客户端的异常信息我们只知道客户端通过OpenFeign调用服务端的testError()方法异常,客户端产生的是状态码为500的FeignException。至于为什么会异常我们通过上述客户端日志无从得知,需要查看服务端日志。很明显这并不能满足我的需求。

三、OpenFeign如何处理调用异常

在解决该问题之前,我们首先要知道通过OpenFeign调用服务端接口异常的话是如何处理的。SynchronousMethodHandler->executeAndDecode这个方法第110行是真正调用服务端方法并返回一个feign.Response。如下图

 同样是该方法,从第138行到152行是对返回结果feign.Response的一个处理。可以看到,如果该结果不满足(response.status() >= 200 && response.status() < 300)且(decode404 && response.status() == 404 && void.class != metadata.returnType())时,会通过错误解码器进行解码,该方法会返回一个异常,然后抛出。如下图。

 OpenFeign提供了一个接口ErrorDecoder,该接口只有一个抽象方法decode,默认情况下是使用其静态内部类Default的decode方法对于Response的status不在2xx范围的HTTP进行解码的。如下图

四、解决方式

通过上面分析,给我们提供了一种思路,如果服务端异常后,我们返回一个状态码非200之类的,并且带有异常信息的响应。客户端实现这个接口并重写decode方法,实现我们自定义的错误解码器,就可以实现我们想要的效果,并且可以转换成我们想要的异常。具体方式如下。

服务端

首先服务端需要全局异常捕获,在服务端添加如下类。该类中主要捕获三种异常,服务端异常细分为系统异常、业务异常、参数校验异常。在捕获到异常后,将异常信息(e.getMessage())添加到响应中。当然,捕获的异常可以再细分。

全局异常捕获类(GlobalExceptionHandler)

  1. import com.example.rtbootconsumer.common.constants.ErrorCode;
  2. import com.example.rtbootconsumer.common.constants.ErrorMessage;
  3. import com.example.rtbootconsumer.common.exception.BusinessException;
  4. import com.example.rtbootconsumer.common.exception.ServerException;
  5. import lombok.AllArgsConstructor;
  6. import lombok.Data;
  7. import lombok.extern.slf4j.Slf4j;
  8. import org.springframework.context.support.DefaultMessageSourceResolvable;
  9. import org.springframework.util.CollectionUtils;
  10. import org.springframework.validation.ObjectError;
  11. import org.springframework.web.bind.MethodArgumentNotValidException;
  12. import org.springframework.web.bind.annotation.ExceptionHandler;
  13. import org.springframework.web.bind.annotation.RestControllerAdvice;
  14. import javax.servlet.http.HttpServletResponse;
  15. import java.io.Serializable;
  16. import java.util.List;
  17. import java.util.stream.Collectors;
  18. /**
  19. * @Description 全局异常捕获
  20. **/
  21. @Slf4j
  22. @RestControllerAdvice
  23. public class GlobalExceptionAdvice {
  24. /**
  25. * 捕获业务异常
  26. */
  27. @ExceptionHandler(value = {BusinessException.class})
  28. public ErrorResponse processBusinessException(HttpServletResponse response, BusinessException exception) {
  29. log.error(exception.getMessage(), exception);
  30. // 设置HTTP状态码
  31. response.setStatus(ErrorCode.BUSINESS_EXCEPTION);
  32. response.setContentType("application/json;charset=UTF-8");
  33. return new ErrorResponse(exception.getCode(), exception.getMessage());
  34. }
  35. /**
  36. * 捕获系统异常
  37. */
  38. @ExceptionHandler(value = {Exception.class, ServerException.class})
  39. public ErrorResponse processServerException(HttpServletResponse response, Exception exception) {
  40. log.error(exception.getMessage(), exception);
  41. // 设置HTTP状态码
  42. response.setStatus(ErrorCode.INTERNAL_SERVER_ERROR);
  43. response.setContentType("application/json;charset=UTF-8");
  44. return new ErrorResponse(ErrorCode.INTERNAL_SERVER_ERROR, exception.getMessage());
  45. }
  46. /**
  47. * 捕获参数校验异常
  48. */
  49. @ExceptionHandler(value = {MethodArgumentNotValidException.class})
  50. public ErrorResponse handleException(HttpServletResponse response, MethodArgumentNotValidException exception) {
  51. log.error(exception.getMessage(), exception);
  52. // 设置HTTP状态码
  53. response.setStatus(ErrorCode.PARAMETER_VALIDATION_ERROR);
  54. List<ObjectError> errorList = exception.getBindingResult().getAllErrors();
  55. ErrorResponse errorResponse;
  56. if (CollectionUtils.isEmpty(errorList)) {
  57. errorResponse = new ErrorResponse(ErrorCode.PARAMETER_VALIDATION_ERROR, ErrorMessage.INTERNAL_SERVER_ERROR);
  58. } else {
  59. List<String> errorMessageList = errorList.stream().map(DefaultMessageSourceResolvable::getDefaultMessage).collect(Collectors.toList());
  60. String errorMessage = String.join(",", errorMessageList);
  61. errorResponse = new ErrorResponse(ErrorCode.PARAMETER_VALIDATION_ERROR, errorMessage);
  62. }
  63. return errorResponse;
  64. }
  65. @Data
  66. @AllArgsConstructor
  67. private class ErrorResponse implements Serializable {
  68. private static final long serialVersionUID = -4306504824565204107L;
  69. private int code;
  70. private String message;
  71. }
  72. }

错误码常量类(ErrorCode)

  1. public class ErrorCode {
  2. /**
  3. * 系统内部异常
  4. */
  5. public static final int INTERNAL_SERVER_ERROR = 500;
  6. /**
  7. * 业务异常
  8. */
  9. public static final int BUSINESS_EXCEPTION = 550;
  10. /**
  11. * 参数不合法
  12. */
  13. public static final int PARAMETER_VALIDATION_ERROR = 450;
  14. }

自定义业务异常(BusinessException)

  1. import lombok.Data;
  2. /**
  3. * @Description 业务异常
  4. **/
  5. @Data
  6. public class BusinessException extends RuntimeException {
  7. private static final long serialVersionUID = 843904622605906038L;
  8. private int code;
  9. private String message;
  10. public BusinessException(int code, String message) {
  11. super(message);
  12. this.code = code;
  13. this.message = message;
  14. }
  15. public BusinessException(int code, String message, Throwable cause) {
  16. super(message, cause);
  17. this.code = code;
  18. this.message = message;
  19. }
  20. }

自定义参数校验异常(ParamValidateException)

  1. import lombok.Data;
  2. /**
  3. * @Description 参数校验异常
  4. **/
  5. @Data
  6. public class ParamValidateException extends RuntimeException {
  7. private static final long serialVersionUID = 806485898682779782L;
  8. private int code;
  9. private String message;
  10. public ParamValidateException(int code, String message) {
  11. super(message);
  12. this.code = code;
  13. this.message = message;
  14. }
  15. public ParamValidateException(int code, String message, Throwable cause) {
  16. super(message, cause);
  17. this.code = code;
  18. this.message = message;
  19. }
  20. }

自定义系统异常(ServerException )

  1. import lombok.Data;
  2. /**
  3. * @Description 系统异常
  4. **/
  5. @Data
  6. public class ServerException extends RuntimeException {
  7. private static final long serialVersionUID = 761184035642900420L;
  8. private int code;
  9. private String message;
  10. public ServerException(int code, String message) {
  11. super(message);
  12. this.code = code;
  13. this.message = message;
  14. }
  15. public ServerException(int code, String message, Throwable cause) {
  16. super(message, cause);
  17. this.code = code;
  18. this.message = message;
  19. }
  20. }

客户端

客户端新增自定义错误解码器

自定义错误解码器(FeignClientErrorDecoder)

  1. import com.example.rtbootprovider.common.constants.ErrorCode;
  2. import com.example.rtbootprovider.common.exception.BusinessException;
  3. import com.example.rtbootprovider.common.exception.ParamValidateException;
  4. import com.example.rtbootprovider.common.exception.ServerException;
  5. import com.google.gson.Gson;
  6. import feign.Response;
  7. import feign.Util;
  8. import feign.codec.ErrorDecoder;
  9. import lombok.extern.slf4j.Slf4j;
  10. import org.springframework.context.annotation.Configuration;
  11. import org.springframework.http.HttpStatus;
  12. import java.io.IOException;
  13. /**
  14. * @Description 自定义错误解码器
  15. **/
  16. @Slf4j
  17. @Configuration
  18. public class FeignClientErrorDecoder implements ErrorDecoder {
  19. @Override
  20. public Exception decode(String methodKey, Response response) {
  21. try {
  22. if (response.body() != null) {
  23. String errorContent = Util.toString(response.body().asReader());
  24. Gson gson = new Gson();
  25. if (response.status() == ErrorCode.INTERNAL_SERVER_ERROR) {
  26. return gson.fromJson(errorContent, ServerException.class);
  27. } else if (response.status() == ErrorCode.PARAMETER_VALIDATION_ERROR) {
  28. return gson.fromJson(errorContent, ParamValidateException.class);
  29. } else if (response.status() == ErrorCode.BUSINESS_EXCEPTION) {
  30. return gson.fromJson(errorContent, BusinessException.class);
  31. }
  32. }
  33. } catch (IOException e) {
  34. log.error("FeignClientErrorDecoder decode exception:{}.", e);
  35. return e;
  36. }
  37. return new Exception("服务端未知异常!");
  38. }
  39. }

效果展示 

添加完上面两个类就可以实现我们期望的效果,重新启动客户端和服务端项目并查看效果,客户端日志如下。

可以看出,客户端 打印出了服务端详细的异常信息了。

注意:

1、如果将异常信息String转换为IOException类,gson.fromJson(errorContent,IOException.class);捕获到的是FeignException。

 因为如果通过错误解码器生成的是IOException,会被catch到并通过errorReading(request, response, e);方法转换为FeignException。

2、错误解码器中response的status即为我们在全局异常捕获中设置的HttpServletResponse的status,即response.setStatus();我们在全局异常捕获中返回的自定义响应ResultBody中的内容就是错误解码器中入参Response的body的字符串内容。

如有错误,烦请指正,谢谢!

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家小花儿/article/detail/84605
推荐阅读
相关标签
  

闽ICP备14008679号