我在我的asp.net应用程序中使用AdoNetAppender(SQL服务器),并希望使用RollingFileAppender与SQL的任何连接问题。有没有办法配置只在AdoNetAppender出现问题时使用RollingFileAppender?
谢谢
由
在log4net中没有对这种故障转移方案的内置支持,问题是appender在log4net体系结构中相互隔离。
一个常见的设置是让两个appender并行记录,只是文件appender只保留一周的数据。如果AdoNetAppender失败,您将始终拥有文件中的最新数据。
但我明确地看到这里的情况是一个appender,它可以有一个sub-appender的优先级列表,在发生故障时进行一些简单的故障转移。在the AppenderSkeleton上构建这个应该不会太难。
我已经实现了这样一个appender并在博客上发布了here和here(镜像)。代码可以找到here。
我扩展了AppenderSkeleton并创建了一个名为FailoverAppender的新Appender,其中有两个类型为AppenderSkeleton的成员。
使用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>