inflearn logo
강의

강의

N
챌린지

챌린지

멘토링

멘토링

N
클립

클립

로드맵

로드맵

지식공유

스프링 MVC 2편 - 백엔드 웹 개발 활용 기술

스프링 인터셉터 - 인증 체크

인터셉터의 호출 순서 질문입니다.

해결된 질문

629

김회민

작성한 질문수 6

1

강의를 따라가다 문득 preHandle, postHandle, afterCompletion 의 순서를 알고 싶어서 LoginCheckInterceptor 에 해당 메서드를 추가하여 로그를 찍어보았습니다.

질문은 맨 밑에 있으며, 코드는 이해를 돕기 위해 첨부했습니다.

LogInterceptor

@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);

        // @RequestMapping: HandlerMethod
        // 정적 리소스: ResourceHttpRequestMethod
        if (handler instanceof HandlerMethod) {
            // 호출할 컨트롤러 메서드의 모든 정보가 포함되어 있다.
            HandlerMethod hm = (HandlerMethod) handler;
        }

        log.info("[{}][{}] LogInterceptor preHandle", requestURI, uuid);
        return true;
    }

    @Override
    public void postHandle(
            HttpServletRequest request,
            HttpServletResponse response,
            Object handler,
            ModelAndView modelAndView
    ) throws Exception {
        String requestURI = request.getRequestURI();
        String uuid = (String) request.getAttribute(LOG_ID);

        log.info("[{}][{}] LogInterceptor postHandle", requestURI, uuid);
    }

    @Override
    public void afterCompletion(
            HttpServletRequest request,
            HttpServletResponse response,
            Object handler,
            Exception ex
    ) throws Exception {
        String requestURI = request.getRequestURI();
        String uuid = (String) request.getAttribute(LOG_ID);

        log.info("[{}][{}] LogInterceptor afterComplete", requestURI, uuid);

        if (ex != null) {
            log.error("LogInterceptor afterComplete Error: ", ex);
            ex.printStackTrace();
        }
    }
}

LoginCheckInterceptor

@Slf4j
public class LoginCheckInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(
            HttpServletRequest request,
            HttpServletResponse response,
            Object handler
    ) throws Exception {
        String requestURI = request.getRequestURI();
        String uuid = (String) request.getAttribute(LogInterceptor.LOG_ID);
        log.info("[{}][{}] LoginCheckInterceptor preHandle", requestURI, uuid);

        HttpSession session = request.getSession();
        if (session == null || session.getAttribute(SessionConst.LOGIN_MEMBER) == null) {
            log.info("미인증 사용자 요청 {}", requestURI);
            response.sendRedirect("/login?redirectURL=" + requestURI);
            return false;
        }

        return true;
    }

    @Override
    public void postHandle(
            HttpServletRequest request,
            HttpServletResponse response,
            Object handler,
            ModelAndView modelAndView
    ) throws Exception {
        String requestURI = request.getRequestURI();
        String uuid = (String) request.getAttribute(LogInterceptor.LOG_ID);

        log.info("[{}][{}] LoginCheckInterceptor postHandle", requestURI, uuid);
    }

    @Override
    public void afterCompletion(
            HttpServletRequest request,
            HttpServletResponse response,
            Object handler,
            Exception ex
    ) throws Exception {
        String requestURI = request.getRequestURI();
        String uuid = (String) request.getAttribute(LogInterceptor.LOG_ID);

        log.info("[{}][{}] LoginCheckInterceptor afterCompletion", requestURI, uuid);
    }
}

InterceptorConfig

@Configuration
public class InterceptorConfig implements WebMvcConfigurer {
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new LogInterceptor())
                .order(1)
                .addPathPatterns("/**") // 모두 허용
                .excludePathPatterns("/css/**", "/*.ico", "/error"); // BlackList

        registry.addInterceptor(new LoginCheckInterceptor())
                .order(2)
                .addPathPatterns("/**")
                .excludePathPatterns(
                        "/", "/members/css", "/login",
                        "/logout", "/css/**", "*.ico", "/error"
                );
    }
}

결과 1 - 미인증 사용자 요청

// /items 요청
[/items][45e50a37-57a0-4298-b50e-e42141005426] LogInterceptor preHandle
[/items][45e50a37-57a0-4298-b50e-e42141005426] LoginCheckInterceptor preHandle
미인증 사용자 요청 /items
[/items][45e50a37-57a0-4298-b50e-e42141005426] LogInterceptor afterComplete

// Redirect - /login
[/login][18f052ab-b849-4690-83ec-43866660f570] LogInterceptor preHandle
[/login][18f052ab-b849-4690-83ec-43866660f570] LogInterceptor postHandle
[/login][18f052ab-b849-4690-83ec-43866660f570] LogInterceptor afterComplete

로그상으로 LoginCheckInterceptorafterComplete누락되었습니다.

결과2 - 정상 처리

// /login 요청
[/login][19bad338-e02d-4bbe-8b3a-5dfc55ad4428] LogInterceptor preHandle
LoginService: 'test', 'test!'
login? Member(id=1, loginId=test, name=테스터, password=test!)
[/login][19bad338-e02d-4bbe-8b3a-5dfc55ad4428] LogInterceptor postHandle
[/login][19bad338-e02d-4bbe-8b3a-5dfc55ad4428] LogInterceptor afterComplete

// Redirect - /items
[/items][679dc279-ff3a-4ee3-a424-26b18ac8dbbd] LogInterceptor preHandle
[/items][679dc279-ff3a-4ee3-a424-26b18ac8dbbd] LoginCheckInterceptor preHandle
[/items][679dc279-ff3a-4ee3-a424-26b18ac8dbbd] LoginCheckInterceptor postHandle
[/items][679dc279-ff3a-4ee3-a424-26b18ac8dbbd] LogInterceptor postHandle
[/items][679dc279-ff3a-4ee3-a424-26b18ac8dbbd] LoginCheckInterceptor afterCompletion
[/items][679dc279-ff3a-4ee3-a424-26b18ac8dbbd] LogInterceptor afterComplete

preHandleorder의 순위대로 로그가 찍히는데, postHandleafterCompletion의 경우 order의 순위와 반대로 호출이 됩니다.

 

혹시 위 2개의 결과에 대한 이유를 설명해주실 수 있을까요..? 그냥 로직때문에 그런 것인지 궁금합니다.

mvc MVC spring

답변 1

3

김영한

안녕하세요. 김회민님

생각해보시면 다음과 같이 동작합니다.

  • A 시작

    • B 시작

    • B 종료

  • A 종료

 

따라서 결과는 A시작 -> B 시작 -> B 종료 -> A 종료 순서가 됩니다.

감사합니다.

0

김회민

앗, 감사합니다. 결과 2에 대한건 해결이 되었습니다.

그럼 결과 1에 관한, afterCompletion이 호출되지 않는 이유는 preHandle에서 false를 반환하면, 이에 해당하는 인터셉터는 더이상 실행되지 않고, 등록에서 빠지게 되는 것이라고 해석하면 될까요?

0

김영한

네 맞습니다. :)

0

김회민

답변 정말 감사드립니다!!!

덕분에 머리속이 정리되었네요 !

이미지 업로드와 db 트랜잭션 묶는법

0

50

1

Could not resolve org.springframework.boot:spring-boot-starter-validation:2.4.4

0

61

2

MessageSourceTest 코드

0

51

1

인터셉터 에러 설정

0

51

1

resolveArgument()메서드 질문

0

59

1

43강 검증1 에서 실패 로직 관련 질문있습니다.

0

64

2

타임리프 3.X 버전 rendering, serializer 에러 해결 방법

2

134

3

스프링 빈에 등록이 안되는거 같은데 어떻게 하면 좋을까요?ㅠㅠ

0

93

3

pdf 오타 문의

0

58

1

ItemUpdateForm 검증 관련 질문입니다.

0

52

1

22page 링크 주소 변경

0

59

2

특정 데이터와 파일을 함께 저장 시, 테이블 구조 질문

0

53

1

섹션3번 수업에 대한 질문입니다.

0

85

2

@Autowired 보다 더 좋은 방법이 어떤 걸까요?

0

87

2

타입컨버터 가 람다랑 비슷해 보이는데 저의 생각이 맞는지?.

0

66

1

자바스크립트 인라인에서 객체 직렬화 시 오류가 납니다

0

143

3

스프링부트 - 오류페이지2 에서 500.html 에서 쓰인 객체 질문

0

66

1

톰캣 에러 페이지가 안보입니다.

0

108

2

apiEceptionController에서 센드 에러 호출하면 안되는지?

0

81

1

세션 타임아웃시 쿠키 삭제 방법이 없나요?

0

120

2

ApiExceptionController 질문드립니다.

0

64

1

셀렉박스 챕터에서 option value에 ==배송 방식 선택== 이것을 넣은 이유가 궁금함, 이렇게 구상해도 되는지?

0

67

1

MemberRepository 필드의 fianl 선언 유무

0

85

2

혹시 index.html 에서는 fragment 사용이 안되는건가요

0

60

1