Log4net - 动态切换日志输出器(Appender)在AdoNetAppender和RollingFileAppender之间。

7
我在我的asp.net应用程序中使用AdoNetAppender (SQL server),如果出现与SQL的连接问题,我想使用RollingFileAppender。有没有一种方法可以配置只有在AdoNetAppender出现问题时才使用RollingFileAppender?
谢谢。
2个回答

4

log4net中没有针对此类故障转移场景的内置支持,问题在于在log4net架构中,附加器相互之间相当独立。

然而,常见的设置是使两个附加器并行记录,只不过文件附加器仅保留一周的数据。如果AdoNetAppender失败,您将始终拥有文件中最新的数据。

但我明确看到这里需要一个附加器,它可以具有子附加器的优先级列表,在发生故障时进行一些简单的故障转移。这应该不太难实现,可以建立在AppenderSkeleton的基础上。


感谢Peter宝贵的时间和出色的解释。我会看一下AppenderSkeletton。 - porhills

2
我已经实现了这样的一个appender并在这里这里(镜像)博客中介绍。代码可以在这里找到。
我扩展了AppenderSkeleton并创建了一个新的Appender,称为FailoverAppender,它有两个类型为AppenderSkeleton的成员。
  • 默认的appender称为“PrimaryAppender”- 默认情况下使用,直到失败。
  • 故障转移appender称为“FailoverAppender”-仅在主要appender失败后使用。
“PrimaryAppender”和“FailoverAppender”的实际类型是使用log4net的xml配置语法进行配置的(请参见下面的示例)。
代码片段:
public class FailoverAppender : AppenderSkeleton
{
    private AppenderSkeleton _primaryAppender;
    private AppenderSkeleton _failOverAppender;
     ....
}

在Append方法的实现中,默认情况下,我只将LoggingEvents发送到PrimaryAppender,并用try-catch包围它。如果PrimaryAppender抛出异常(失败),我会发出一个信号并将LoggingEvent发送到FailoverAppender。 接下来的LoggingEvents将直接且仅发送到FailoverAppender。
protected override void Append(LoggingEvent loggingEvent)
{
    if (LogToFailOverAppender)
    {
        _failOverAppender?.DoAppend(loggingEvent);
    }
    else
    {
        try
        {
            _primaryAppender?.DoAppend(loggingEvent);
        }
        catch
        {
            ActivateFailOverMode();
            Append(loggingEvent);
        }
    }
}

此外,我创建了一个自定义的ErrorHandler,它将传播内部附加器的异常以表示附加器已经在内部失败,这将使日志事件仅发送到FailoverAppender。
class FailOverErrorHandler : IErrorHandler
{
    public FailOverAppender FailOverAppender { get; set; }

    public FailOverErrorHandler(FailOverAppender failOverAppender)
    {
        FailOverAppender = failOverAppender;
    }

    public void Error(string message, Exception e, ErrorCode errorCode)
        => FailOverAppender.ActivateFailOverMode();

    public void Error(string message, Exception e)
        => FailOverAppender.ActivateFailOverMode();

    public void Error(string message)
        => FailOverAppender.ActivateFailOverMode();
}

配置示例:
<!--This custom appender handles failovers. If the first appender fails, it'll delegate the message to the back appender-->
<appender name="FailoverAppender" type="MoreAppenders.FailoverAppender">
    <layout type="log4net.Layout.PatternLayout">
        <conversionPattern value="%date [%thread] %-5level %logger - %message%newline"/>
    </layout>

    <!--This is a custom test appender that will always throw an exception -->
    <!--The first and the default appender that will be used.-->
    <PrimaryAppender type="MoreAppenders.ExceptionThrowerAppender" >
        <ThrowExceptionForCount value="1" />
        <layout type="log4net.Layout.PatternLayout">
            <conversionPattern value="%date [%thread] %-5level %logger - %message%newline"/>
        </layout>        
    </PrimaryAppender>

    <!--This appender will be used only if the PrimaryAppender has failed-->
    <FailOverAppender type="log4net.Appender.RollingFileAppender">
        <file value="log.txt"/>
        <rollingStyle value="Size"/>
        <maxSizeRollBackups value="10"/>
        <maximumFileSize value="100mb"/>
        <appendToFile value="true"/>
        <staticLogFileName value="true"/>
        <layout type="log4net.Layout.PatternLayout">
            <conversionPattern value="%date [%thread] %-5level %logger - %message%newline"/>
        </layout>
    </FailOverAppender>
</appender>

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