New: Show processing time for pending items on Calendar and Queue

This commit is contained in:
Mark McDowall 2014-08-01 18:08:32 -07:00
parent 3b26e82644
commit c21ffcb5e4
7 changed files with 47 additions and 14 deletions

View File

@ -1,7 +1,6 @@
using System; using System;
using NzbDrone.Api.REST; using NzbDrone.Api.REST;
using NzbDrone.Core.Qualities; using NzbDrone.Core.Qualities;
using NzbDrone.Core.Tv;
using NzbDrone.Api.Series; using NzbDrone.Api.Series;
using NzbDrone.Api.Episodes; using NzbDrone.Api.Episodes;
@ -16,6 +15,7 @@ namespace NzbDrone.Api.Queue
public String Title { get; set; } public String Title { get; set; }
public Decimal Sizeleft { get; set; } public Decimal Sizeleft { get; set; }
public TimeSpan? Timeleft { get; set; } public TimeSpan? Timeleft { get; set; }
public DateTime? EstimatedCompletionTime { get; set; }
public String Status { get; set; } public String Status { get; set; }
public String ErrorMessage { get; set; } public String ErrorMessage { get; set; }
} }

View File

@ -29,21 +29,18 @@ namespace NzbDrone.Core.Download.Pending
private readonly IPendingReleaseRepository _repository; private readonly IPendingReleaseRepository _repository;
private readonly ISeriesService _seriesService; private readonly ISeriesService _seriesService;
private readonly IParsingService _parsingService; private readonly IParsingService _parsingService;
private readonly IDownloadService _downloadService;
private readonly IEventAggregator _eventAggregator; private readonly IEventAggregator _eventAggregator;
private readonly Logger _logger; private readonly Logger _logger;
public PendingReleaseService(IPendingReleaseRepository repository, public PendingReleaseService(IPendingReleaseRepository repository,
ISeriesService seriesService, ISeriesService seriesService,
IParsingService parsingService, IParsingService parsingService,
IDownloadService downloadService,
IEventAggregator eventAggregator, IEventAggregator eventAggregator,
Logger logger) Logger logger)
{ {
_repository = repository; _repository = repository;
_seriesService = seriesService; _seriesService = seriesService;
_parsingService = parsingService; _parsingService = parsingService;
_downloadService = downloadService;
_eventAggregator = eventAggregator; _eventAggregator = eventAggregator;
_logger = logger; _logger = logger;
} }
@ -141,6 +138,9 @@ namespace NzbDrone.Core.Download.Pending
{ {
foreach (var episode in pendingRelease.RemoteEpisode.Episodes) foreach (var episode in pendingRelease.RemoteEpisode.Episodes)
{ {
var ect = pendingRelease.Release.PublishDate.AddHours(
pendingRelease.RemoteEpisode.Series.Profile.Value.GrabDelay);
var queue = new Queue.Queue var queue = new Queue.Queue
{ {
Id = episode.Id ^ (pendingRelease.Id << 16), Id = episode.Id ^ (pendingRelease.Id << 16),
@ -150,10 +150,8 @@ namespace NzbDrone.Core.Download.Pending
Title = pendingRelease.Title, Title = pendingRelease.Title,
Size = pendingRelease.RemoteEpisode.Release.Size, Size = pendingRelease.RemoteEpisode.Release.Size,
Sizeleft = pendingRelease.RemoteEpisode.Release.Size, Sizeleft = pendingRelease.RemoteEpisode.Release.Size,
Timeleft = Timeleft = ect.Subtract(DateTime.UtcNow),
pendingRelease.Release.PublishDate.AddHours( EstimatedCompletionTime = ect,
pendingRelease.RemoteEpisode.Series.Profile.Value.GrabDelay)
.Subtract(DateTime.UtcNow),
Status = "Pending", Status = "Pending",
RemoteEpisode = pendingRelease.RemoteEpisode RemoteEpisode = pendingRelease.RemoteEpisode
}; };

View File

@ -15,6 +15,7 @@ namespace NzbDrone.Core.Queue
public String Title { get; set; } public String Title { get; set; }
public Decimal Sizeleft { get; set; } public Decimal Sizeleft { get; set; }
public TimeSpan? Timeleft { get; set; } public TimeSpan? Timeleft { get; set; }
public DateTime? EstimatedCompletionTime { get; set; }
public String Status { get; set; } public String Status { get; set; }
public String ErrorMessage { get; set; } public String ErrorMessage { get; set; }
public RemoteEpisode RemoteEpisode { get; set; } public RemoteEpisode RemoteEpisode { get; set; }

View File

@ -1,4 +1,5 @@
using System.Linq; using System;
using System.Linq;
using System.Collections.Generic; using System.Collections.Generic;
using NLog; using NLog;
using NzbDrone.Core.Download; using NzbDrone.Core.Download;
@ -52,11 +53,16 @@ namespace NzbDrone.Core.Queue
RemoteEpisode = queueItem.DownloadItem.RemoteEpisode RemoteEpisode = queueItem.DownloadItem.RemoteEpisode
}; };
if (queueItem.HasError) if (queueItem.HasError)
{ {
queue.ErrorMessage = queueItem.StatusMessage; queue.ErrorMessage = queueItem.StatusMessage;
} }
if (queue.Timeleft.HasValue)
{
queue.EstimatedCompletionTime = DateTime.UtcNow.Add(queue.Timeleft.Value);
}
queued.Add(queue); queued.Add(queue);
} }
} }

View File

@ -10,10 +10,11 @@ define(
'System/StatusModel', 'System/StatusModel',
'History/Queue/QueueCollection', 'History/Queue/QueueCollection',
'Config', 'Config',
'Shared/FormatHelpers',
'Mixins/backbone.signalr.mixin', 'Mixins/backbone.signalr.mixin',
'fullcalendar', 'fullcalendar',
'jquery.easypiechart' 'jquery.easypiechart'
], function ($, vent, Marionette, moment, CalendarCollection, StatusModel, QueueCollection, Config) { ], function ($, vent, Marionette, moment, CalendarCollection, StatusModel, QueueCollection, Config, FormatHelpers) {
return Marionette.ItemView.extend({ return Marionette.ItemView.extend({
storageKey: 'calendar.view', storageKey: 'calendar.view',
@ -62,6 +63,16 @@ define(
container: 'body' container: 'body'
}); });
} }
if (event.pending) {
this.$(element).find('.fc-event-time')
.after('<span class="pending pull-right"><i class="icon-time"></i></span>');
this.$(element).find('.pending').tooltip({
title: 'Release will be processed {0}'.format(event.pending),
container: 'body'
});
}
}, },
_getEvents: function (view) { _getEvents: function (view) {
@ -94,7 +105,9 @@ define(
allDay : false, allDay : false,
statusLevel : self._getStatusLevel(model, end), statusLevel : self._getStatusLevel(model, end),
progress : self._getDownloadProgress(model), progress : self._getDownloadProgress(model),
pending : self._getPendingInfo(model),
releaseTitle: self._getReleaseTitle(model), releaseTitle: self._getReleaseTitle(model),
downloading : QueueCollection.findEpisode(model.get('id')),
model : model model : model
}; };
@ -151,6 +164,16 @@ define(
return 100 - (downloading.get('sizeleft') / downloading.get('size') * 100); return 100 - (downloading.get('sizeleft') / downloading.get('size') * 100);
}, },
_getPendingInfo: function (element) {
var pending = QueueCollection.findEpisode(element.get('id'));
if (!pending || pending.get('status').toLocaleLowerCase() !== 'pending') {
return undefined;
}
return new moment(pending.get('estimatedCompletionTime')).fromNow();
},
_getReleaseTitle: function (element) { _getReleaseTitle: function (element) {
var downloading = QueueCollection.findEpisode(element.get('id')); var downloading = QueueCollection.findEpisode(element.get('id'));

View File

@ -21,6 +21,10 @@
.fc-event { .fc-event {
.clickable; .clickable;
.pending {
margin-right : 4px;
}
} }
th { th {

View File

@ -3,8 +3,9 @@
define( define(
[ [
'Cells/NzbDroneCell', 'Cells/NzbDroneCell',
'filesize' 'filesize',
], function (NzbDroneCell, fileSize) { 'moment'
], function (NzbDroneCell, fileSize, Moment) {
return NzbDroneCell.extend({ return NzbDroneCell.extend({
className: 'timeleft-cell', className: 'timeleft-cell',
@ -17,7 +18,7 @@ define(
//If the release is pending we want to use the timeleft as the time it will be processed at //If the release is pending we want to use the timeleft as the time it will be processed at
if (this.cellValue.get('status').toLowerCase() === 'pending') { if (this.cellValue.get('status').toLowerCase() === 'pending') {
this.$el.html('-'); this.$el.html('-');
this.$el.attr('title', 'Will be processed again in: {0}'.format(this.cellValue.get('timeleft'))); this.$el.attr('title', 'Will be processed {0}'.format(new Moment(this.cellValue.get('estimatedCompletionTime')).calendar()));
this.$el.attr('data-container', 'body'); this.$el.attr('data-container', 'body');
return this; return this;