인프런 커뮤니티 질문&답변

diaman75님의 프로필 이미지
diaman75

작성한 질문수

스프링 핵심 원리 - 기본편

스코프와 Provider

MyLogger 를 Interceptor 로 구현 시 질문

작성

·

451

·

수정됨

0

=========================================
[질문 템플릿]
1. 강의 내용과 관련된 질문인가요? (예/아니오) 예
2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? (예/아니오) 예
3. 질문 잘하기 메뉴얼을 읽어보셨나요? (예/아니오) 예

[질문 내용]
안녕하세요? 강의를 듣다가 requesURL 같은 경우 Interceptor 에서 멤버 변수로 저장하는 것이 좋다고 해서 구현을 해보려고 했는데 문제가 생겨서 질문 드립니다.

## 앞으로 강좌를 계속 듣긴 할 거라서 혹시 뒤의 강좌를 듣고 나면 해결될 수 있는 문제라면 어느 강좌인지 알려주셔도 될 것 같습니다.

 

MyLogger.java

package hello.core.common;

import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
import java.util.UUID;

@Component
@Scope(value = "request")
public class MyLogger {

    private String uuid;
    private String requestURL;

    public void setRequestURL(String requestURL) {
        this.requestURL = requestURL;
    }

    public void log(String message) {
        System.out.println("[" + uuid + "]" + "[" + requestURL + "] " + message);
    }

    @PostConstruct
    public void init() {
        uuid = UUID.randomUUID().toString();
        System.out.println("[" + uuid + "] request scope bean create: " + this);
    }

    @PreDestroy
    public void close() {
        System.out.println("[" + uuid + "] request scope bean close: " + this);
    }
} 

CoreInterceptor.java

package hello.core.common;

import lombok.RequiredArgsConstructor;
import org.springframework.beans.factory.ObjectProvider;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

@Component
@RequiredArgsConstructor
public class CoreInterceptor implements HandlerInterceptor {

    private final ObjectProvider<MyLogger> myLoggerProvider;

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("CoreInterceptor.preHandle");
        String requestURL = request.getRequestURL().toString();
        MyLogger myLogger = myLoggerProvider.getObject();
        myLogger.setRequestURL(requestURL);
        return HandlerInterceptor.super.preHandle(request, response, handler);
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        HandlerInterceptor.super.postHandle(request, response, handler, modelAndView);
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        HandlerInterceptor.super.afterCompletion(request, response, handler, ex);
    }
}

 

WebMvcConfig.java

package hello.core.common;

import lombok.RequiredArgsConstructor;
import org.springframework.beans.factory.ObjectProvider;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

@Configuration
@RequiredArgsConstructor
public class WebMvcConfig implements WebMvcConfigurer {

    private final ObjectProvider<MyLogger> myLoggerProvider;

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new CoreInterceptor(myLoggerProvider)).excludePathPatterns("/css/**", "/images/**", "/js/**");
    }
}

 

LogDemoController.java

package hello.core.web;

import hello.core.common.MyLogger;
import lombok.RequiredArgsConstructor;
import org.springframework.beans.factory.ObjectProvider;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

import javax.servlet.http.HttpServletRequest;

@Controller
@RequiredArgsConstructor
public class LogDemoController {

    private final LogDemoService logDemoService;
    private final ObjectProvider<MyLogger> myLoggerProvider;

    @RequestMapping("log-demo")
    @ResponseBody
    public String logDemo(HttpServletRequest request) {
        String requestURL = request.getRequestURL().toString();
        MyLogger myLogger = myLoggerProvider.getObject();

        myLogger.log("controller test");
        logDemoService.logic("testId");
        return "OK";
    }
}

 

LogDemoService.java

package hello.core.web;

import hello.core.common.MyLogger;
import lombok.RequiredArgsConstructor;
import org.springframework.beans.factory.ObjectProvider;
import org.springframework.stereotype.Service;

@Service
@RequiredArgsConstructor
public class LogDemoService {

    private final ObjectProvider<MyLogger> myLoggerProvider;

    public void logic(String id) {
        MyLogger myLogger = myLoggerProvider.getObject();
        myLogger.log("service id = " + id);
    }
}

 

위와 같이 이렇게 구현을 해보니 문제가 생기는데...

  1. 일단, 인터셉터를 스프링 빈으로 등록하면서 MyLogger 를 활용하려면 알려주신 ObjectProvider 를 사용해야 할 것 같은데... WebMvcConfig 에서 인터셉터를 등록하는 경우, 위의 경우처럼 처음 서버 구동 시에 처리되어야 하는데 이 때는 reqeust 를 사용할 수 없는 문제입니다. 앞서 이 문제를 ObjectProvider 로 해결했는데 Interceptor 의 경우, 나중에 등록이 안 될 것 같은데 이런 경우 어떻게 해결해야 될까요?

    저렇게 인터셉터를 null 로 호출하니 실행해 보면 WebMvcConfig.java -> registry.addInterceptor(new CoreInterceptor(null))...

java.lang.NullPointerException: Cannot invoke "org.springframework.beans.factory.ObjectProvider.getObject()" because "this.myLoggerProvider" is null

위와 같은 에러가 발생합니다.

관련된 소스 첨부하였습니다.

도움 부탁 드립니다.

 

위의 소스는 제대로 된 걸로 다시 첨부하였습니다.

registry.addInterceptor(new CoreInterceptor(null))...

->

private final ObjectProvider<MyLogger> myLoggerProvider;

@Override
public void addInterceptors(InterceptorRegistry registry) {
    registry.addInterceptor(new CoreInterceptor(myLoggerProvider)).excludePathPatterns("/css/**", "/images/**", "/js/**");
}

 

위와 같이 수정하였습니다. 이런 저런 버그들과 강의에서 가르쳐 주신 ObjectProvider 를 사용해서 설정파일에서 request 가 들어오는 시점에 로그를 주입해서 처리되도록 수정하였습니다.

Interceptor 에서는 왜 ObjectProvider 를 사용하지 못할 거라고 생각했는지 모르겠네요. 답변 글 보고 질문을 수정하다가 생각이 나서 소스를 수정해 보니 잘 되는 것 같네요.

감사합니다.

답변 1

1

안녕하세요, 인프런 AI 인턴이에요.

MyLogger를 Interceptor로 구현하려고 하신다는 거죠?
문제가 생겨서 질문을 남기셨다면, 어떤 문제가 발생했는지 먼저 알려주세요. 그리고 MyLogger 클래스의 코드도 함께 알려주시면 더욱 도움이 될 것 같아요.

아직 해당 내용으로는 어떤 강의에서 문제가 발생하는지 알 수 없기 때문에, 뒤의 강좌에서 문제가 해결될 수 있는지에 대해 답변드리기는 어렵습니다. 하지만 강좌 내에서 이러한 문제를 다룰 예정이라면, 해당 내용은 적절한 타이밍에 다루게 될 것입니다.

아래의 코드 조각은 CoreInterceptor와 WebMvcConfig 클래스의 일부만 제공하셨습니다. 누락된 부분이 많아서 문제 발생 여부를 정확히 파악하기는 어려울 수 있습니다. 문제가 발생하는 부분에 대한 자세한 설명과 코드를 알려주세요. 그러면 더 정확한 도움을 드릴 수 있을 것입니다.

더 나아가서, MyLogger 클래스의 코드와 해당 인터셉터를 어떤 용도로 사용하려고 하는지도 함께 알려주시면 더욱 도움이 될 것입니다. 감사합니다.

diaman75님의 프로필 이미지
diaman75
질문자

잘 해결됐습니다. 답변 감사합니다.

diaman75님의 프로필 이미지
diaman75

작성한 질문수

질문하기