日志回日志记录模式中的进程 ID

2022-09-02 10:12:16

我具有以下日志备份模式:

<pattern>
    {"hostname": "${HOSTNAME}", 
     "level": "%p", 
     "method": "%M", 
     "process_id": "${process}", 
     "thread_id": "%t", 
     "timestamp": "%d{Y-M-d}T%d{H:M:S.s}", 
     "mesg":"%msg"}%n
</pattern>

不幸的是,当日志消息实际生成时,我看到:"process_id": "process_IS_UNDEFINED"

是否有任何自动为进程 ID 设置的变量,例如为主机名设置的变量?我在日志返回文档中查找此类自动设置的变量的记录列表时遇到了很多麻烦,有人知道更好的文档来源吗?

编辑:我知道映射的诊断上下文,但希望有一个不需要这种设置的内置解决方案,就像主机名的工作方式一样。


答案 1

您可以使用映射的诊断上下文解决您的问题:

import org.slf4j.MDC;

public class Main {
    public static void main(String... args) {
        // put process ID early
        MDC.put("process_id", 
            ManagementFactory.getRuntimeMXBean().getName());
    }
}

之后,您只需要按如下方式重新定义您的模式:

<pattern>{..., "process_id": "%X{process_id}"}</pattern>

编辑

您也可以创建自己的编码器和转换器,并在以下位置使用它们:logback.xml

import ch.qos.logback.classic.PatternLayout;
import ch.qos.logback.classic.encoder.PatternLayoutEncoder;

public class ExtendedPatternLayoutEncoder extends PatternLayoutEncoder {
    @Override
    public void start() {
        // put your converter
        PatternLayout.defaultConverterMap.put(
            "process_id", ProcessIdConverter.class.getName());
        super.start();
    }
}
import ch.qos.logback.classic.pattern.ClassicConverter;
import ch.qos.logback.classic.spi.ILoggingEvent;

import java.lang.management.ManagementFactory;

public class ProcessIdConverter extends ClassicConverter {
    private static final String PROCESS_ID =
            ManagementFactory.getRuntimeMXBean().getName();

    @Override
    public String convert(final ILoggingEvent event) {
        // for every logging event return processId from mx bean
        // (or better alternative)
        return PROCESS_ID;
    }
}
<encoder class="some.package.ExtendedPatternLayoutEncoder">
    <pattern>{..., "process_id": "%process_id"}</pattern>
</encoder>

完整示例:

    <encoder class="some.package.ExtendedPatternLayoutEncoder">
        <pattern>%d{dd.MM.yyyy HH:mm:ss.SSS} PID:%process_id [%thread] %-5level %logger{36} - %msg%n</pattern>
    </encoder>

答案 2

有一个比@vsminkov更好的解决方案。我在这里找到了它:布局,如果它说创建自定义转换说明符。基本上,您可以创建转换器,但不是扩展,而是将转换规则添加到配置中:PatternLayoutEncoder

<configuration>

  <conversionRule conversionWord="pid" 
                  converterClass="my.custom.converter.ProcessIdConverter" />
  <conversionRule conversionWord="processId" 
                  converterClass="my.custom.converter.ProcessIdConverter" />

  <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
    <encoder>
      <pattern>%-6pid [%thread] - %msg%n</pattern>
    </encoder>
  </appender>
  ...
</configuration>

这样您就可以摆脱编码器


推荐