基于级别,配置log4net将错误信息发送到不同的附加器

43

我希望将 Info Level 及以上级别的日志发送到 XML appender,而将 Error/Fatal Level 级别的日志发送到 EventLog appender。

我了解到需要修改配置文件的根元素,但是语法让我很困惑。如何在配置文件中指定特定级别或级别范围的日志记录到正确的 appender 中?

以下是当前的配置:

<log4net>
  <appender name="SomeXmlAppender" type="log4net.Appender.RollingFileAppender">
    ...
  </appender>
  <appender name="SomeEventLogAppender" type="log4net.Appender.EventLogAppender">
    ...
  </appender>
  <root>
    <level value="DEBUG" />
    <appender-ref ref="SomeXmlAppender" />
    <appender-ref ref="SomeEventLogAppender" />
  </root>
</log4net>

编辑:感谢@agileguy。那篇帖子确实包含了我所需的语法。现在的解决方案如下:

<log4net>
  <appender name="SomeXmlAppender" type="log4net.Appender.RollingFileAppender">
    ...
    <evaluator type="log4net.Core.LevelEvaluator">
      <threshold value="INFO"/>
    </evaluator>
    <filter type="log4net.Filter.LevelRangeFilter">
      <levelMin value="INFO" />
      <acceptOnMatch value="true" />
    </filter>
    <filter type="log4net.Filter.DenyAllFilter" />
  </appender>
  <appender name="SomeEventLogAppender" type="log4net.Appender.EventLogAppender">
    ...
    <evaluator type="log4net.Core.LevelEvaluator">
      <threshold value="ERROR"/>
    </evaluator>
    <filter type="log4net.Filter.LevelRangeFilter">
      <levelMin value="ERROR" />
      <acceptOnMatch value="true" />
    </filter>
    <filter type="log4net.Filter.DenyAllFilter" />
  </appender>
  <root>
    <level value="DEBUG" />
    <appender-ref ref="SomeXmlAppender" />
    <appender-ref ref="SomeEventLogAppender" />
  </root>
</log4net>

我认为Graham先生在这里的文章(http://weblogs.asp.net/tgraham/archive/2007/03/15/a-realistic-log4net-config.aspx)会对你有所帮助。 - Daniel Elliott
5
请注意,只有支持缓冲的记录器才会使用评估器。如果您想为每个记录器指定特定的级别,只需要单独使用<threshold>元素即可——不需要将其包装在评估器中。 - Brian
<evaluator> 元素应仅用于实现 BufferingAppenderSkeleton 的 appender(即不包括 RollingFileAppender)。 - dave
3个回答

15

可以使用 appender 中的 thresholdfilter 元素来完成此操作。

请注意,阈值可以直接放在 appender 下面,作为一个包含过滤器,或者放在 evaluator 下面。

<evaluator type="log4net.Core.LevelEvaluator">
  <threshold value="ERROR"/>
</evaluator>

在适用的情况下,它充当一个包容性过滤器,用于跳过缓冲(立即输出)。



完整解释(来源):

<threshold value="ERROR" />
阈值是在AppenderSkeleton中实现的,因此几乎所有的追加器都支持它。这只是一个简单的测试,用于忽略级别低于阈值的日志事件。阈值会早早地进行检查,因为它是一个简单的测试,所以性能非常高效。
还有另一种通过过滤器来指定与阈值相同行为的方法。过滤器更加灵活,并且可以插入自定义逻辑到过滤器链中,因为它们是可插拔的。
<filter type="log4net.Filter.LevelRangeFilter">
    <levelMin value="ERROR" />
    <levelMax value="OFF" />
</filter>

和阈值检查一样,过滤器(Filter)也是在AppenderSkelton基类中实现的,并且几乎所有的Appender都支持它们。上面的过滤器(Filter)与<threshold value="ERROR" />具有相同的效果。它是一个LevelRangeFilter,将允许通过任何级别在ERROROFF(包括这两者)之间的事件。请注意,OFF是最高级别的名称,而ALL则是最低级别的名称。

过滤器(Filter)非常灵活,因为可以将多个过滤器(Filter)链接在一起,以提供对输出事件的细粒度控制。由于这种灵活性,它们在性能方面的成本也更高,链中的每个过滤器(Filter)都是一个对象,并被要求决定正确的操作。在阈值过滤的简单情况下,应该优先使用Threshold属性而不是过滤器(Filter)。

Evaluator(评估器)BufferingAppenderSkeleton实现,因此仅受到扩展此基类并提供缓冲支持的Appender的支持。SmtpAppender就是这样一个appender(Appender)。

Evaluator(评估器)是一种可插拔的对象,由BufferingAppenderSkeleton用于确定是否不应缓冲日志事件,而应立即写入/发送。如果Evaluator(评估器)认为该事件很重要,则将整个当前缓冲区的内容与事件一起发送。evaluator(评估器)不像阈值或过滤器(Filter)那样丢弃事件。


6
您可以为每个appender设置不同的阈值属性。低于阈值级别的所有日志事件都将被该appender忽略。下面我将粘贴两个appenders,一个用于文件,另一个用于数据库(您需要设置连接字符串)。数据库appender有一个阈值属性,表示只会将错误保存到数据库中。
<configuration>
  <log4net>
    <!--Database appender-->
    <appender name="DbAppender" type="log4net.Appender.ADONetAppender">
      <bufferSize value="0" />
      <connectionType value="System.Data.SqlClient.SqlConnection, System.Data, Version=1.0.3300.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" />
      <connectionString value="Data Source=AEG-PC\SQLEXPRESS;Initial Catalog=JCZ6;Uid=sa;Pwd=qwerty;" />
      <commandText value="INSERT INTO Log4Net ([date],[thread],[level],[logger],[message],[exception]) VALUES 
         (@log_date, @thread, @log_level, @logger, @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="%t" />
        </layout>
      </parameter>
      <parameter>
        <parameterName value="@log_level" />
        <dbType value="String" />
        <size value="10" />
        <layout type="log4net.Layout.PatternLayout">
          <conversionPattern value="%p" />
        </layout>
      </parameter>
      <parameter>
        <parameterName value="@logger" />
        <dbType value="String" />
        <size value="255" />
        <layout type="log4net.Layout.PatternLayout">
          <conversionPattern value="%c" />
        </layout>
      </parameter>
      <parameter>
        <parameterName value="@message" />
        <dbType value="String" />
        <size value="4000" />
        <layout type="log4net.Layout.PatternLayout">
          <conversionPattern value="%m" />
        </layout>
      </parameter>
      <parameter>
        <parameterName value="@exception" />
        <dbType value="String" />
        <size value="4000" />
        <layout type="log4net.Layout.ExceptionLayout" />
      </parameter>
      <threshold value="Error" /> <!--THIS IS THE IMPORTANT LINE-->
    </appender>
    <!--File appender-->
    <appender name="FileAppender" type="log4net.Appender.FileAppender">
      <file value="log-file.txt" />
      <appendToFile value="true" />
      <layout type="log4net.Layout.PatternLayout">
        <conversionPattern value="%date [%thread] %-5level %logger [%property{NDC}] - %message%newline" />
      </layout>
    </appender>
    <root>
      <level value="INFO" />
      <appender-ref ref="FileAppender" />
      <appender-ref ref="DbAppender" />
    </root>
  </log4net>
</configuration>

2

我有同样的问题。如果我理解原始问题的话,阈值不起作用,因为它会将某些输出发送到一个 appender,将其余部分发送到另一个 appender。如上所建议,我能够使用 LevelRangeFilter 使其正常工作。我希望 ERROR、INFO 和 WARN 能够发送到一个 appender,而其他所有内容则发送到另一个 appender,但不包括 ERROR、INFO 和 WARN。

这是对我有用的配置:

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
    <configSections>
        <section name="log4net" type="log4net.Config.Log4NetConfigurationSectionHandler, log4net" />
    </configSections>
    <log4net>
        <appender name="LoggingAppender" type="log4net.Appender.FileAppender" >
            <file value="logs.txt" />
            <filter type="log4net.Filter.LevelRangeFilter">
                <levelMin value="INFO"/>
                <levelMax value="OFF"/>
            </filter>
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%date [%thread] %-5level %logger [%ndc] - %message%newline" />
            </layout>
        </appender>
        <appender name="TracingAppender" type="log4net.Appender.FileAppender" >
            <file value="traces.txt" />
            <filter type="log4net.Filter.LevelRangeFilter">
                <levelMin value="ALL"/>
                <levelMax value="DEBUG"/>
            </filter>
            <layout type="log4net.Layout.PatternLayout">
                <conversionPattern value="%date [%thread] %-5level %logger [%ndc] - %message%newline" />
            </layout>
        </appender>
        <root>
            <appender-ref ref="LoggingAppender"/>
            <appender-ref ref="TracingAppender"/>
        </root>
    </log4net>
</configuration>

感谢您,Nick。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接