diff --git a/src/NzbDrone.Core/HealthCheck/HealthCheckFailedEvent.cs b/src/NzbDrone.Core/HealthCheck/HealthCheckFailedEvent.cs index c7bdb481b..b4917262c 100644 --- a/src/NzbDrone.Core/HealthCheck/HealthCheckFailedEvent.cs +++ b/src/NzbDrone.Core/HealthCheck/HealthCheckFailedEvent.cs @@ -5,10 +5,12 @@ namespace NzbDrone.Core.HealthCheck public class HealthCheckFailedEvent : IEvent { public HealthCheck HealthCheck { get; private set; } + public bool IsInStartupGraceperiod { get; private set; } - public HealthCheckFailedEvent(HealthCheck healthCheck) + public HealthCheckFailedEvent(HealthCheck healthCheck, bool isInStartupGraceperiod) { HealthCheck = healthCheck; + IsInStartupGraceperiod = isInStartupGraceperiod; } } } \ No newline at end of file diff --git a/src/NzbDrone.Core/HealthCheck/HealthCheckService.cs b/src/NzbDrone.Core/HealthCheck/HealthCheckService.cs index d367fd9df..3f8f02198 100644 --- a/src/NzbDrone.Core/HealthCheck/HealthCheckService.cs +++ b/src/NzbDrone.Core/HealthCheck/HealthCheckService.cs @@ -3,16 +3,12 @@ using System.Linq; using NLog; using NzbDrone.Common.Cache; -using NzbDrone.Common.Extensions; +using NzbDrone.Common.EnvironmentInfo; using NzbDrone.Common.Messaging; using NzbDrone.Common.Reflection; -using NzbDrone.Core.Configuration.Events; -using NzbDrone.Core.Download; -using NzbDrone.Core.Indexers; using NzbDrone.Core.Lifecycle; using NzbDrone.Core.Messaging.Commands; using NzbDrone.Core.Messaging.Events; -using NzbDrone.Core.ThingiProvider.Events; namespace NzbDrone.Core.HealthCheck { @@ -26,6 +22,7 @@ public class HealthCheckService : IHealthCheckService, IHandleAsync, IHandleAsync { + private readonly DateTime _startupGracePeriodEndTime; private readonly IProvideHealthCheck[] _healthChecks; private readonly IProvideHealthCheck[] _startupHealthChecks; private readonly IProvideHealthCheck[] _scheduledHealthChecks; @@ -36,9 +33,13 @@ public class HealthCheckService : IHealthCheckService, private readonly ICached _healthCheckResults; + private bool _hasRunHealthChecksAfterGracePeriod = false; + private bool _isRunningHealthChecksAfterGracePeriod = false; + public HealthCheckService(IEnumerable healthChecks, IEventAggregator eventAggregator, ICacheManager cacheManager, + IRuntimeInfo runtimeInfo, Logger logger) { _healthChecks = healthChecks.ToArray(); @@ -51,6 +52,7 @@ public HealthCheckService(IEnumerable healthChecks, _startupHealthChecks = _healthChecks.Where(v => v.CheckOnStartup).ToArray(); _scheduledHealthChecks = _healthChecks.Where(v => v.CheckOnSchedule).ToArray(); _eventDrivenHealthChecks = GetEventDrivenHealthChecks(); + _startupGracePeriodEndTime = runtimeInfo.StartTime + TimeSpan.FromMinutes(15); } public List Results() @@ -88,7 +90,7 @@ private void PerformHealthCheck(IProvideHealthCheck[] healthChecks) { if (_healthCheckResults.Find(result.Source.Name) == null) { - _eventAggregator.PublishEvent(new HealthCheckFailedEvent(result)); + _eventAggregator.PublishEvent(new HealthCheckFailedEvent(result, !_hasRunHealthChecksAfterGracePeriod)); } _healthCheckResults.Set(result.Source.Name, result); @@ -122,6 +124,31 @@ public void HandleAsync(IEvent message) return; } + // If we haven't previously re-run health checks after startup grace period run startup checks again and track so they aren't run again. + // Return early after re-running checks to avoid triggering checks multiple times. + + if (!_hasRunHealthChecksAfterGracePeriod && !_isRunningHealthChecksAfterGracePeriod && DateTime.UtcNow > _startupGracePeriodEndTime) + { + _isRunningHealthChecksAfterGracePeriod = true; + + PerformHealthCheck(_startupHealthChecks); + + // Update after running health checks so new failure notifications aren't sent 2x. + _hasRunHealthChecksAfterGracePeriod = true; + + // Explicitly notify for any failed checks since existing failed results would not have sent events. + var results = _healthCheckResults.Values.ToList(); + + foreach (var result in results) + { + _eventAggregator.PublishEvent(new HealthCheckFailedEvent(result, false)); + } + + _isRunningHealthChecksAfterGracePeriod = false; + + return; + } + IEventDrivenHealthCheck[] checks; if (!_eventDrivenHealthChecks.TryGetValue(message.GetType(), out checks)) { @@ -143,7 +170,6 @@ public void HandleAsync(IEvent message) } } - // TODO: Add debounce PerformHealthCheck(filteredChecks.ToArray()); diff --git a/src/NzbDrone.Core/Notifications/NotificationService.cs b/src/NzbDrone.Core/Notifications/NotificationService.cs index 18996a1a4..1934e6743 100644 --- a/src/NzbDrone.Core/Notifications/NotificationService.cs +++ b/src/NzbDrone.Core/Notifications/NotificationService.cs @@ -272,6 +272,14 @@ public void Handle(SeriesDeletedEvent message) public void Handle(HealthCheckFailedEvent message) { + // Don't send health check notifications during the start up grace period, + // once that duration expires they they'll be retested and fired off if necessary. + + if (message.IsInStartupGraceperiod) + { + return; + } + foreach (var notification in _notificationFactory.OnHealthIssueEnabled()) { try