繁体   English   中英

将上下文传递给异步记录器

[英]Pass context to async Logger

我正在尝试记录来自 http 客户端的原始请求/响应。 我正在遵循这些 日志记录说明中的log4j2 配置。

HttpAsync 客户端依赖:- httpasyncclient(版本 4.1.1)

log4j2.xml

<?xml version="1.0" encoding="UTF-8"?>
<Configuration status="WARN">
<Appenders>
    <RollingRandomAccessFile name="app-log" fileName="${log.path}/app.log"
                             filePattern="${log.path}/app-%d{yyyy-MM-dd}.gz">
        <PatternLayout>
            <pattern>[%-5level] [%X{uuid}] %d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %c{1} - %msg%n</pattern>
        </PatternLayout>
        <Policies>
            <TimeBasedTriggeringPolicy interval="1" modulate="true"/>
        </Policies>
    </RollingRandomAccessFile>
</Appenders>
<Loggers>
    <AsyncLogger name="org.apache.http.impl.conn.Wire" level="debug">
        <AppenderRef ref="app-log"/>
    </AsyncLogger>
    <AsyncRoot level="debug" includeLocation="true">
        <AppenderRef ref="app-log"/>
    </AsyncRoot>
</Loggers>
</Configuration>

它打印正常,但线程上下文没有传递到线记录器上。

示例:-

// with uuid, output of logger.debug(ThreadContext.getImmutableContext().toString());

[DEBUG] [c48b97f7-0094-44af-82af-3d6b43d76014] 2016-11-14 17:06:03.408 [http-bio-8080-exec-1] OutboundRequestHandler - {uuid=c48b97f7-0094-44af-82af-3d6b43d76014}

// without uuid
[DEBUG] [] 2016-11-14 17:06:03.440 [I/O dispatcher 1] headers - http-outgoing-0 >> POST /abcd.json HTTP/1.1
[DEBUG] [] 2016-11-14 17:06:03.441 [I/O dispatcher 1] headers - http-outgoing-0 >> Content-Length: 2
[DEBUG] [] 2016-11-14 17:06:03.441 [I/O dispatcher 1] headers - http-outgoing-0 >> Content-Type: text/plain; charset=ISO-8859-1
[DEBUG] [] 2016-11-14 17:06:03.441 [I/O dispatcher 1] headers - http-outgoing-0 >> Host: 127.0.0.1:80
[DEBUG] [] 2016-11-14 17:06:03.441 [I/O dispatcher 1] headers - http-outgoing-0 >> Connection: Keep-Alive
[DEBUG] [] 2016-11-14 17:06:03.441 [I/O dispatcher 1] headers - http-outgoing-0 >> User-Agent: Apache-HttpAsyncClient/4.1.1 (Java/1.8.0_92)

如何将 ThreadContext 传递给记录器?

谢谢。

由于设置 UUID 的线程 (http-bio-8080-exec-1) 与应用程序中执行日志记录的线程(I/O 调度程序 1)不同,因此它们具有不同的 ThreadContext 映射,并且第二个线程不能看看第一个线程在它的地图中放了什么。

从 log4j 2.7 开始,可以创建一个自定义上下文数据注入器,它可以从 ThreadLocal 映射之外的其他地方获取上下文数据(因为这本质上就是 ThreadContext )。

这确实意味着您需要创建一个自定义外观,类似于 log4j 的 ThreadContext,您可以在其中将键值对放入某些数据结构中。 我不熟悉 AsyncHttpClient,但我找不到允许不同线程共享与同一会话关联的数据的“上下文”概念。

您的自定义上下文注入器实现将获取键值对的快照,并将其注入到每个日志消息的 LogEvent 中。

我不确定这在 2019 年有多重要,因为提问者肯定会转移到其他事情上。 但它仍然在这里:

在 log4j2 中,我们必须使用 ThreadContext.put 而不是用于 log4j 的 MDC.put。 这将完美地记录一切。 休息一切照旧

v2.13.2 ,建议创建ContextDataProvider而不是ContextDataInjector

ContextDataProvider(在 Log4j 2.13.2 中引入)是一个接口应用程序和库,可用于将额外的键值对注入到 LogEvent 的上下文数据中。 Log4j 的 ThreadContextDataInjector 使用 java.util.ServiceLoader 来定位和加载 ContextDataProvider 实例。 Log4j 本身使用 org.apache.logging.log4j.core.impl.ThreadContextDataProvider 将 ThreadContextData 添加到 LogEvent。 自定义实现应实现 org.apache.logging.log4j.core.util.ContextDataProvider 接口,并通过在名为 META-INF/services/org.apache.logging.log4j.core 的文件中定义实现类来将其声明为服务。 util.ContextDataProvider。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM