@ -3,6 +3,9 @@ using System.Collections.Generic;
using System.Linq ;
using System.Linq ;
using NLog ;
using NLog ;
using NzbDrone.Common.Cache ;
using NzbDrone.Common.Cache ;
using NzbDrone.Common.Extensions ;
using NzbDrone.Common.Messaging ;
using NzbDrone.Common.Reflection ;
using NzbDrone.Core.Configuration.Events ;
using NzbDrone.Core.Configuration.Events ;
using NzbDrone.Core.Download ;
using NzbDrone.Core.Download ;
using NzbDrone.Core.Indexers ;
using NzbDrone.Core.Indexers ;
@ -21,13 +24,12 @@ namespace NzbDrone.Core.HealthCheck
public class HealthCheckService : IHealthCheckService ,
public class HealthCheckService : IHealthCheckService ,
IExecute < CheckHealthCommand > ,
IExecute < CheckHealthCommand > ,
IHandleAsync < ApplicationStartedEvent > ,
IHandleAsync < ApplicationStartedEvent > ,
IHandleAsync < ConfigSavedEvent > ,
IHandleAsync < IEvent >
IHandleAsync < ProviderUpdatedEvent < IIndexer > > ,
IHandleAsync < ProviderDeletedEvent < IIndexer > > ,
IHandleAsync < ProviderUpdatedEvent < IDownloadClient > > ,
IHandleAsync < ProviderDeletedEvent < IDownloadClient > >
{
{
private readonly IEnumerable < IProvideHealthCheck > _healthChecks ;
private readonly IProvideHealthCheck [ ] _healthChecks ;
private readonly IProvideHealthCheck [ ] _startupHealthChecks ;
private readonly IProvideHealthCheck [ ] _scheduledHealthChecks ;
private readonly Dictionary < Type , IProvideHealthCheck [ ] > _eventDrivenHealthChecks ;
private readonly IEventAggregator _eventAggregator ;
private readonly IEventAggregator _eventAggregator ;
private readonly ICacheManager _cacheManager ;
private readonly ICacheManager _cacheManager ;
private readonly Logger _logger ;
private readonly Logger _logger ;
@ -39,12 +41,16 @@ namespace NzbDrone.Core.HealthCheck
ICacheManager cacheManager ,
ICacheManager cacheManager ,
Logger logger )
Logger logger )
{
{
_healthChecks = healthChecks ;
_healthChecks = healthChecks .ToArray ( ) ;
_eventAggregator = eventAggregator ;
_eventAggregator = eventAggregator ;
_cacheManager = cacheManager ;
_cacheManager = cacheManager ;
_logger = logger ;
_logger = logger ;
_healthCheckResults = _cacheManager . GetCache < HealthCheck > ( GetType ( ) ) ;
_healthCheckResults = _cacheManager . GetCache < HealthCheck > ( GetType ( ) ) ;
_startupHealthChecks = _healthChecks . Where ( v = > v . CheckOnStartup ) . ToArray ( ) ;
_scheduledHealthChecks = _healthChecks . Where ( v = > v . CheckOnSchedule ) . ToArray ( ) ;
_eventDrivenHealthChecks = GetEventDrivenHealthChecks ( ) ;
}
}
public List < HealthCheck > Results ( )
public List < HealthCheck > Results ( )
@ -52,10 +58,17 @@ namespace NzbDrone.Core.HealthCheck
return _healthCheckResults . Values . ToList ( ) ;
return _healthCheckResults . Values . ToList ( ) ;
}
}
private void PerformHealthCheck ( Func < IProvideHealthCheck , bool > predicate )
private Dictionary < Type , IProvideHealthCheck [ ] > GetEventDrivenHealthChecks ( )
{
return _healthChecks
. SelectMany ( h = > h . GetType ( ) . GetAttributes < CheckOnAttribute > ( ) . Select ( a = > Tuple . Create ( a . EventType , h ) ) )
. GroupBy ( t = > t . Item1 , t = > t . Item2 )
. ToDictionary ( g = > g . Key , g = > g . ToArray ( ) ) ;
}
private void PerformHealthCheck ( IProvideHealthCheck [ ] healthChecks )
{
{
var results = _healthChecks . Where ( predicate )
var results = healthChecks . Select ( c = > c . Check ( ) )
. Select ( c = > c . Check ( ) )
. ToList ( ) ;
. ToList ( ) ;
foreach ( var result in results )
foreach ( var result in results )
@ -76,37 +89,37 @@ namespace NzbDrone.Core.HealthCheck
public void Execute ( CheckHealthCommand message )
public void Execute ( CheckHealthCommand message )
{
{
PerformHealthCheck ( c = > message . Trigger = = CommandTrigger . Manual | | c . CheckOnSchedule ) ;
if ( message . Trigger = = CommandTrigger . Manual )
{
PerformHealthCheck ( _healthChecks ) ;
}
else
{
PerformHealthCheck ( _scheduledHealthChecks ) ;
}
}
}
public void HandleAsync ( ApplicationStartedEvent message )
public void HandleAsync ( ApplicationStartedEvent message )
{
{
PerformHealthCheck ( c = > c . CheckOnStartup ) ;
PerformHealthCheck ( _startupHealthChecks ) ;
}
public void HandleAsync ( ConfigSavedEvent message )
{
PerformHealthCheck ( c = > c . CheckOnConfigChange ) ;
}
}
public void HandleAsync ( ProviderUpdatedEvent< IIndexer > message )
public void HandleAsync ( IEvent message )
{
{
PerformHealthCheck ( c = > c . CheckOnConfigChange ) ;
if ( message is HealthCheckCompleteEvent )
}
{
return ;
}
public void HandleAsync ( ProviderDeletedEvent < IIndexer > message )
IProvideHealthCheck [ ] checks ;
{
if ( ! _eventDrivenHealthChecks . TryGetValue ( message . GetType ( ) , out checks ) )
PerformHealthCheck ( c = > c . CheckOnConfigChange ) ;
{
}
return ;
}
public void HandleAsync ( ProviderUpdatedEvent < IDownloadClient > message )
// TODO: Add debounce
{
PerformHealthCheck ( c = > c . CheckOnConfigChange ) ;
}
public void HandleAsync ( ProviderDeletedEvent < IDownloadClient > message )
PerformHealthCheck ( checks ) ;
{
PerformHealthCheck ( c = > c . CheckOnConfigChange ) ;
}
}
}
}
}
}