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