4

我正在尝试从 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 传递给记录器?

谢谢。

4

3 回答 3

2

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

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

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

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

于 2016-11-14T14:41:08.567 回答
0

v2.13.2建议创建 a而ContextDataProvider不是 a 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。

于 2020-09-18T14:12:04.213 回答
0

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

在 log4j2 中,我们必须使用 ThreadContext.put 而不是用于 log4j 的 MDC.put。这将完美地记录所有内容。休息一切都保持不变

于 2019-03-29T11:35:09.417 回答