Программное добавление и удаление приложений журнала в log4net
У меня есть компонент, который использует log4net. Я хочу создать модульные тесты, которые подтверждают, что определенные условия ошибки приводят к правильной регистрации.
Я думал, что лучший способ сделать это - создать реализацию ILogAppender, например, макет. Затем я добавляю log appender в log4net во время настройки теста, проверяю, что было написано во время проверки теста, и снова удаляю его во время демонтажа теста.
Это возможно?
4 ответа
Я использую BasicConfigurator, настроенный с помощью MemoryAppender. Это приложение позволяет вам получить доступ к сообщениям в памяти, записанным во время теста.
Использование BasicConfigurator прекрасно подходит для модульного тестирования (то, что запросил ОП, но не то, что в теме). Другие ответы получают вывод для конкретного регистратора.
Я хотел все это (это была страница "самопроверки" на веб-сайте). В итоге я сделал в основном следующее:
var root = ((log4net.Repository.Hierarchy.Hierarchy)LogManager.GetRepository()).Root;
var attachable = root as IAppenderAttachable;
var appender = new log4net.Appender.MemoryAppender();
if(attachable!=null)
attachable.AddAppender(appender);
// do stuff
var loggingEvents = appender.GetEvents();
foreach (var loggingEvent in loggingEvents)
loggingEvent.WriteRenderedMessage(writer);
if(attachable!=null)
attachable.RemoveAppender(appender);
... но обернутый как Одноразовый согласно подходу @ Павла
ОБНОВЛЕНИЕ: ответ Павла был удален, поэтому я добавляю его ссылку здесь: Программно проверить журнал Log4Net.
Следующий код изначально был найден в архивах списков рассылки apache и должен решить проблему добавления и удаления в коде дополнений log4net
/// <summary>
/// dataLog
/// </summary>
protected static readonly IDeviceCommunicationsLog dataLog =
DeviceCommunicationsLogManager.GetLogger("LIS3.Data");
Each connection adds and removes a file appender programmatically:
/// <summary>
/// add connection specific appender
/// </summary>
void AddAppender()
{
// check if logging is endabled
if( this.IsLoggingEnabled() )
{
try
{
// get the interface
IAppenderAttachable connectionAppender = (IAppenderAttachable)this.DataLog.Logger;
// need some application configuration settings
NameValueCollection appSettings = ConfigurationSettings.AppSettings;
// get the layout string
string log4netLayoutString = appSettings["log4net.LIS3.LayoutString"];
if( log4netLayoutString == null )
{
// use default setting
log4netLayoutString = "%d [%x]%n %m%n %P MessageData}%n%n";
}
// get logging path
string log4netPath = appSettings["log4net.Path"];
if( log4netPath == null )
{
// use default path
log4netPath = ".\\";
}
// create the appender
this.rollingFileAppender = new RollingFileAppender();
// setup the appender
this.rollingFileAppender.MaxFileSize = 10000000;
this.rollingFileAppender.MaxSizeRollBackups = 2;
this.rollingFileAppender.RollingStyle = RollingFileAppender.RollingMode.Size;
this.rollingFileAppender.StaticLogFileName = true;
string appenderPath = LogSourceName + ".log";
// log source name may have a colon - if soreplace with underscore
appenderPath = appenderPath.Replace( ':', '_' );
// now add to log4net path
appenderPath = Path.Combine( log4netPath, appenderPath );
// update file property of appender
this.rollingFileAppender.File = appenderPath;
// add the layout
PatternLayout patternLayout = new PatternLayout( log4netLayoutString );
this.rollingFileAppender.Layout = patternLayout;
// add the filter for the log source
NDCFilter sourceFilter = new NDCFilter();
sourceFilter.StringToMatch = this.LogSourceName;
this.rollingFileAppender.AddFilter( sourceFilter);
// now add the deny all filter to end of the chain
DenyAllFilter denyAllFilter = new DenyAllFilter();
this.rollingFileAppender.AddFilter( denyAllFilter );
// activate the options
this.rollingFileAppender.ActivateOptions();
// add the appender
connectionAppender.AddAppender( this.rollingFileAppender );
}
catch( Exception x )
{
this.ErrorLog.Error( "Error creating LIS3 data log appender for " + LogSourceName, x );
}
}
}
/// <summary>
/// remove connection specific appender
/// </summary>
void RemoveAppender()
{
// check if we have one
if( this.rollingFileAppender != null )
{
// cast to required interface
IAppenderAttachable connectionAppender = (IAppenderAttachable)this.DataLog.Logger;
// remove the appendier
connectionAppender.RemoveAppender( rollingFileAppender );
// set to null
this.rollingFileAppender = null;
}
}
Как насчет:
((log4net.Repository.Hierarchy.Logger) theLogger.Logger).RemoveAppender("SomeAppender");
то же самое для доп.