From cb511dc19d3c0693437d0d89d850e9855a5258ff Mon Sep 17 00:00:00 2001 From: Mark McDowall Date: Sat, 20 May 2023 20:51:24 -0700 Subject: [PATCH] Fixed: Don't retry grabbing the same release if download client is unavailable (cherry picked from commit b38c1255dc19d72ee10db4af67e76a4ce95f288f) Closes #2520 --- .../Download/DownloadServiceFixture.cs | 2 +- .../Download/DownloadClientProvider.cs | 18 +++++++++++++----- src/NzbDrone.Core/Download/DownloadService.cs | 4 +++- .../Download/Pending/PendingReleaseService.cs | 9 ++++++++- src/NzbDrone.Core/Parser/Model/ReleaseInfo.cs | 6 ++++++ 5 files changed, 31 insertions(+), 8 deletions(-) diff --git a/src/NzbDrone.Core.Test/Download/DownloadServiceFixture.cs b/src/NzbDrone.Core.Test/Download/DownloadServiceFixture.cs index 46d19add8..eb33ac6a7 100644 --- a/src/NzbDrone.Core.Test/Download/DownloadServiceFixture.cs +++ b/src/NzbDrone.Core.Test/Download/DownloadServiceFixture.cs @@ -31,7 +31,7 @@ namespace NzbDrone.Core.Test.Download .Returns(_downloadClients); Mocker.GetMock() - .Setup(v => v.GetDownloadClient(It.IsAny(), It.IsAny())) + .Setup(v => v.GetDownloadClient(It.IsAny(), It.IsAny(), It.IsAny())) .Returns((v, i) => _downloadClients.FirstOrDefault(d => d.Protocol == v)); var episodes = Builder.CreateListOfSize(2) diff --git a/src/NzbDrone.Core/Download/DownloadClientProvider.cs b/src/NzbDrone.Core/Download/DownloadClientProvider.cs index df9249dcc..370b66dea 100644 --- a/src/NzbDrone.Core/Download/DownloadClientProvider.cs +++ b/src/NzbDrone.Core/Download/DownloadClientProvider.cs @@ -9,7 +9,7 @@ namespace NzbDrone.Core.Download { public interface IProvideDownloadClient { - IDownloadClient GetDownloadClient(DownloadProtocol downloadProtocol, int indexerId = 0); + IDownloadClient GetDownloadClient(DownloadProtocol downloadProtocol, int indexerId = 0, bool filterBlockedClients = false); IEnumerable GetDownloadClients(bool filterBlockedClients = false); IDownloadClient Get(int id); } @@ -35,8 +35,9 @@ namespace NzbDrone.Core.Download _lastUsedDownloadClient = cacheManager.GetCache(GetType(), "lastDownloadClientId"); } - public IDownloadClient GetDownloadClient(DownloadProtocol downloadProtocol, int indexerId = 0) + public IDownloadClient GetDownloadClient(DownloadProtocol downloadProtocol, int indexerId = 0, bool filterBlockedClients = false) { + var blockedProviders = new HashSet(_downloadClientStatusService.GetBlockedProviders().Select(v => v.ProviderId)); var availableProviders = _downloadClientFactory.GetAvailableProviders().Where(v => v.Protocol == downloadProtocol).ToList(); if (!availableProviders.Any()) @@ -52,12 +53,15 @@ namespace NzbDrone.Core.Download { var client = availableProviders.SingleOrDefault(d => d.Definition.Id == indexer.DownloadClientId); - return client ?? throw new DownloadClientUnavailableException($"Indexer specified download client is not available"); + if (client == null || (filterBlockedClients && blockedProviders.Contains(client.Definition.Id))) + { + throw new DownloadClientUnavailableException($"Indexer specified download client is not available"); + } + + return client; } } - var blockedProviders = new HashSet(_downloadClientStatusService.GetBlockedProviders().Select(v => v.ProviderId)); - if (blockedProviders.Any()) { var nonBlockedProviders = availableProviders.Where(v => !blockedProviders.Contains(v.Definition.Id)).ToList(); @@ -66,6 +70,10 @@ namespace NzbDrone.Core.Download { availableProviders = nonBlockedProviders; } + else if (filterBlockedClients) + { + throw new DownloadClientUnavailableException($"All download clients for {downloadProtocol} are not available"); + } else { _logger.Trace("No non-blocked Download Client available, retrying blocked one."); diff --git a/src/NzbDrone.Core/Download/DownloadService.cs b/src/NzbDrone.Core/Download/DownloadService.cs index 6687ac20e..7714c26cf 100644 --- a/src/NzbDrone.Core/Download/DownloadService.cs +++ b/src/NzbDrone.Core/Download/DownloadService.cs @@ -6,6 +6,7 @@ using NzbDrone.Common.Http; using NzbDrone.Common.Instrumentation.Extensions; using NzbDrone.Common.TPL; using NzbDrone.Core.Download.Clients; +using NzbDrone.Core.Download.Pending; using NzbDrone.Core.Exceptions; using NzbDrone.Core.Indexers; using NzbDrone.Core.Messaging.Events; @@ -54,7 +55,8 @@ namespace NzbDrone.Core.Download Ensure.That(remoteBook.Books, () => remoteBook.Books).HasItems(); var downloadTitle = remoteBook.Release.Title; - var downloadClient = _downloadClientProvider.GetDownloadClient(remoteBook.Release.DownloadProtocol, remoteBook.Release.IndexerId); + var filterBlockedClients = remoteBook.Release.PendingReleaseReason == PendingReleaseReason.DownloadClientUnavailable; + var downloadClient = _downloadClientProvider.GetDownloadClient(remoteBook.Release.DownloadProtocol, remoteBook.Release.IndexerId, filterBlockedClients); if (downloadClient == null) { diff --git a/src/NzbDrone.Core/Download/Pending/PendingReleaseService.cs b/src/NzbDrone.Core/Download/Pending/PendingReleaseService.cs index 65d0fa452..2dcdaf3a2 100644 --- a/src/NzbDrone.Core/Download/Pending/PendingReleaseService.cs +++ b/src/NzbDrone.Core/Download/Pending/PendingReleaseService.cs @@ -139,7 +139,14 @@ namespace NzbDrone.Core.Download.Pending public List GetPending() { - var releases = _repository.All().Select(p => p.Release).ToList(); + var releases = _repository.All().Select(p => + { + var release = p.Release; + + release.PendingReleaseReason = p.Reason; + + return release; + }).ToList(); if (releases.Any()) { diff --git a/src/NzbDrone.Core/Parser/Model/ReleaseInfo.cs b/src/NzbDrone.Core/Parser/Model/ReleaseInfo.cs index 21f86f0a9..eafa3a932 100644 --- a/src/NzbDrone.Core/Parser/Model/ReleaseInfo.cs +++ b/src/NzbDrone.Core/Parser/Model/ReleaseInfo.cs @@ -1,6 +1,8 @@ using System; using System.Collections.Generic; using System.Text; +using Newtonsoft.Json; +using NzbDrone.Core.Download.Pending; using NzbDrone.Core.Indexers; namespace NzbDrone.Core.Parser.Model @@ -27,6 +29,10 @@ namespace NzbDrone.Core.Parser.Model public string Codec { get; set; } public List Categories { get; set; } + // Used to track pending releases that are being reprocessed + [JsonIgnore] + public PendingReleaseReason? PendingReleaseReason { get; set; } + public int Age { get { return DateTime.UtcNow.Subtract(PublishDate).Days; }