[Spring] Sl4fj MDC로 요청 구분하기
Slf4j MDC란?
Slf4j(Simple Logging Facade for Java)는 자바 로깅 라이브러리 인터페이스고,
MDC(Mapped Diagnostic Context)는 Sl4fj에서 제공하는 기능으로 Thread Local에 정보를 Map 형태로 저장할 수 있다.
Thread Local은 각 스레드가 따로 가지는 저장 공간이라고 생각하면 된다.
그러면 Thread Local에 값을 저장하면 되는거 아니야? 라고 할 수 있는데, Map으로 key-value를 관리하기에 더 다루기 편한 것도 있고
로깅 시스템과 통합돼있어서 다루기 편리하다. 필요에 따라 context를 복사해야할 때도 있는데, 이에 대한 기능도 제공해 간편하다.
이걸 왜 써야 하나요?
요청마다 타는 로직이 다를 수 있다. 그런데 이런 요청이 너무 많아진다면? 서버가 싱글 스레드로 도는게 아니라서 순식간에 로그가
뒤죽박죽 섞일 것이다. 이때 장애가 발생하면 어떤 로직을 탔는지 확인이 필요할 때 요청을 구분할 수 없기때문에 매우 곤란해진다.
그래서 로그를 봤을 때 어느 요청에 의한 로그인지 구분할 수 있어야 한다. 요청을 구분할 수 있는 정보를 스레드별로 관리하려고
Thread Local을 쓰는 거고, 동시에 로깅에 편리한 방법이 MDC를 사용하는 것이다.
MDC 적용하기
저장은 MDC.put(key, value)
, 조회는 MDC.get(key)
를 쓰면 된다. (String만 가능)
MDC.put("hi", "hello")
val result = MDC.get("hi")
logger.info("result = $result") // "hello"
Servlet Filter
요청마다 UUID
를 생성해 요청을 구분한다고 해보자. 그러면 Controller마다 UUID를 생성해서 MDC.put()
을 호출해야 할까?
가능은 하지만 수정 범위가 너무 많고 실수할 여지도 많다. 그렇게 하지말고 Servlet Filter를 만들어 등록해주자.
@Component
class MdcFilter: Filter {
override fun init(filterConfig: FilterConfig?) {
super.init(filterConfig)
}
override fun doFilter(request: ServletRequest, response: ServletResponse, chain: FilterChain) {
val traceId = UUID.randomUUID().toString().substring(0, 6) // 요청을 구분할 UUID 앞 6자리
MDC.put("traceId", traceId) // 요청이 들어올 때마다 MDC에 저장해주자
chain.doFilter(request, response)
}
}
- 이렇게 하면 요청이 들어올 때마다
MdcFilter
를 거치면서 요청을 구분할 수 있게 MDC에traceId
를 저장함
MDC.get("traceId")
도 로그에 직접 넣어주지 말고, logback-spring.xml 파일을 resources에 만들어주자.
<?xml version="1.0" encoding="UTF-8" ?>
<configuration>
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<pattern>[%X{traceId:-NO TRACE ID}] %msg%n</pattern>
</encoder>
</appender>
<root>
<level value="INFO" />
<appender-ref ref="STDOUT" />
</root>
</configuration>
- pattern 태그를 보면,
%X{키 이름}
을 적어주면 MDC에서{키 이름}
으로 저장된 값을 알아서 등록해줌 - MDC에
{키 이름}
으로 등록된 값이 없으면%X{키 이름:-기본값}
으로기본값
을 설정해줄 수 있음
그러면 로그에 MDC 정도를 직접 적어주지 않아도 같이 출력된다.
logger.info("hello")
>> [6f56cf] hello
6f56cf
로 요청을 구분하면 된다!
스레드가 달라지면 MDC가 달라진다
같은 요청인데 다른 스레드를 사용하는 케이스가 뭐가 있을까?
EventListener에 @Async가 붙어서 새 스레드로 동작하는 경우가 있고, 다른 context의 코루틴에서 동작하는 경우가 있다.
커스텀 스레드풀 등록하기 (MDC를 곁들인)
AsyncConfigurer
를 구현한 스레드풀 설정을 등록해놓으면, @Async처럼 새 스레드에서 동작하는 경우 등록한 스레드풀을 사용하게
만들 수 있다. 이때 스레드풀에 있는 스레드들도 같은 MDC를 품을 수 있도록 decorator를 적용해야한다.
class MdcTaskDecorator: TaskDecorator {
override fun decorate(runnable: Runnable): Runnable {
val context = MDC.getCopyOfContextMap() // MDC 복사
return Runnable {
MDC.setContextMap(context) // 새 스레드 MDC에 저장
runnable.run()
}
}
}
AsyncConfigurer
를 구현한 설정 클래스 AsyncConfig
도 만들어서 등록해주자.
@Configuration
@EnableAsync
class AsyncConfig : AsyncConfigurer {
@Bean
fun customTaskExecutor(): ThreadPoolTaskExecutor {
val taskExecutor = ThreadPoolTaskExecutor()
// ... 스레드풀 설정 생략
taskExecutor.setTaskDecorator(MdcDecorator()) // decorator 설정
taskExecutor.initialize()
return taskExecutor
}
override fun getAsyncExecutor(): Executor {
return customTaskExecutor()
}
}
하지만 이 방법은 delay()
같은 중단점을 만나 다른 스레드로 전환되는 경우 MDC에 넣어둔 값을 유지하지 못한다.
아래에서 설명하는 MDCContext() 를 사용한다면 해결할 수 있다.
다른 스레드에서 동작하는 코루틴
부모의 context를 물려받는 코루틴이라면 MDC가 유지돼서 괜찮다.
예) 파라미터가 없는 runBlocking, launch 등
하지만 withContext()
를 사용해서 다른 스레드풀을 사용한다면?
launch
로 코루틴은 생성할 건데 파라미터로 다른 스레드풀을 넣어준다면?
이때는 다른 스레드에서 동작하는거니까 MDC가 달라진다. 여기는 위에처럼 decorator 설정을 추가해주지 않아도 된다.
MDCContext()를 사용해서 간편하게 기존 MDC를 복사할 수 있다. CoroutineContext
에 + 연산자 오버로딩이 된다!
이렇게 기존 context(=CoroutineContext)에 복사할 수 있는 건 MDCContext의 updateThreadContext()
를 호출하기
때문이다.
해당 로직은 kotlinx-coroutines-slf4j에서 참조할 수 있다.
아래처럼 설정해줄 수 있다.
suspend fun callInCoroutine() {
// [0995ec] hi from withContext
withContext(Dispatchers.IO + MDCContext()) {
logger.info("hi from withContext")
}
// [0995ec] hi from launch
coroutineScope {
launch(Dispatchers.IO + MDCContext()) {
logger.info("hi from launch")
}
}
// [NO TRACE ID] hi from launch in IO Dispatcher
coroutineScope {
launch(Dispatchers.IO) {
logger.info("hi from launch in IO Dispatcher")
}
}
}
launch(Dispatchers.IO)
에서NO TRACE ID
로 나온 이유는 Dispatchers.IO 스레드풀의 스레드는 기존 MDC가 없기 때문
댓글남기기