오봉이와 함께하는 개발 블로그
스프링 MVC 2 - ExceptionResolver Part. 2 본문
API 예외 처리 - 스프링이 제공하는 ExceptionResolver 2
DefaultHandlerExceptionResolver
를 살펴보자.
DefaultHandlerExceptionResolver
는 스프링 내부에서 터진 오류를 HTTP 스펙에 맞는 상태 코드로 바꿔서 오류를 처리해준다.
DefaultHandlerExceptionResolver
는 스프링 내부에서 발생하는 스프링 예외를 해결한다.
대표적으로 파라미터 바인딩 시점에 타입이 맞지 않으면 내부에서 TypeMismatchException
이 발생하는데, 이 경우 예외가 발생했기 때문에 그냥 두면 서블릿 컨테이너까지 오류가 올라가고, 결과적으로 500 오류가 발생한다.
그런데 파라미터 바인딩은 대부분 클라이언트가 HTTP 요청 정보를 잘못 호출해서 발생하는 문제이다.(예 : 숫자 입력 폼에 문자 입력)
HTTP에서는 이런 경우 HTTP 상태 코드 400을 사용하도록 되어 있다.DefaultHandlerExceptionResolver
는 이것을 500 오류가 아니라 HTTP 상태 코드 400 오류로 변경한다.
스프링 내부 오류를 어떻게 처리할지 수 많은 내용이 정의되어 있다.
코드 확인DefaultHandlerExceptionResolver.handleTypeMismatch
를 보면 다음과 같은 코드를 확인할 수 있다.response.sendError(HttpServletResponse.SC_BAD_REQUEST)
(400 에러)
결국 response.sendError()
를 통해서 문제를 해결한다.
sendError(400)
을 호출했기 때문에 WAS에서 다시 오류 페이지(/error
)를 내부 요청한다.
ApiExceptionController - 추가
@GetMapping("/api/default-handler-ex")
public String defaultException(@RequestParam Integer data) {
return "ok";
}
Integer data에 문자를 입력하면 내부에서 TypeMismatchException이 발생한다.
결과
{
"timestamp": "2022-08-30T17:18:37.696+00:00",
"status": 400,
"error": "Bad Request",
"exception": "org.springframework.web.method.annotation.MethodArgumentTypeMismatchException",
"message": "Failed to convert value of type 'java.lang.String' to required type 'java.lang.Integer'; nested exception is java.lang.NumberFormatException: For input string: \"hello\"",
"path": "/api/default-handler-ex"
}
정리
ResponseStatusExceptionResolver
-> HTTP 응답 코드 변경DefaultHandlerExceptionResolver
-> 스프링 내부 예외 처리
HTTP 상태 코드를 변경하고(ResponseStatusExceptionResolver
), 스프링 내부 예외의 상태코드를 변경하는 기능(DefaultHandlerExceptionResolver
)도 알아보았다.
그런데 HandlerExceptionResolver
를 직접 사용하기는 복잡하다.
API 오류 응답의 경우 response
에 직접 데이터를 넣어야 해서 매우 불편하고 번거롭다.ModelAndView
를 반환해야 하는 것도 API에는 잘 맞지 않는다.
직접 구현한 HandlerExceptionResolver
@Slf4j
public class UserHandlerExceptionResolver implements HandlerExceptionResolver {
private final ObjectMapper objectMapper = new ObjectMapper();
@Override
public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
try {
if (ex instanceof UserException) {
log.info("UserException resolver to 400");
String acceptHeader = request.getHeader("accept"); // HTTP Header에서 accept에 대한 정보 얻어옴
if ("application/json".equals(acceptHeader)) {
Map<String, Object> errorResult = new HashMap<>();
errorResult.put("ex", ex.getClass());
errorResult.put("message", ex.getMessage());
String result = objectMapper.writeValueAsString(errorResult);
response.setContentType("application/json");
response.setCharacterEncoding("UTF-8");
response.getWriter().write(result);
return new ModelAndView();
} else {
// TEXT/HTML
return new ModelAndView("error/500"); // templates/error/500.html 반환
}
}
} catch (IOException e) {
log.error("resolver ex", e);
}
return null;
}
}
스프링은 이 문제를 해결하기 위해 @ExceptionHandler
라는 매우 혁신적인 예외 처리 기능을 제공한다.
그것이 아직 소개하지 않은 ExceptionHandlerExceptionResolver
이다.
출처 : 인프런 김영한 지식공유자님 강의 스프링 MVC 2편 - 백엔드 웹 개발 활용 기술
'BE > Spring' 카테고리의 다른 글
스프링 MVC 2 - @ControllerAdvice (0) | 2022.08.31 |
---|---|
스프링 MVC 2 - @ExceptionHandler (0) | 2022.08.31 |
스프링 MVC 2 - ExceptionResolver Part. 1 (0) | 2022.08.31 |
스프링 MVC 2 - HandlerExceptionResolver 활용 (0) | 2022.08.31 |
스프링 MVC 2 - HandlerExceptionResolver 시작 (0) | 2022.08.31 |