繁体   English   中英

Log4J2 JdbcAppender 没有编写正确的 INSERT 语句

[英]Log4J2 JdbcAppender doesn't compose correct INSERT statement

我正在尝试在我的 Spring 应用程序中使用这个汇集的 JDBC 记录器登录 PostgreSQL 数据库。

主要思想来自这里: How to initialize log4j with Spring Boot application?

@Component
public class JdbcAppenderConfiguration implements 
                     ApplicationListener<ContextRefreshedEvent> {

  private static Logger LOGGER = 
      LogManager.getLogger(JdbcAppenderConfiguration.class);

  private final DataSource dataSource;

  public JdbcAppenderConfiguration(DataSource dataSource) {
      this.dataSource = dataSource;
  }

  @Override
  public void onApplicationEvent(ContextRefreshedEvent event) {

    final LoggerContext ctx = (LoggerContext) LogManager.getContext(false);
    final ColumnConfig[] cc = {
        ColumnConfig.newBuilder()
            .setConfiguration(ctx.getConfiguration())
            .setName("level")
            .setPattern("%level")
            .setUnicode(false)
            .build()
    };

    final Appender appender = JdbcAppender
        .newBuilder()
        .setName("databaseAppender")
        .setIgnoreExceptions(false)
        .setConnectionSource(new ConnectionFactory(dataSource))
        .setTableName("logs2")
        .setColumnConfigs(cc)
        .build();

    appender.start();

    ctx.getConfiguration().addAppender(appender);

    final LoggerConfig loggerConfig =
        ctx.getConfiguration()
           .getLoggerConfig(LogManager.ROOT_LOGGER_NAME);
    loggerConfig.addAppender(appender, null, null);

    ctx.updateLoggers();

  }

}

和连接工厂:

public class ConnectionFactory extends AbstractConnectionSource {

  private final DataSource dataSource;

  public ConnectionFactory(DataSource dataSource) {
    AssertUtils.notNull(dataSource, "dataSource");
    this.dataSource = dataSource;
  }

  @Override
  public Connection getConnection() throws SQLException {
    return dataSource.getConnection();
  }

}

但是在记录时我得到这个错误:

2020-12-12 08:03:05,860 restartedMain 错误无法写入数据库 [JdbcManager{name=databaseAppender, bufferSize=0, tableName=logs2, columnConfigs=[{ name=level, layout=%level, literal=null, timestamp =false }], columnMappings=null}] 用于附加程序 [databaseAppender]。 org.apache.logging.log4j.core.appender.db.DbAppenderLoggingException: Failed to insert record for log event in JDBC manager: org.postgresql.util.PSQLException: ERROR: syntax error at or near ")" Pozice: 20 [columnConfigs =[{ name=level, layout=%level, literal=null, timestamp=false }], sqlStatement=insert into logs2 () values (?), factoryData=FactoryData [connectionSource=ConnectionFactory@ca2c455, tableName=logs2, columnConfigs= [{ name=level, layout=%level, literal=null, timestamp=false }], columnMappings=null, immediateFail=false, retry=true, reconnectIntervalMillis=5000, truncateStrings=true], connection=HikariProxyConnection@1535828764 包装组织。 postgresql.jdbc.PgConnection@603942ae,statement=HikariProxyPreparedStatement@1512335087 包装插入到logs2()值('INFO'),reconnector=null,isBatchSupported=true列元数据={}]

无效的 SQL INSERT 语句存在问题:

sqlStatement=insert into logs2 () values (?),

问题:您知道为什么 log4J 不能正确/完整地编写命令吗? 我的代码有什么问题?

这是 Log4j2 中的一个错误。

I set up a quick project in IntelliJ, added the log4j2 2.14 api and core JARs and a database driver JAR and ran your code, and I was able to reproduce the incorrectly-formatted INSERT statement in your error message.

我使用 IntelliJ 下载了 log4j2 源代码 JAR 并将其附加,并逐步执行了将INSERT语句放在一起的代码。 将列名称添加到 SQL 字符串的方法称为appendColumnNames并且位于 class JdbcDatabaseManager 它包含以下代码,为了清楚起见,我对其进行了缩写:

        if (data.columnMappings != null) {
            for (final ColumnMapping colMapping : data.columnMappings) {
                // append column from colMapping
            }
            if (data.columnConfigs != null) {
                for (final ColumnConfig colConfig : data.columnConfigs) {
                    // append column from colConfig
                }
            }
        }

在这种情况下,我们可以看到如果ColumnMappingnull ,则不使用来自ColumnConfig的列名。 这种行为是不正确的:您应该只能指定ColumnConfig 第二个if不应该嵌套在第一个里面。

Log4j2 支持使用ColumnMappingColumnConfig来指定要在日志记录表中写入的列,因此一种解决方法是改用它们。 另一种解决方法是添加该行

        .setColumnMappings(new ColumnMapping[0])

构建您的JdbcAppender 我对您的代码进行了此更改,Log4j2 生成了正确的 SQL 字符串。

暂无
暂无

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

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