Merge pull request #116 from NzbDrone/cdh-improvements

CDH Improvements
This commit is contained in:
Mark McDowall 2014-09-28 16:20:57 -07:00
commit 73963011ee
26 changed files with 292 additions and 149 deletions

View File

@ -1,5 +1,7 @@
using System; using System;
using System.Collections.Generic;
using NzbDrone.Api.REST; using NzbDrone.Api.REST;
using NzbDrone.Core.Download;
using NzbDrone.Core.Qualities; using NzbDrone.Core.Qualities;
using NzbDrone.Api.Series; using NzbDrone.Api.Series;
using NzbDrone.Api.Episodes; using NzbDrone.Api.Episodes;
@ -17,6 +19,7 @@ namespace NzbDrone.Api.Queue
public TimeSpan? Timeleft { get; set; } public TimeSpan? Timeleft { get; set; }
public DateTime? EstimatedCompletionTime { get; set; } public DateTime? EstimatedCompletionTime { get; set; }
public String Status { get; set; } public String Status { get; set; }
public String ErrorMessage { get; set; } public String TrackedDownloadStatus { get; set; }
public List<TrackedDownloadStatusMessage> StatusMessages { get; set; }
} }
} }

View File

@ -69,10 +69,7 @@ namespace NzbDrone.Core.Test.DecisionEngineTests
queue.Add(new TrackedDownload queue.Add(new TrackedDownload
{ {
State = state, State = state,
DownloadItem = new DownloadClientItem
{
RemoteEpisode = remoteEpisode RemoteEpisode = remoteEpisode
}
}); });
} }

View File

@ -9,11 +9,13 @@ using NzbDrone.Common.Disk;
using NzbDrone.Core.Configuration; using NzbDrone.Core.Configuration;
using NzbDrone.Core.Download; using NzbDrone.Core.Download;
using NzbDrone.Core.History; using NzbDrone.Core.History;
using NzbDrone.Core.IndexerSearch.Definitions;
using NzbDrone.Core.Parser;
using NzbDrone.Core.Test.Framework; using NzbDrone.Core.Test.Framework;
using NzbDrone.Core.MediaFiles; using NzbDrone.Core.MediaFiles;
using NzbDrone.Core.MediaFiles.EpisodeImport; using NzbDrone.Core.MediaFiles.EpisodeImport;
using NzbDrone.Test.Common;
using NzbDrone.Core.Tv; using NzbDrone.Core.Tv;
using NzbDrone.Test.Common;
using NzbDrone.Core.Parser.Model; using NzbDrone.Core.Parser.Model;
namespace NzbDrone.Core.Test.Download namespace NzbDrone.Core.Test.Download
@ -30,20 +32,23 @@ namespace NzbDrone.Core.Test.Download
.All() .All()
.With(h => h.Status = DownloadItemStatus.Completed) .With(h => h.Status = DownloadItemStatus.Completed)
.With(h => h.OutputPath = @"C:\DropFolder\MyDownload".AsOsAgnostic()) .With(h => h.OutputPath = @"C:\DropFolder\MyDownload".AsOsAgnostic())
.With(h => h.RemoteEpisode = new RemoteEpisode .With(h => h.Title = "Drone.S01E01.HDTV")
{
Episodes = new List<Episode> { new Episode { Id = 1 } }
})
.Build() .Build()
.ToList(); .ToList();
var remoteEpisode = new RemoteEpisode
{
Series = new Series(),
Episodes = new List<Episode> {new Episode {Id = 1}}
};
Mocker.GetMock<IProvideDownloadClient>() Mocker.GetMock<IProvideDownloadClient>()
.Setup(c => c.GetDownloadClients()) .Setup(c => c.GetDownloadClients())
.Returns( new IDownloadClient[] { Mocker.GetMock<IDownloadClient>().Object }); .Returns( new IDownloadClient[] { Mocker.GetMock<IDownloadClient>().Object });
Mocker.GetMock<IDownloadClient>() Mocker.GetMock<IDownloadClient>()
.SetupGet(c => c.Definition) .SetupGet(c => c.Definition)
.Returns(new Core.Download.DownloadClientDefinition { Id = 1, Name = "testClient" }); .Returns(new DownloadClientDefinition { Id = 1, Name = "testClient" });
Mocker.GetMock<IConfigService>() Mocker.GetMock<IConfigService>()
.SetupGet(s => s.EnableCompletedDownloadHandling) .SetupGet(s => s.EnableCompletedDownloadHandling)
@ -57,6 +62,14 @@ namespace NzbDrone.Core.Test.Download
.Setup(s => s.Failed()) .Setup(s => s.Failed())
.Returns(new List<History.History>()); .Returns(new List<History.History>());
Mocker.GetMock<IParsingService>()
.Setup(s => s.Map(It.IsAny<ParsedEpisodeInfo>(), It.IsAny<Int32>(), It.IsAny<IEnumerable<Int32>>()))
.Returns(remoteEpisode);
Mocker.GetMock<IParsingService>()
.Setup(s => s.Map(It.IsAny<ParsedEpisodeInfo>(), It.IsAny<Int32>(), (SearchCriteriaBase)null))
.Returns(remoteEpisode);
Mocker.SetConstant<ICompletedDownloadService>(Mocker.Resolve<CompletedDownloadService>()); Mocker.SetConstant<ICompletedDownloadService>(Mocker.Resolve<CompletedDownloadService>());
} }
@ -311,10 +324,7 @@ namespace NzbDrone.Core.Test.Download
.All() .All()
.With(h => h.Status = DownloadItemStatus.Completed) .With(h => h.Status = DownloadItemStatus.Completed)
.With(h => h.OutputPath = @"C:\DropFolder\MyDownload".AsOsAgnostic()) .With(h => h.OutputPath = @"C:\DropFolder\MyDownload".AsOsAgnostic())
.With(h => h.RemoteEpisode = new RemoteEpisode .With(h => h.Title = "Drone.S01E01.HDTV")
{
Episodes = new List<Episode> { new Episode { Id = 1 } }
})
.Build()); .Build());
var grabbedHistory = Builder<History.History>.CreateListOfSize(2) var grabbedHistory = Builder<History.History>.CreateListOfSize(2)

View File

@ -5,6 +5,7 @@ using Moq;
using NUnit.Framework; using NUnit.Framework;
using FluentAssertions; using FluentAssertions;
using NzbDrone.Common.Http; using NzbDrone.Common.Http;
using NzbDrone.Core.IndexerSearch.Definitions;
using NzbDrone.Core.Test.Framework; using NzbDrone.Core.Test.Framework;
using NzbDrone.Core.Parser.Model; using NzbDrone.Core.Parser.Model;
using NzbDrone.Core.Parser; using NzbDrone.Core.Parser;
@ -30,7 +31,7 @@ namespace NzbDrone.Core.Test.Download.DownloadClientTests
.Returns(30); .Returns(30);
Mocker.GetMock<IParsingService>() Mocker.GetMock<IParsingService>()
.Setup(s => s.Map(It.IsAny<ParsedEpisodeInfo>(), It.IsAny<int>(), null)) .Setup(s => s.Map(It.IsAny<ParsedEpisodeInfo>(), It.IsAny<int>(), (SearchCriteriaBase)null))
.Returns(() => CreateRemoteEpisode()); .Returns(() => CreateRemoteEpisode());
Mocker.GetMock<IHttpClient>() Mocker.GetMock<IHttpClient>()
@ -64,11 +65,7 @@ namespace NzbDrone.Core.Test.Download.DownloadClientTests
{ {
downloadClientItem.DownloadClient.Should().Be(Subject.Definition.Name); downloadClientItem.DownloadClient.Should().Be(Subject.Definition.Name);
downloadClientItem.DownloadClientId.Should().NotBeNullOrEmpty(); downloadClientItem.DownloadClientId.Should().NotBeNullOrEmpty();
downloadClientItem.Title.Should().NotBeNullOrEmpty(); downloadClientItem.Title.Should().NotBeNullOrEmpty();
downloadClientItem.RemoteEpisode.Should().NotBeNull();
} }
protected void VerifyQueued(DownloadClientItem downloadClientItem) protected void VerifyQueued(DownloadClientItem downloadClientItem)

View File

@ -7,8 +7,12 @@ using NUnit.Framework;
using NzbDrone.Core.Configuration; using NzbDrone.Core.Configuration;
using NzbDrone.Core.Download; using NzbDrone.Core.Download;
using NzbDrone.Core.History; using NzbDrone.Core.History;
using NzbDrone.Core.IndexerSearch.Definitions;
using NzbDrone.Core.Messaging.Events; using NzbDrone.Core.Messaging.Events;
using NzbDrone.Core.Parser;
using NzbDrone.Core.Parser.Model;
using NzbDrone.Core.Test.Framework; using NzbDrone.Core.Test.Framework;
using NzbDrone.Core.Tv;
using NzbDrone.Test.Common; using NzbDrone.Test.Common;
namespace NzbDrone.Core.Test.Download namespace NzbDrone.Core.Test.Download
@ -25,22 +29,30 @@ namespace NzbDrone.Core.Test.Download
_completed = Builder<DownloadClientItem>.CreateListOfSize(5) _completed = Builder<DownloadClientItem>.CreateListOfSize(5)
.All() .All()
.With(h => h.Status = DownloadItemStatus.Completed) .With(h => h.Status = DownloadItemStatus.Completed)
.With(h => h.Title = "Drone.S01E01.HDTV")
.Build() .Build()
.ToList(); .ToList();
_failed = Builder<DownloadClientItem>.CreateListOfSize(1) _failed = Builder<DownloadClientItem>.CreateListOfSize(1)
.All() .All()
.With(h => h.Status = DownloadItemStatus.Failed) .With(h => h.Status = DownloadItemStatus.Failed)
.With(h => h.Title = "Drone.S01E01.HDTV")
.Build() .Build()
.ToList(); .ToList();
var remoteEpisode = new RemoteEpisode
{
Series = new Series(),
Episodes = new List<Episode> { new Episode { Id = 1 } }
};
Mocker.GetMock<IProvideDownloadClient>() Mocker.GetMock<IProvideDownloadClient>()
.Setup(c => c.GetDownloadClients()) .Setup(c => c.GetDownloadClients())
.Returns( new IDownloadClient[] { Mocker.GetMock<IDownloadClient>().Object }); .Returns( new IDownloadClient[] { Mocker.GetMock<IDownloadClient>().Object });
Mocker.GetMock<IDownloadClient>() Mocker.GetMock<IDownloadClient>()
.SetupGet(c => c.Definition) .SetupGet(c => c.Definition)
.Returns(new Core.Download.DownloadClientDefinition { Id = 1, Name = "testClient" }); .Returns(new DownloadClientDefinition { Id = 1, Name = "testClient" });
Mocker.GetMock<IConfigService>() Mocker.GetMock<IConfigService>()
.SetupGet(s => s.EnableFailedDownloadHandling) .SetupGet(s => s.EnableFailedDownloadHandling)
@ -50,6 +62,14 @@ namespace NzbDrone.Core.Test.Download
.Setup(s => s.Imported()) .Setup(s => s.Imported())
.Returns(new List<History.History>()); .Returns(new List<History.History>());
Mocker.GetMock<IParsingService>()
.Setup(s => s.Map(It.IsAny<ParsedEpisodeInfo>(), It.IsAny<Int32>(), It.IsAny<IEnumerable<Int32>>()))
.Returns(remoteEpisode);
Mocker.GetMock<IParsingService>()
.Setup(s => s.Map(It.IsAny<ParsedEpisodeInfo>(), It.IsAny<Int32>(), (SearchCriteriaBase)null))
.Returns(remoteEpisode);
Mocker.SetConstant<IFailedDownloadService>(Mocker.Resolve<FailedDownloadService>()); Mocker.SetConstant<IFailedDownloadService>(Mocker.Resolve<FailedDownloadService>());
} }

View File

@ -34,7 +34,7 @@ namespace NzbDrone.Core.DecisionEngine.Specifications
{ {
var queue = _downloadTrackingService.GetQueuedDownloads() var queue = _downloadTrackingService.GetQueuedDownloads()
.Where(v => v.State == TrackedDownloadState.Downloading) .Where(v => v.State == TrackedDownloadState.Downloading)
.Select(q => q.DownloadItem.RemoteEpisode).ToList(); .Select(q => q.RemoteEpisode).ToList();
if (IsInQueue(subject, queue)) if (IsInQueue(subject, queue))
{ {

View File

@ -201,9 +201,6 @@ namespace NzbDrone.Core.Download.Clients.Nzbget
{ {
if (downloadClientItem.Category == Settings.TvCategory) if (downloadClientItem.Category == Settings.TvCategory)
{ {
downloadClientItem.RemoteEpisode = GetRemoteEpisode(downloadClientItem.Title);
if (downloadClientItem.RemoteEpisode == null) continue;
yield return downloadClientItem; yield return downloadClientItem;
} }
} }

View File

@ -103,9 +103,6 @@ namespace NzbDrone.Core.Download.Clients.Pneumatic
historyItem.Status = DownloadItemStatus.Completed; historyItem.Status = DownloadItemStatus.Completed;
} }
historyItem.RemoteEpisode = GetRemoteEpisode(historyItem.Title);
if (historyItem.RemoteEpisode == null) continue;
yield return historyItem; yield return historyItem;
} }
} }

View File

@ -188,9 +188,6 @@ namespace NzbDrone.Core.Download.Clients.Sabnzbd
{ {
if (downloadClientItem.Category == Settings.TvCategory) if (downloadClientItem.Category == Settings.TvCategory)
{ {
downloadClientItem.RemoteEpisode = GetRemoteEpisode(downloadClientItem.Title);
if (downloadClientItem.RemoteEpisode == null) continue;
yield return downloadClientItem; yield return downloadClientItem;
} }
} }

View File

@ -90,9 +90,6 @@ namespace NzbDrone.Core.Download.Clients.UsenetBlackhole
historyItem.RemainingTime = TimeSpan.Zero; historyItem.RemainingTime = TimeSpan.Zero;
} }
historyItem.RemoteEpisode = GetRemoteEpisode(historyItem.Title);
if (historyItem.RemoteEpisode == null) continue;
yield return historyItem; yield return historyItem;
} }
@ -121,9 +118,6 @@ namespace NzbDrone.Core.Download.Clients.UsenetBlackhole
historyItem.RemainingTime = TimeSpan.Zero; historyItem.RemainingTime = TimeSpan.Zero;
} }
historyItem.RemoteEpisode = GetRemoteEpisode(historyItem.Title);
if (historyItem.RemoteEpisode == null) continue;
yield return historyItem; yield return historyItem;
} }
} }

View File

@ -8,7 +8,6 @@ using NzbDrone.Core.Configuration;
using NzbDrone.Core.History; using NzbDrone.Core.History;
using NzbDrone.Core.MediaFiles; using NzbDrone.Core.MediaFiles;
using NzbDrone.Core.MediaFiles.EpisodeImport; using NzbDrone.Core.MediaFiles.EpisodeImport;
using NzbDrone.Core.Messaging.Events;
using System.IO; using System.IO;
namespace NzbDrone.Core.Download namespace NzbDrone.Core.Download
@ -20,21 +19,18 @@ namespace NzbDrone.Core.Download
public class CompletedDownloadService : ICompletedDownloadService public class CompletedDownloadService : ICompletedDownloadService
{ {
private readonly IEventAggregator _eventAggregator;
private readonly IConfigService _configService; private readonly IConfigService _configService;
private readonly IDiskProvider _diskProvider; private readonly IDiskProvider _diskProvider;
private readonly IDownloadedEpisodesImportService _downloadedEpisodesImportService; private readonly IDownloadedEpisodesImportService _downloadedEpisodesImportService;
private readonly IHistoryService _historyService; private readonly IHistoryService _historyService;
private readonly Logger _logger; private readonly Logger _logger;
public CompletedDownloadService(IEventAggregator eventAggregator, public CompletedDownloadService(IConfigService configService,
IConfigService configService,
IDiskProvider diskProvider, IDiskProvider diskProvider,
IDownloadedEpisodesImportService downloadedEpisodesImportService, IDownloadedEpisodesImportService downloadedEpisodesImportService,
IHistoryService historyService, IHistoryService historyService,
Logger logger) Logger logger)
{ {
_eventAggregator = eventAggregator;
_configService = configService; _configService = configService;
_diskProvider = diskProvider; _diskProvider = diskProvider;
_downloadedEpisodesImportService = downloadedEpisodesImportService; _downloadedEpisodesImportService = downloadedEpisodesImportService;
@ -61,7 +57,7 @@ namespace NzbDrone.Core.Download
if (!grabbedItems.Any() && trackedDownload.DownloadItem.Category.IsNullOrWhiteSpace()) if (!grabbedItems.Any() && trackedDownload.DownloadItem.Category.IsNullOrWhiteSpace())
{ {
UpdateStatusMessage(trackedDownload, LogLevel.Debug, "Download wasn't grabbed by drone or not in a category, ignoring download."); UpdateStatusMessage(trackedDownload, LogLevel.Warn, "Download wasn't grabbed by drone or not in a category, ignoring download.");
return; return;
} }
@ -73,10 +69,16 @@ namespace NzbDrone.Core.Download
UpdateStatusMessage(trackedDownload, LogLevel.Debug, "Already added to history as imported."); UpdateStatusMessage(trackedDownload, LogLevel.Debug, "Already added to history as imported.");
} }
else if (trackedDownload.Status != TrackedDownloadStatus.Ok)
{
_logger.Debug("Tracked download status is: {0}, skipping import.", trackedDownload.Status);
return;
}
else else
{ {
string downloadedEpisodesFolder = _configService.DownloadedEpisodesFolder; var downloadedEpisodesFolder = _configService.DownloadedEpisodesFolder;
string downloadItemOutputPath = trackedDownload.DownloadItem.OutputPath; var downloadItemOutputPath = trackedDownload.DownloadItem.OutputPath;
if (downloadItemOutputPath.IsNullOrWhiteSpace()) if (downloadItemOutputPath.IsNullOrWhiteSpace())
{ {
UpdateStatusMessage(trackedDownload, LogLevel.Warn, "Download doesn't contain intermediate path, ignoring download."); UpdateStatusMessage(trackedDownload, LogLevel.Warn, "Download doesn't contain intermediate path, ignoring download.");
@ -105,7 +107,7 @@ namespace NzbDrone.Core.Download
{ {
if (grabbedItems.Any()) if (grabbedItems.Any())
{ {
var episodeIds = trackedDownload.DownloadItem.RemoteEpisode.Episodes.Select(v => v.Id).ToList(); var episodeIds = trackedDownload.RemoteEpisode.Episodes.Select(v => v.Id).ToList();
// Check if we can associate it with a previous drone factory import. // Check if we can associate it with a previous drone factory import.
importedItems = importedHistory.Where(v => v.Data.GetValueOrDefault(DownloadTrackingService.DOWNLOAD_CLIENT_ID) == null && importedItems = importedHistory.Where(v => v.Data.GetValueOrDefault(DownloadTrackingService.DOWNLOAD_CLIENT_ID) == null &&
@ -171,7 +173,7 @@ namespace NzbDrone.Core.Download
if (trackedDownload.StatusMessage != statusMessage) if (trackedDownload.StatusMessage != statusMessage)
{ {
trackedDownload.HasError = logLevel >= LogLevel.Warn; trackedDownload.SetStatusLevel(logLevel);
trackedDownload.StatusMessage = statusMessage; trackedDownload.StatusMessage = statusMessage;
_logger.Log(logLevel, logMessage); _logger.Log(logLevel, logMessage);
} }
@ -200,6 +202,9 @@ namespace NzbDrone.Core.Download
.Select(v => v.Errors.Aggregate(Path.GetFileName(v.ImportDecision.LocalEpisode.Path), (a, r) => a + "\r\n- " + r)) .Select(v => v.Errors.Aggregate(Path.GetFileName(v.ImportDecision.LocalEpisode.Path), (a, r) => a + "\r\n- " + r))
.Aggregate("Failed to import:", (a, r) => a + "\r\n" + r); .Aggregate("Failed to import:", (a, r) => a + "\r\n" + r);
trackedDownload.StatusMessages = importResults.Where(v => v.Result == ImportResultType.Skipped || v.Result == ImportResultType.Rejected)
.Select(v => new TrackedDownloadStatusMessage(Path.GetFileName(v.ImportDecision.LocalEpisode.Path), v.Errors)).ToList();
UpdateStatusMessage(trackedDownload, LogLevel.Error, errors); UpdateStatusMessage(trackedDownload, LogLevel.Error, errors);
} }
} }

View File

@ -1,6 +1,5 @@
using System; using System;
using System.IO; using System.IO;
using System.Linq;
using System.Collections.Generic; using System.Collections.Generic;
using NzbDrone.Common; using NzbDrone.Common;
using NzbDrone.Common.Disk; using NzbDrone.Common.Disk;
@ -21,7 +20,6 @@ namespace NzbDrone.Core.Download
{ {
protected readonly IConfigService _configService; protected readonly IConfigService _configService;
protected readonly IDiskProvider _diskProvider; protected readonly IDiskProvider _diskProvider;
protected readonly IParsingService _parsingService;
protected readonly IRemotePathMappingService _remotePathMappingService; protected readonly IRemotePathMappingService _remotePathMappingService;
protected readonly Logger _logger; protected readonly Logger _logger;
@ -55,7 +53,6 @@ namespace NzbDrone.Core.Download
{ {
_configService = configService; _configService = configService;
_diskProvider = diskProvider; _diskProvider = diskProvider;
_parsingService = parsingService;
_remotePathMappingService = remotePathMappingService; _remotePathMappingService = remotePathMappingService;
_logger = logger; _logger = logger;
} }
@ -76,17 +73,6 @@ namespace NzbDrone.Core.Download
public abstract String RetryDownload(string id); public abstract String RetryDownload(string id);
public abstract DownloadClientStatus GetStatus(); public abstract DownloadClientStatus GetStatus();
protected RemoteEpisode GetRemoteEpisode(String title)
{
var parsedEpisodeInfo = Parser.Parser.ParseTitle(title);
if (parsedEpisodeInfo == null) return null;
var remoteEpisode = _parsingService.Map(parsedEpisodeInfo, 0);
if (remoteEpisode.Series == null) return null;
return remoteEpisode;
}
public ValidationResult Test() public ValidationResult Test()
{ {
var failures = new List<ValidationFailure>(); var failures = new List<ValidationFailure>();

View File

@ -1,5 +1,4 @@
using NzbDrone.Core.Parser.Model; using System;
using System;
namespace NzbDrone.Core.Download namespace NzbDrone.Core.Download
{ {
@ -21,7 +20,5 @@ namespace NzbDrone.Core.Download
public DownloadItemStatus Status { get; set; } public DownloadItemStatus Status { get; set; }
public Boolean IsEncrypted { get; set; } public Boolean IsEncrypted { get; set; }
public Boolean IsReadOnly { get; set; } public Boolean IsReadOnly { get; set; }
public RemoteEpisode RemoteEpisode { get; set; }
} }
} }

View File

@ -1,9 +1,4 @@
using System; namespace NzbDrone.Core.Download
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace NzbDrone.Core.Download
{ {
public enum DownloadItemStatus public enum DownloadItemStatus
{ {

View File

@ -9,6 +9,7 @@ using NzbDrone.Core.History;
using NzbDrone.Core.Messaging.Commands; using NzbDrone.Core.Messaging.Commands;
using NzbDrone.Core.Messaging.Events; using NzbDrone.Core.Messaging.Events;
using NzbDrone.Core.Lifecycle; using NzbDrone.Core.Lifecycle;
using NzbDrone.Core.Parser;
using NzbDrone.Core.Queue; using NzbDrone.Core.Queue;
namespace NzbDrone.Core.Download namespace NzbDrone.Core.Download
@ -22,7 +23,10 @@ namespace NzbDrone.Core.Download
void MarkAsFailed(Int32 historyId); void MarkAsFailed(Int32 historyId);
} }
public class DownloadTrackingService : IDownloadTrackingService, IExecute<CheckForFinishedDownloadCommand>, IHandleAsync<ApplicationStartedEvent>, IHandle<EpisodeGrabbedEvent> public class DownloadTrackingService : IDownloadTrackingService,
IExecute<CheckForFinishedDownloadCommand>,
IHandleAsync<ApplicationStartedEvent>,
IHandle<EpisodeGrabbedEvent>
{ {
private readonly IProvideDownloadClient _downloadClientProvider; private readonly IProvideDownloadClient _downloadClientProvider;
private readonly IHistoryService _historyService; private readonly IHistoryService _historyService;
@ -30,6 +34,7 @@ namespace NzbDrone.Core.Download
private readonly IConfigService _configService; private readonly IConfigService _configService;
private readonly IFailedDownloadService _failedDownloadService; private readonly IFailedDownloadService _failedDownloadService;
private readonly ICompletedDownloadService _completedDownloadService; private readonly ICompletedDownloadService _completedDownloadService;
private readonly IParsingService _parsingService;
private readonly Logger _logger; private readonly Logger _logger;
private readonly ICached<TrackedDownload[]> _trackedDownloadCache; private readonly ICached<TrackedDownload[]> _trackedDownloadCache;
@ -44,6 +49,7 @@ namespace NzbDrone.Core.Download
ICacheManager cacheManager, ICacheManager cacheManager,
IFailedDownloadService failedDownloadService, IFailedDownloadService failedDownloadService,
ICompletedDownloadService completedDownloadService, ICompletedDownloadService completedDownloadService,
IParsingService parsingService,
Logger logger) Logger logger)
{ {
_downloadClientProvider = downloadClientProvider; _downloadClientProvider = downloadClientProvider;
@ -52,6 +58,7 @@ namespace NzbDrone.Core.Download
_configService = configService; _configService = configService;
_failedDownloadService = failedDownloadService; _failedDownloadService = failedDownloadService;
_completedDownloadService = completedDownloadService; _completedDownloadService = completedDownloadService;
_parsingService = parsingService;
_logger = logger; _logger = logger;
_trackedDownloadCache = cacheManager.GetCache<TrackedDownload[]>(GetType()); _trackedDownloadCache = cacheManager.GetCache<TrackedDownload[]>(GetType());
@ -73,7 +80,7 @@ namespace NzbDrone.Core.Download
{ {
return _trackedDownloadCache.Get("queued", () => return _trackedDownloadCache.Get("queued", () =>
{ {
UpdateTrackedDownloads(); UpdateTrackedDownloads(_historyService.Grabbed());
return FilterQueuedDownloads(GetTrackedDownloads()); return FilterQueuedDownloads(GetTrackedDownloads());
@ -119,7 +126,7 @@ namespace NzbDrone.Core.Download
.ToList(); .ToList();
} }
private Boolean UpdateTrackedDownloads() private Boolean UpdateTrackedDownloads(List<History.History> grabbedHistory)
{ {
var downloadClients = _downloadClientProvider.GetDownloadClients(); var downloadClients = _downloadClientProvider.GetDownloadClients();
@ -140,13 +147,9 @@ namespace NzbDrone.Core.Download
if (!oldTrackedDownloads.TryGetValue(trackingId, out trackedDownload)) if (!oldTrackedDownloads.TryGetValue(trackingId, out trackedDownload))
{ {
trackedDownload = new TrackedDownload trackedDownload = GetTrackedDownload(trackingId, downloadClient.Definition.Id, downloadItem, grabbedHistory);
{
TrackingId = trackingId, if (trackedDownload == null) continue;
DownloadClient = downloadClient.Definition.Id,
StartedTracking = DateTime.UtcNow,
State = TrackedDownloadState.Unknown
};
_logger.Debug("[{0}] Started tracking download with id {1}.", downloadItem.Title, trackingId); _logger.Debug("[{0}] Started tracking download with id {1}.", downloadItem.Title, trackingId);
stateChanged = true; stateChanged = true;
@ -182,9 +185,9 @@ namespace NzbDrone.Core.Download
var failedHistory = _historyService.Failed(); var failedHistory = _historyService.Failed();
var importedHistory = _historyService.Imported(); var importedHistory = _historyService.Imported();
var stateChanged = UpdateTrackedDownloads(); var stateChanged = UpdateTrackedDownloads(grabbedHistory);
var downloadClients = _downloadClientProvider.GetDownloadClients(); var downloadClients = _downloadClientProvider.GetDownloadClients().ToList();
var trackedDownloads = GetTrackedDownloads(); var trackedDownloads = GetTrackedDownloads();
foreach (var trackedDownload in trackedDownloads) foreach (var trackedDownload in trackedDownloads)
@ -215,6 +218,50 @@ namespace NzbDrone.Core.Download
} }
} }
private TrackedDownload GetTrackedDownload(String trackingId, Int32 downloadClient, DownloadClientItem downloadItem, List<History.History> grabbedHistory)
{
var trackedDownload = new TrackedDownload
{
TrackingId = trackingId,
DownloadClient = downloadClient,
DownloadItem = downloadItem,
StartedTracking = DateTime.UtcNow,
State = TrackedDownloadState.Unknown,
Status = TrackedDownloadStatus.Ok,
};
var historyItems = grabbedHistory.Where(h =>
{
var downloadClientId = h.Data.GetValueOrDefault(DOWNLOAD_CLIENT_ID);
if (downloadClientId == null) return false;
return downloadClientId.Equals(trackedDownload.DownloadItem.DownloadClientId);
}).ToList();
var parsedEpisodeInfo = Parser.Parser.ParseTitle(trackedDownload.DownloadItem.Title);
if (parsedEpisodeInfo == null) return null;
var remoteEpisode = _parsingService.Map(parsedEpisodeInfo, 0);
if (remoteEpisode.Series == null)
{
if (historyItems.Empty()) return null;
trackedDownload.Status = TrackedDownloadStatus.Warning;
trackedDownload.StatusMessages.Add(new TrackedDownloadStatusMessage(
trackedDownload.DownloadItem.Title,
"Series title mismatch, automatic import is not possible")
);
remoteEpisode = _parsingService.Map(parsedEpisodeInfo, historyItems.First().SeriesId, historyItems.Select(h => h.EpisodeId));
}
trackedDownload.RemoteEpisode = remoteEpisode;
return trackedDownload;
}
public void Execute(CheckForFinishedDownloadCommand message) public void Execute(CheckForFinishedDownloadCommand message)
{ {
ProcessTrackedDownloads(); ProcessTrackedDownloads();

View File

@ -3,10 +3,8 @@ using System.Collections.Generic;
using System.Linq; using System.Linq;
using NLog; using NLog;
using NzbDrone.Common; using NzbDrone.Common;
using NzbDrone.Common.Cache;
using NzbDrone.Core.Configuration; using NzbDrone.Core.Configuration;
using NzbDrone.Core.History; using NzbDrone.Core.History;
using NzbDrone.Core.Messaging.Commands;
using NzbDrone.Core.Messaging.Events; using NzbDrone.Core.Messaging.Events;
namespace NzbDrone.Core.Download namespace NzbDrone.Core.Download
@ -68,7 +66,7 @@ namespace NzbDrone.Core.Download
if (!grabbedItems.Any()) if (!grabbedItems.Any())
{ {
UpdateStatusMessage(trackedDownload, LogLevel.Debug, "Download was not grabbed by drone, ignoring download"); UpdateStatusMessage(trackedDownload, LogLevel.Warn, "Download was not grabbed by drone, ignoring download");
return; return;
} }
@ -92,7 +90,7 @@ namespace NzbDrone.Core.Download
if (!grabbedItems.Any()) if (!grabbedItems.Any())
{ {
UpdateStatusMessage(trackedDownload, LogLevel.Debug, "Download wasn't grabbed by drone or not in a category, ignoring download."); UpdateStatusMessage(trackedDownload, LogLevel.Warn, "Download wasn't grabbed by drone or not in a category, ignoring download.");
return; return;
} }
@ -244,8 +242,12 @@ namespace NzbDrone.Core.Download
if (trackedDownload.StatusMessage != statusMessage) if (trackedDownload.StatusMessage != statusMessage)
{ {
trackedDownload.HasError = logLevel >= LogLevel.Warn; trackedDownload.SetStatusLevel(logLevel);
trackedDownload.StatusMessage = statusMessage; trackedDownload.StatusMessage = statusMessage;
trackedDownload.StatusMessages = new List<TrackedDownloadStatusMessage>
{
new TrackedDownloadStatusMessage(trackedDownload.DownloadItem.Title, statusMessage)
};
_logger.Log(logLevel, logMessage); _logger.Log(logLevel, logMessage);
} }
else else

View File

@ -1,5 +1,7 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using NLog;
using NzbDrone.Core.Parser.Model;
namespace NzbDrone.Core.Download namespace NzbDrone.Core.Download
{ {
@ -9,12 +11,33 @@ namespace NzbDrone.Core.Download
public Int32 DownloadClient { get; set; } public Int32 DownloadClient { get; set; }
public DownloadClientItem DownloadItem { get; set; } public DownloadClientItem DownloadItem { get; set; }
public TrackedDownloadState State { get; set; } public TrackedDownloadState State { get; set; }
public TrackedDownloadStatus Status { get; set; }
public DateTime StartedTracking { get; set; } public DateTime StartedTracking { get; set; }
public DateTime LastRetry { get; set; } public DateTime LastRetry { get; set; }
public Int32 RetryCount { get; set; } public Int32 RetryCount { get; set; }
public Boolean HasError { get; set; }
public String StatusMessage { get; set; } public String StatusMessage { get; set; }
public List<String> StatusMessages { get; set; } public RemoteEpisode RemoteEpisode { get; set; }
public List<TrackedDownloadStatusMessage> StatusMessages { get; set; }
public TrackedDownload()
{
StatusMessages = new List<TrackedDownloadStatusMessage>();
}
public void SetStatusLevel(LogLevel logLevel)
{
if (logLevel == LogLevel.Warn)
{
Status = TrackedDownloadStatus.Warning;
}
if (logLevel >= LogLevel.Error)
{
Status = TrackedDownloadStatus.Error;
}
else Status = TrackedDownloadStatus.Ok;
}
} }
public enum TrackedDownloadState public enum TrackedDownloadState
@ -25,4 +48,11 @@ namespace NzbDrone.Core.Download
DownloadFailed, DownloadFailed,
Removed Removed
} }
public enum TrackedDownloadStatus
{
Ok,
Warning,
Error
}
} }

View File

@ -0,0 +1,23 @@
using System;
using System.Collections.Generic;
namespace NzbDrone.Core.Download
{
public class TrackedDownloadStatusMessage
{
public String Title { get; set; }
public List<String> Messages { get; set; }
public TrackedDownloadStatusMessage(String title, List<String> messages)
{
Title = title;
Messages = messages;
}
public TrackedDownloadStatusMessage(String title, String message)
{
Title = title;
Messages = new List<String>{ message };
}
}
}

View File

@ -317,6 +317,7 @@
<Compile Include="Download\Clients\UsenetBlackhole\UsenetBlackhole.cs" /> <Compile Include="Download\Clients\UsenetBlackhole\UsenetBlackhole.cs" />
<Compile Include="Download\Clients\UsenetBlackhole\UsenetBlackholeSettings.cs" /> <Compile Include="Download\Clients\UsenetBlackhole\UsenetBlackholeSettings.cs" />
<Compile Include="Download\CompletedDownloadService.cs" /> <Compile Include="Download\CompletedDownloadService.cs" />
<Compile Include="Download\TrackedDownloadStatusMessage.cs" />
<Compile Include="Download\UsenetClientBase.cs" /> <Compile Include="Download\UsenetClientBase.cs" />
<Compile Include="Download\DownloadClientBase.cs" /> <Compile Include="Download\DownloadClientBase.cs" />
<Compile Include="Download\DownloadClientDefinition.cs" /> <Compile Include="Download\DownloadClientDefinition.cs" />

View File

@ -16,7 +16,8 @@ namespace NzbDrone.Core.Parser
{ {
LocalEpisode GetLocalEpisode(string filename, Series series, bool sceneSource); LocalEpisode GetLocalEpisode(string filename, Series series, bool sceneSource);
Series GetSeries(string title); Series GetSeries(string title);
RemoteEpisode Map(ParsedEpisodeInfo parsedEpisodeInfo, int tvRageId, SearchCriteriaBase searchCriteria = null); RemoteEpisode Map(ParsedEpisodeInfo parsedEpisodeInfo, Int32 tvRageId, SearchCriteriaBase searchCriteria = null);
RemoteEpisode Map(ParsedEpisodeInfo parsedEpisodeInfo, Int32 seriesId, IEnumerable<Int32> episodeIds);
List<Episode> GetEpisodes(ParsedEpisodeInfo parsedEpisodeInfo, Series series, bool sceneSource, SearchCriteriaBase searchCriteria = null); List<Episode> GetEpisodes(ParsedEpisodeInfo parsedEpisodeInfo, Series series, bool sceneSource, SearchCriteriaBase searchCriteria = null);
ParsedEpisodeInfo ParseSpecialEpisodeTitle(string title, int tvRageId, SearchCriteriaBase searchCriteria = null); ParsedEpisodeInfo ParseSpecialEpisodeTitle(string title, int tvRageId, SearchCriteriaBase searchCriteria = null);
ParsedEpisodeInfo ParseSpecialEpisodeTitle(string title, Series series); ParsedEpisodeInfo ParseSpecialEpisodeTitle(string title, Series series);
@ -99,7 +100,7 @@ namespace NzbDrone.Core.Parser
return series; return series;
} }
public RemoteEpisode Map(ParsedEpisodeInfo parsedEpisodeInfo, int tvRageId, SearchCriteriaBase searchCriteria = null) public RemoteEpisode Map(ParsedEpisodeInfo parsedEpisodeInfo, Int32 tvRageId, SearchCriteriaBase searchCriteria = null)
{ {
var remoteEpisode = new RemoteEpisode var remoteEpisode = new RemoteEpisode
{ {
@ -120,6 +121,18 @@ namespace NzbDrone.Core.Parser
return remoteEpisode; return remoteEpisode;
} }
public RemoteEpisode Map(ParsedEpisodeInfo parsedEpisodeInfo, Int32 seriesId, IEnumerable<Int32> episodeIds)
{
var remoteEpisode = new RemoteEpisode
{
ParsedEpisodeInfo = parsedEpisodeInfo,
Series = _seriesService.GetSeries(seriesId),
Episodes = _episodeService.GetEpisodes(episodeIds)
};
return remoteEpisode;
}
public List<Episode> GetEpisodes(ParsedEpisodeInfo parsedEpisodeInfo, Series series, bool sceneSource, SearchCriteriaBase searchCriteria = null) public List<Episode> GetEpisodes(ParsedEpisodeInfo parsedEpisodeInfo, Series series, bool sceneSource, SearchCriteriaBase searchCriteria = null)
{ {
var result = new List<Episode>(); var result = new List<Episode>();

View File

@ -1,5 +1,7 @@
using System; using System;
using System.Collections.Generic;
using NzbDrone.Core.Datastore; using NzbDrone.Core.Datastore;
using NzbDrone.Core.Download;
using NzbDrone.Core.Qualities; using NzbDrone.Core.Qualities;
using NzbDrone.Core.Tv; using NzbDrone.Core.Tv;
using NzbDrone.Core.Parser.Model; using NzbDrone.Core.Parser.Model;
@ -17,7 +19,8 @@ namespace NzbDrone.Core.Queue
public TimeSpan? Timeleft { get; set; } public TimeSpan? Timeleft { get; set; }
public DateTime? EstimatedCompletionTime { get; set; } public DateTime? EstimatedCompletionTime { get; set; }
public String Status { get; set; } public String Status { get; set; }
public String ErrorMessage { get; set; } public String TrackedDownloadStatus { get; set; }
public List<TrackedDownloadStatusMessage> StatusMessages { get; set; }
public RemoteEpisode RemoteEpisode { get; set; } public RemoteEpisode RemoteEpisode { get; set; }
} }
} }

View File

@ -28,33 +28,30 @@ namespace NzbDrone.Core.Queue
return MapQueue(queueItems); return MapQueue(queueItems);
} }
private List<Queue> MapQueue(IEnumerable<TrackedDownload> queueItems) private List<Queue> MapQueue(IEnumerable<TrackedDownload> trackedDownloads)
{ {
var queued = new List<Queue>(); var queued = new List<Queue>();
foreach (var queueItem in queueItems) foreach (var trackedDownload in trackedDownloads)
{ {
foreach (var episode in queueItem.DownloadItem.RemoteEpisode.Episodes) foreach (var episode in trackedDownload.RemoteEpisode.Episodes)
{ {
var queue = new Queue var queue = new Queue
{ {
Id = episode.Id ^ (queueItem.DownloadItem.DownloadClientId.GetHashCode().GetHashCode() << 16), Id = episode.Id ^ (trackedDownload.DownloadItem.DownloadClientId.GetHashCode().GetHashCode() << 16),
Series = queueItem.DownloadItem.RemoteEpisode.Series, Series = trackedDownload.RemoteEpisode.Series,
Episode = episode, Episode = episode,
Quality = queueItem.DownloadItem.RemoteEpisode.ParsedEpisodeInfo.Quality, Quality = trackedDownload.RemoteEpisode.ParsedEpisodeInfo.Quality,
Title = queueItem.DownloadItem.Title, Title = trackedDownload.DownloadItem.Title,
Size = queueItem.DownloadItem.TotalSize, Size = trackedDownload.DownloadItem.TotalSize,
Sizeleft = queueItem.DownloadItem.RemainingSize, Sizeleft = trackedDownload.DownloadItem.RemainingSize,
Timeleft = queueItem.DownloadItem.RemainingTime, Timeleft = trackedDownload.DownloadItem.RemainingTime,
Status = queueItem.DownloadItem.Status.ToString(), Status = trackedDownload.DownloadItem.Status.ToString(),
RemoteEpisode = queueItem.DownloadItem.RemoteEpisode RemoteEpisode = trackedDownload.RemoteEpisode,
TrackedDownloadStatus = trackedDownload.Status.ToString(),
StatusMessages = trackedDownload.StatusMessages
}; };
if (queueItem.HasError)
{
queue.ErrorMessage = queueItem.StatusMessage;
}
if (queue.Timeleft.HasValue) if (queue.Timeleft.HasValue)
{ {
queue.EstimatedCompletionTime = DateTime.UtcNow.Add(queue.Timeleft.Value); queue.EstimatedCompletionTime = DateTime.UtcNow.Add(queue.Timeleft.Value);

View File

@ -14,6 +14,7 @@ namespace NzbDrone.Core.Tv
public interface IEpisodeService public interface IEpisodeService
{ {
Episode GetEpisode(int id); Episode GetEpisode(int id);
List<Episode> GetEpisodes(IEnumerable<Int32> ids);
Episode FindEpisode(int seriesId, int seasonNumber, int episodeNumber); Episode FindEpisode(int seriesId, int seasonNumber, int episodeNumber);
Episode FindEpisode(int seriesId, int absoluteEpisodeNumber); Episode FindEpisode(int seriesId, int absoluteEpisodeNumber);
Episode FindEpisodeByName(int seriesId, int seasonNumber, string episodeTitle); Episode FindEpisodeByName(int seriesId, int seasonNumber, string episodeTitle);
@ -57,6 +58,11 @@ namespace NzbDrone.Core.Tv
return _episodeRepository.Get(id); return _episodeRepository.Get(id);
} }
public List<Episode> GetEpisodes(IEnumerable<int> ids)
{
return _episodeRepository.Get(ids).ToList();
}
public Episode FindEpisode(int seriesId, int seasonNumber, int episodeNumber) public Episode FindEpisode(int seriesId, int seasonNumber, int episodeNumber)
{ {
return _episodeRepository.Find(seriesId, seasonNumber, episodeNumber); return _episodeRepository.Find(seriesId, seasonNumber, episodeNumber);

View File

@ -204,3 +204,7 @@
.icon-nd-deleted:before { .icon-nd-deleted:before {
.icon(@trash); .icon(@trash);
} }
.icon-nd-warning:before {
color: @brand-warning;
}

View File

@ -2,20 +2,26 @@
define( define(
[ [
'marionette',
'Cells/NzbDroneCell' 'Cells/NzbDroneCell'
], function (NzbDroneCell) { ], function (Marionette, NzbDroneCell) {
return NzbDroneCell.extend({ return NzbDroneCell.extend({
className: 'queue-status-cell', className : 'queue-status-cell',
template : 'History/Queue/QueueStatusCellTemplate',
render: function () { render: function () {
this.$el.empty(); this.$el.empty();
if (this.cellValue) { if (this.cellValue) {
var status = this.cellValue.get('status').toLowerCase(); var status = this.cellValue.get('status').toLowerCase();
var errorMessage = (this.cellValue.get('errorMessage') || ''); var trackedDownloadStatus = this.cellValue.get('trackedDownloadStatus').toLowerCase();
var hasError = this.cellValue.get('hasError') || false;
var hasWarning = this.cellValue.get('hasWarning') || false;
var icon = 'icon-nd-downloading'; var icon = 'icon-nd-downloading';
var title = 'Downloading'; var title = 'Downloading';
var itemTitle = this.cellValue.get('title');
var content = itemTitle;
if (status === 'paused') { if (status === 'paused') {
icon = 'icon-pause'; icon = 'icon-pause';
@ -39,7 +45,7 @@ define(
if (status === 'failed') { if (status === 'failed') {
icon = 'icon-nd-download-failed'; icon = 'icon-nd-download-failed';
title = 'Download failed: check download client for more details'; title = 'Download failed';
} }
if (status === 'warning') { if (status === 'warning') {
@ -47,19 +53,31 @@ define(
title = 'Download warning: check download client for more details'; title = 'Download warning: check download client for more details';
} }
if (errorMessage !== '') { if (trackedDownloadStatus === 'warning') {
icon += ' icon-nd-warning';
// title = 'Download failed';
this.templateFunction = Marionette.TemplateCache.get(this.template);
content = this.templateFunction(this.cellValue.toJSON());
}
if (trackedDownloadStatus === 'error') {
if (status === 'completed') { if (status === 'completed') {
icon = 'icon-nd-import-failed'; icon = 'icon-nd-import-failed';
title = 'Import failed'; title = 'Import failed: ' + itemTitle;
} }
else { else {
icon = 'icon-nd-download-failed'; icon = 'icon-nd-download-failed';
title = 'Download failed'; title = 'Download failed';
} }
this.$el.html('<i class="{0}"></i>'.format(icon));
this.templateFunction = Marionette.TemplateCache.get(this.template);
content = this.templateFunction(this.cellValue.toJSON());
}
this.$el.html('<i class="{0}"></i>'.format(icon));
this.$el.popover({ this.$el.popover({
content : errorMessage.replace(new RegExp('\r\n', 'g'), '<br/>'), content : content,
html : true, html : true,
trigger : 'hover', trigger : 'hover',
title : title, title : title,
@ -67,10 +85,6 @@ define(
container: this.$el container: this.$el
}); });
} }
else {
this.$el.html('<i class="{0}" title="{1}"></i>'.format(icon, title));
}
}
return this; return this;
} }

View File

@ -0,0 +1,8 @@
{{#each statusMessages}}
{{title}}
<ul>
{{#each messages}}
<li>{{this}}</li>
{{/each}}
</ul>
{{/each}}