Log4net - 在AdoNetAppender和RollingFileAppender之间动态切换appender

问题描述 投票:7回答:2

我在我的asp.net应用程序中使用AdoNetAppender(SQL服务器),并希望使用RollingFileAppender与SQL的任何连接问题。有没有办法配置只在AdoNetAppender出现问题时使用RollingFileAppender?

谢谢

log4net
2个回答
4
投票

在log4net中没有对这种故障转移方案的内置支持,问题是appender在log4net体系结构中相互隔离。

一个常见的设置是让两个appender并行记录,只是文件appender只保留一周的数据。如果AdoNetAppender失败,您将始终拥有文件中的最新数据。

但我明确地看到这里的情况是一个appender,它可以有一个sub-appender的优先级列表,在发生故障时进行一些简单的故障转移。在the AppenderSkeleton上构建这个应该不会太难。


1
投票

我已经实现了这样一个appender并在博客上发布了herehere(镜像)。代码可以找到here

我扩展了AppenderSkeleton并创建了一个名为FailoverAppender的新Appender,其中有两个类型为AppenderSkeleton的成员。

  • 一个名为“PrimaryAppender”的默认appender - 默认使用,直到失败。
  • 名为“FailoverAppender”的故障转移追加程序 - 仅在主要故障后使用。

使用log4net的xml配置语法配置PrimaryAppender和FailoverAppender的实际类型(请参阅下面的示例)。

一个片段:

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,它将传播内部appender异常以表示appender在内部发生故障,这将使LoggingEvents仅发送到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>
© www.soinside.com 2019 - 2024. All rights reserved.