일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
- equals()
- http 메시지 컨버터
- 김영한
- html form
- 백준
- 오블완
- HttpServletResponse
- java
- 오버라이딩
- 싱글톤
- 참조변수
- 프록시
- 인터페이스
- DI
- @configuration
- fielderror
- 추상클래스
- 티스토리챌린지
- 스프링
- 스프링컨테이너
- 코드트리
- 테스트코드
- objecterror
- 코딩테스트
- ocp
- 서블릿
- 코드트리조별과제
- 의존관계
- JSON
- 다형성
- Today
- Total
minOS
서블릿 예외 처리 - 필터 ,인터셉터 본문
서블릿 예외 처리 - 필터
목표
예외 처리에 따른 필터와 인터셉터 그리고 서블릿이 제공하는 `DispatchType` 이해하기
예외 발생과 오류 페이지 요청 흐름
1. WAS(여기까지 전파) <- 필터 <- 서블릿 <- 인터셉터 <- 컨트롤러(예외발생) 2. WAS `/error-page/500` 다시 요청 -> 필터 -> 서블릿 -> 인터셉터 -> 컨트롤러(/error-page/ 500) -> View
오류가 발생하면 오류 페이지를 출력하기 위해 WAS 내부에서 다시 한번 호출이 발생한다. 이때 필터, 서블릿, 인터셉터도 모두 다시 호출된다. 그런데 로그인 인증 체크 같은 경우를 생각해보면, 이미 한번 필터나, 인터셉터에서 로그인 체크를 완료했다. 따라서 서버 내부에서 오류 페이지를 호출한다고 해서 해당 필터나 인터셉트가 한번 더 호출되는 것은 매우 비효율적이다.
결국 클라이언트로 부터 발생한 정상 요청인지, 아니면 오류 페이지를 출력하기 위한 내부 요청인지 구분할 수 있어야 한다. 서블릿은 이런 문제를 해결하기 위해 `DispatcherType` 이라는 추가 정보를 제공한다.
DispatcherType
필터는 이런 경우를 위해서 `dispatcherTypes` 라는 옵션을 제공한다.고객이 처음 요청하면 `dispatcherType=REQUEST`이다. 이렇듯 서블릿 스펙은 실제 고객이 요청한 것인지, 서버가 내부에서 오류 페이지를 요청하는 것인지`DispatcherType`으로 구분할 수 있는 방법을 제공한다.
`javax.servlet.DispatcherType` ``` public enum DispatcherType { FORWARD, INCLUDE, REQUEST, ASYNC, ERROR }
DispatcherType
`REQUEST` : 클라이언트 요청
`ERROR` : 오류 요청
`FORWARD` : MVC에서 배웠던 서블릿에서 다른 서블릿이나 JSP를 호출할 때`RequestDispatcher.forward(request, response);`
`INCLUDE` : 서블릿에서 다른 서블릿이나 JSP의 결과를 포함할 때`RequestDispatcher.include(request, response);`
`ASYNC` : 서블릿 비동기 호출
필터와 DispatcherType
package hello.exception.filter; import lombok.extern.slf4j.Slf4j; import jakarta.servlet.*; import jakarta.servlet.http.HttpServletRequest; import java.io.IOException; import java.util.UUID; @Slf4j public class LogFilter implements Filter { @Override public void init(FilterConfig filterConfig) throws ServletException { log.info("log filter init"); } @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { HttpServletRequest httpRequest = (HttpServletRequest) request; String requestURI = httpRequest.getRequestURI(); String uuid = UUID.randomUUID().toString(); try { log.info("REQUEST [{}][{}][{}]", uuid, request.getDispatcherType(), //dispacherType 추가 requestURI); chain.doFilter(request, response); } catch (Exception e) { throw e; } finally { log.info("RESPONSE [{}][{}][{}]", uuid, request.getDispatcherType(),//dispacherType 추가 requestURI); } } @Override public void destroy() { log.info("log filter destroy"); } }
WebConfig
package hello.exception; import hello.exception.filter.LogFilter; import jakarta.servlet.DispatcherType; import jakarta.servlet.Filter; import org.springframework.boot.web.servlet.FilterRegistrationBean; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; @Configuration public class WebConfig implements WebMvcConfigurer { @Bean public FilterRegistrationBean logFilter(){ FilterRegistrationBean<Filter> filterFilterRegistrationBean = new FilterRegistrationBean<>(); filterFilterRegistrationBean.setFilter(new LogFilter()); filterFilterRegistrationBean.setOrder(1); filterFilterRegistrationBean.addUrlPatterns("/*"); filterFilterRegistrationBean.setDispatcherTypes(DispatcherType.REQUEST,DispatcherType.ERROR); return filterFilterRegistrationBean; } }
`filterRegistrationBean.setDispatcherTypes(DispatcherType.REQUEST,DispatcherType.ERROR);`
이렇게 두 가지를 모두 넣으면 클라이언트 요청은 물론이고, 오류 페이지 요청에서도 필터가 호출된다.
아무것도 넣지 않으면 기본 값이 `DispatcherType.REQUEST` 이다. 즉 클라이언트의 요청이 있는 경우에만 필터가 적용된다. 특별히 오류 페이지 경로도 필터를 적용할 것이 아니면, 기본 값을 그대로 사용하면 된다.물론 오류 페이지 요청 전용 필터를 적용하고 싶으면 `DispatcherType.ERROR` 만 지정하면 된다.
로그 출력
was 에서 다시 요청
http://localhost:8080/error-ex
오류 페이지 로그 찍힌 후 마지막 RESONSE 찍힘
서블릿 예외 처리 - 인터셉터
인터셉터 중복 호출 제거
package hello.exception.interceptor; import lombok.extern.slf4j.Slf4j; import org.springframework.web.servlet.HandlerInterceptor; import org.springframework.web.servlet.ModelAndView; import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; import java.util.UUID; @Slf4j public class LogInterceptor implements HandlerInterceptor { public static final String LOG_ID = "logId"; @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { String requestURI = request.getRequestURI(); String uuid = UUID.randomUUID().toString(); request.setAttribute(LOG_ID, uuid); log.info("REQUEST [{}][{}][{}][{}]", uuid, request.getDispatcherType(), requestURI, handler); //request.getDispatcherType() 추 return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { log.info("postHandle [{}]", modelAndView); // 컨트롤러에서 에러 발생시 호출 x } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { String requestURI = request.getRequestURI(); String logId = (String) request.getAttribute(LOG_ID); log.info("RESPONSE [{}][{}][{}]", logId, request.getDispatcherType(), requestURI); if (ex != null) { log.error("afterCompletion error!!", ex); } } }
앞서 필터의 경우에는 필터를 등록할 때 어떤 `DispatcherType` 인 경우에 필터를 적용할 지 선택할 수 있었다. 그런
데 인터셉터는 서블릿이 제공하는 기능이 아니라 스프링이 제공하는 기능이다. 따라서 `DispatcherType` 과 무관하게 항상 호출된다. 대신에 인터셉터는 다음과 같이 요청 경로에 따라서 추가하거나 제외하기 쉽게 되어 있기 때문에, 이러한 설정을 사용해
서 오류 페이지 경로를 `excludePathPatterns` 를 사용해서 빼주면 된다.
@Configuration public class WebConfig implements WebMvcConfigurer { @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor( new LogInterceptor()) .order(1) .addPathPatterns("/**") .excludePathPatterns("/css/**","*.ico","/error","/error-page/**"); //오류 페이지 하부 빼버림 }
전체 흐름 정리
필터, 인터셉터 오류 처리 호출에서 빠짐
`/error-ex` 오류 요청 필터는 `DispatchType` 으로 중복 호출 제거 (`dispatchType=REQUEST` ) 인터셉터는 경로 정보로 중복 호출 제거(`excludePathPatterns("/error-page/**")` )
1. WAS(/error-ex, dispatchType=REQUEST) -> 필터 -> 서블릿 -> 인터셉터 -> 컨트롤러 2. WAS(여기까지 전파) <- 필터 <- 서블릿 <- 인터셉터 <- 컨트롤러(예외발생) 3. WAS 오류 페이지 확인 4. WAS(/error-page/500, dispatchType=ERROR) -> 필터(x) -> 서블릿 -> 인터셉터(x) -> 컨트롤러(/error-page/500) -> View
'TIL > 김영한의 스프링 MVC 2편 - 백엔드 웹 개발 핵심 기술' 카테고리의 다른 글
API 예외 처리 - 시작 (0) | 2024.10.02 |
---|---|
스프링 부트 - 오류 페이지 (0) | 2024.10.01 |
서블릿 예외 처리 - 시작 ,오류 화면 제공 , 오류 페이지 작동 원리 (1) | 2024.10.01 |
로그인 처리2 - 스프링 인터셉터 소개 , 스프링 인터셉터 요청 로그 , 스프링 인터셉터 인증 체크 (2) | 2024.09.27 |
로그인 처리2 - 서블릿 필터 소개 , 서블릿 필터 요청 로그 , 서블릿 필터 인증 체크 (2) | 2024.09.27 |