Log4Net 将两个不同级别的日志记录到同一记录器的两个不同附加程序

Posted

技术标签:

【中文标题】Log4Net 将两个不同级别的日志记录到同一记录器的两个不同附加程序【英文标题】:Log4Net Logging of two different levels to two different appenders for the same logger 【发布时间】:2014-02-06 19:28:56 【问题描述】:

我们有两个启用了 Log4net 日志记录的不同 asp.net 应用程序。 它们都有相同版本的 Log4Net,1.2.10.0。

我们已将log4net.Appender.AdoNetAppender 记录器添加到它们两者中,并希望将根记录器的信息级别记录到它,但也希望将根记录器的错误级别记录到文件追加器。 我们的配置如下;

<?xml version="1.0" encoding="utf-8"?>

<log4net>
    <appender name="filelogAppender" type="log4net.Appender.RollingFileAppender" >
        <file value="..\logs\app.debug.log" />
        <encoding value="utf-8" />
        <staticLogFileName value="true" />
        <datePattern value=".yyyyMMdd.'log'" />
        <rollingStyle value="Composite" />
        <lockingModel type="log4net.Appender.FileAppender+MinimalLock" />
        <appendToFile value="true" />
        <maximumFileSize value="1MB" />
        <layout type="log4net.Layout.PatternLayout">
            <conversionPattern value="%date [%3thread] %-5level %propertylog4net:HostName [%propertyRevision] %logger %message%n" />
        </layout>
    </appender>
    <!--    
        use [DB]
        GO
        CREATE TABLE [dbo].[Log] (
            [Id] [int] IDENTITY (1, 1) NOT NULL,
            [Date] [datetime] NOT NULL,
            [Application][varchar] (255) NOT NULL,
            [Thread] [varchar] (255) NOT NULL,
            [Level] [varchar] (50) NOT NULL,
            [Logger] [varchar] (255) NOT NULL,
            [Server][varchar](255) NOT NULL,
            [Revision][varchar](50) NOT NULL,
            [Message] [varchar] (4000) NOT NULL,
            [Exception] [varchar] (2000) NULL
        )
    -->
    <appender name="dbLogAppender" type="log4net.Appender.AdoNetAppender" xdt:Transform="InsertBefore(/log4net/root)">
        <bufferSize value="100" />
        <connectionType value="System.Data.SqlClient.SqlConnection, System.Data, Version=1.0.3300.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" />
        <connectionString value="data source=sqlserver;initial catalog=DB;integrated security=false;persist security info=True;User ID=USER;Password=PASSWORD" />
        <commandText value="INSERT INTO Log ([Date],[Application],[Thread],[Level],[Logger],[Server],[Revision],[Message],[Exception]) VALUES (@log_date,'WebApp1', @thread, @log_level, @logger, @server, @revision, @message, @exception)" />
        <parameter>
            <parameterName value="@log_date" />
            <dbType value="DateTime" />
            <layout type="log4net.Layout.RawTimeStampLayout" />
        </parameter>
        <parameter>
            <parameterName value="@thread" />
            <dbType value="String" />
            <size value="255" />
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%thread" />
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@log_level" />
            <dbType value="String" />
            <size value="50" />
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%level" />
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@logger" />
            <dbType value="String" />
            <size value="255" />
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%logger" />
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@server" />
            <dbType value="String" />
            <size value="255" />
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%propertylog4net:HostName"/>
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@revision" />
            <dbType value="String" />
            <size value="50" />
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%propertyRevision"/>
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@message" />
            <dbType value="String" />
            <size value="4000" />
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%message" />
            </layout>
        </parameter>
        <parameter>
            <parameterName value="@exception" />
            <dbType value="String" />
            <size value="2000" />
            <layout type="log4net.Layout.ExceptionLayout" />
        </parameter>
    </appender>
    <root>
        <level value="Error"/>
        <appender-ref ref="filelogAppender"/>
    </root>
    <root>
        <level value="Info"/>
        <appender-ref ref="dbLogAppender"/>
    </root>
</log4net>  

我们看到的问题是,对于一个应用程序,我们在数据库中看到条目,但对于另一个我们看不到。

Revision 是我们在两个应用程序中在Application_Start() 中设置的GlobalContext 属性。而这两种配置的唯一区别是Application 的硬编码值不同。

我们可以有两个这样的根记录器吗?这可能是我们在一个应用程序中看到的问题的根源吗?

【问题讨论】:

【参考方案1】:

您应该能够分别设置每个 appender 的 threshold 属性并将它们包含在同一个根目录中。

<appender name="filelogAppender" type="log4net.Appender.RollingFileAppender">
  <threshold value="Error" />
</appender>
<appender name="dblogAppender" type="log4net.Appender.AdoNetAppender">
  <threshold value="Info" />
</appender>
<root>
  <appender-ref ref="filelogAppender" />
  <appender-ref ref="dblogAppender" />
</root>

reference

【讨论】:

【参考方案2】:

您可以为每个附加程序指定一个LevelRangeFilter,并在根目录中定义每个附加程序以在不同级别记录。

<appender name="filelogAppender" type="log4net.Appender.RollingFileAppender" >
    <!--File Details/Layout Options-->
    <filter type="log4net.Filter.LevelRangeFilter">
        <levelMin value="INFO"/>
    </filter>
</appender>
<appender name="dbLogAppender" type="log4net.Appender.AdoNetAppender">
    <!--SQL Options-->      
    <filter type="log4net.Filter.LevelRangeFilter">
        <levelMin value="ERROR"/>
    </filter>
</appender>
<root>
    <level value="INFO"/>
    <appender-ref ref="filelogAppender" /> <!--Prints info, warn, error, or fatal logs. -->
    <appender-ref ref="dbLogAppender" />   <!--Prints only error or fatal logs. -->
</root>

【讨论】:

一个很好的答案!并提供了我完全不知道的灵活性。我想我会使用@barry 答案,因为我不需要定义一个范围只是一个阈值。 由于某种原因,“threshold”和“evaluator”节点对我不起作用,但确实如此(我将 levelMin 和 levelMax 指定为相同的值)。 我只是遇到了一个问题,即一旦在 appender 中使用 levelMin,就必须在所有 appender 中指定它,否则过滤在我的情况下无法正常工作,并且只考虑了根级别。 【参考方案3】:

不允许有两个 &lt;root&gt; 元素。

来自documentation:

root     Optional element, maximum of one allowed. Defines the configuration of the root logger.

但是,您可以像您一样拥有一个根记录器,然后拥有一个由命名空间指定的单独记录器。

<root>
    <level value="Error"/>
    <appender-ref ref="filelogAppender"/>
</root>
<logger name="MyCompany.MyApp.Namespace">
    <level value="Info"/>
    <appender-ref ref="dbLogAppender"/>
</logger>

【讨论】:

但是,我们想要将 root 登录到两个位置。我们不想限制记录特定的命名空间。因此我们尝试使用两个根记录器。 [奇怪地在其中一个应用上工作] 否则我们将需要为每个可能的根命名空间创建大量记录器,并将 additivity 设置为 false。【参考方案4】:

只需使用 Evaluator,不需要在所有的 appender 中都给出阈​​值。

<appender name="filelogAppender" type="log4net.Appender.RollingFileAppender">
        <evaluator type="log4net.Core.LevelEvaluator">
          <threshold value="ERROR" />
  </evaluator>
</appender>

【讨论】:

以上是关于Log4Net 将两个不同级别的日志记录到同一记录器的两个不同附加程序的主要内容,如果未能解决你的问题,请参考以下文章

.Net Log4Net配置多文件日志记录

log4net使用详解

我如何设置 log4net 每天将我的文件记录到不同的文件夹中?

转:log4net使用详解

Log4Net之记录日志到数据库

log4net日志系统使用详解