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

时间:2009-05-07 18:53:14

标签: log4net

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

由于

POR

2 个答案:

答案 0 :(得分:4)

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

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

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

答案 1 :(得分:1)

我已经实现了这样的附加程序,并在herehere(镜像)上写了博客。可以找到代码here

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

  • 称为“ PrimaryAppender”的默认附加器-默认使用,直到失败。
  • 称为“ FailoverAppender”的故障转移附加程序-仅在主节点发生故障后使用。

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

摘要:

public class FailoverAppender : AppenderSkeleton
{
    private AppenderSkeleton _primaryAppender;
    private AppenderSkeleton _failOverAppender;
     ....
}

在实现Append方法时,默认情况下,我仅将LoggingEvents发送到PrimaryAppender并用try-catch包围它。如果PrimaryAppender抛出(失败),我将发出信号信号并将LoggingEvent发送到FailoverAppender。 下一个LoggingEvent将直接发送给FailoverAppender。

protected override void Append(LoggingEvent loggingEvent)
{
    if (LogToFailOverAppender)
    {
        _failOverAppender?.DoAppend(loggingEvent);
    }
    else
    {
        try
        {
            _primaryAppender?.DoAppend(loggingEvent);
        }
        catch
        {
            ActivateFailOverMode();
            Append(loggingEvent);
        }
    }
}

此外,我创建了一个自定义ErrorHandler,它将传播内部附加程序异常,以表明附加程序在内部发生故障,这将使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>