From b319c44bbd8f632b7d1ee902fe8ec0da1a110a7c Mon Sep 17 00:00:00 2001 From: Qstick Date: Wed, 19 Feb 2020 20:27:25 +0000 Subject: [PATCH] New: Store Task StartTime, Show Duration in UI --- .../Tasks/Scheduled/ScheduledTaskRow.css | 1 + .../Tasks/Scheduled/ScheduledTaskRow.js | 21 +++++++++++++++++++ .../System/Tasks/Scheduled/ScheduledTasks.js | 5 +++++ src/Lidarr.Api.V1/System/Tasks/TaskModule.cs | 1 + .../System/Tasks/TaskResource.cs | 3 +++ .../Datastore/Migration/040_task_duration.cs | 14 +++++++++++++ src/NzbDrone.Core/Jobs/ScheduledTask.cs | 1 + .../Jobs/ScheduledTaskRepository.cs | 9 ++++---- src/NzbDrone.Core/Jobs/Scheduler.cs | 2 +- src/NzbDrone.Core/Jobs/TaskManager.cs | 2 +- .../Messaging/Commands/Command.cs | 1 + .../Messaging/Commands/CommandQueueManager.cs | 5 +++-- 12 files changed, 57 insertions(+), 8 deletions(-) create mode 100644 src/NzbDrone.Core/Datastore/Migration/040_task_duration.cs diff --git a/frontend/src/System/Tasks/Scheduled/ScheduledTaskRow.css b/frontend/src/System/Tasks/Scheduled/ScheduledTaskRow.css index 924963258..a5c84ee3a 100644 --- a/frontend/src/System/Tasks/Scheduled/ScheduledTaskRow.css +++ b/frontend/src/System/Tasks/Scheduled/ScheduledTaskRow.css @@ -5,6 +5,7 @@ } .lastExecution, +.lastDuration, .nextExecution { composes: cell from '~Components/Table/Cells/TableRowCell.css'; diff --git a/frontend/src/System/Tasks/Scheduled/ScheduledTaskRow.js b/frontend/src/System/Tasks/Scheduled/ScheduledTaskRow.js index 82cedc720..deeb161bc 100644 --- a/frontend/src/System/Tasks/Scheduled/ScheduledTaskRow.js +++ b/frontend/src/System/Tasks/Scheduled/ScheduledTaskRow.js @@ -3,6 +3,7 @@ import PropTypes from 'prop-types'; import React, { Component } from 'react'; import formatDate from 'Utilities/Date/formatDate'; import formatDateTime from 'Utilities/Date/formatDateTime'; +import formatTimeSpan from 'Utilities/Date/formatTimeSpan'; import { icons } from 'Helpers/Props'; import SpinnerIconButton from 'Components/Link/SpinnerIconButton'; import TableRow from 'Components/Table/TableRow'; @@ -91,6 +92,8 @@ class ScheduledTaskRow extends Component { name, interval, lastExecution, + lastStartTime, + lastDuration, nextExecution, isQueued, isExecuting, @@ -108,6 +111,7 @@ class ScheduledTaskRow extends Component { const executeNow = !isDisabled && moment().isAfter(nextExecution); const hasNextExecutionTime = !isDisabled && !executeNow; const duration = moment.duration(interval, 'minutes').humanize().replace(/an?(?=\s)/, '1'); + const hasLastStartTime = moment(lastStartTime).isAfter('2010-01-01'); return ( @@ -125,6 +129,21 @@ class ScheduledTaskRow extends Component { {lastExecutionTime} + { + !hasLastStartTime && + - + } + + { + hasLastStartTime && + + {formatTimeSpan(lastDuration)} + + } + { isDisabled && - @@ -169,6 +188,8 @@ ScheduledTaskRow.propTypes = { name: PropTypes.string.isRequired, interval: PropTypes.number.isRequired, lastExecution: PropTypes.string.isRequired, + lastStartTime: PropTypes.string.isRequired, + lastDuration: PropTypes.string.isRequired, nextExecution: PropTypes.string.isRequired, isQueued: PropTypes.bool.isRequired, isExecuting: PropTypes.bool.isRequired, diff --git a/frontend/src/System/Tasks/Scheduled/ScheduledTasks.js b/frontend/src/System/Tasks/Scheduled/ScheduledTasks.js index 7c6fe8a32..e481d3781 100644 --- a/frontend/src/System/Tasks/Scheduled/ScheduledTasks.js +++ b/frontend/src/System/Tasks/Scheduled/ScheduledTasks.js @@ -22,6 +22,11 @@ const columns = [ label: 'Last Execution', isVisible: true }, + { + name: 'lastDuration', + label: 'Last Duration', + isVisible: true + }, { name: 'nextExecution', label: 'Next Execution', diff --git a/src/Lidarr.Api.V1/System/Tasks/TaskModule.cs b/src/Lidarr.Api.V1/System/Tasks/TaskModule.cs index 924f1aef6..61a8c6a7f 100644 --- a/src/Lidarr.Api.V1/System/Tasks/TaskModule.cs +++ b/src/Lidarr.Api.V1/System/Tasks/TaskModule.cs @@ -53,6 +53,7 @@ namespace Lidarr.Api.V1.System.Tasks TaskName = taskName, Interval = scheduledTask.Interval, LastExecution = scheduledTask.LastExecution, + LastStartTime = scheduledTask.LastStartTime, NextExecution = scheduledTask.LastExecution.AddMinutes(scheduledTask.Interval) }; } diff --git a/src/Lidarr.Api.V1/System/Tasks/TaskResource.cs b/src/Lidarr.Api.V1/System/Tasks/TaskResource.cs index e0454b664..bec5bdf1d 100644 --- a/src/Lidarr.Api.V1/System/Tasks/TaskResource.cs +++ b/src/Lidarr.Api.V1/System/Tasks/TaskResource.cs @@ -9,6 +9,9 @@ namespace Lidarr.Api.V1.System.Tasks public string TaskName { get; set; } public int Interval { get; set; } public DateTime LastExecution { get; set; } + public DateTime LastStartTime { get; set; } public DateTime NextExecution { get; set; } + + public TimeSpan LastDuration => LastExecution - LastStartTime; } } diff --git a/src/NzbDrone.Core/Datastore/Migration/040_task_duration.cs b/src/NzbDrone.Core/Datastore/Migration/040_task_duration.cs new file mode 100644 index 000000000..3a0b8e16a --- /dev/null +++ b/src/NzbDrone.Core/Datastore/Migration/040_task_duration.cs @@ -0,0 +1,14 @@ +using FluentMigrator; +using NzbDrone.Core.Datastore.Migration.Framework; + +namespace NzbDrone.Core.Datastore.Migration +{ + [Migration(40)] + public class task_duration : NzbDroneMigrationBase + { + protected override void MainDbUpgrade() + { + Alter.Table("ScheduledTasks").AddColumn("LastStartTime").AsDateTime().Nullable(); + } + } +} diff --git a/src/NzbDrone.Core/Jobs/ScheduledTask.cs b/src/NzbDrone.Core/Jobs/ScheduledTask.cs index dcf7f5908..3cdfa3132 100644 --- a/src/NzbDrone.Core/Jobs/ScheduledTask.cs +++ b/src/NzbDrone.Core/Jobs/ScheduledTask.cs @@ -8,5 +8,6 @@ namespace NzbDrone.Core.Jobs public string TypeName { get; set; } public int Interval { get; set; } public DateTime LastExecution { get; set; } + public DateTime LastStartTime { get; set; } } } diff --git a/src/NzbDrone.Core/Jobs/ScheduledTaskRepository.cs b/src/NzbDrone.Core/Jobs/ScheduledTaskRepository.cs index c565017d2..4b46b2efc 100644 --- a/src/NzbDrone.Core/Jobs/ScheduledTaskRepository.cs +++ b/src/NzbDrone.Core/Jobs/ScheduledTaskRepository.cs @@ -8,7 +8,7 @@ namespace NzbDrone.Core.Jobs public interface IScheduledTaskRepository : IBasicRepository { ScheduledTask GetDefinition(Type type); - void SetLastExecutionTime(int id, DateTime executionTime); + void SetLastExecutionTime(int id, DateTime executionTime, DateTime startTime); } public class ScheduledTaskRepository : BasicRepository, IScheduledTaskRepository @@ -23,15 +23,16 @@ namespace NzbDrone.Core.Jobs return Query.Where(c => c.TypeName == type.FullName).Single(); } - public void SetLastExecutionTime(int id, DateTime executionTime) + public void SetLastExecutionTime(int id, DateTime executionTime, DateTime startTime) { var task = new ScheduledTask { Id = id, - LastExecution = executionTime + LastExecution = executionTime, + LastStartTime = startTime }; - SetFields(task, scheduledTask => scheduledTask.LastExecution); + SetFields(task, scheduledTask => scheduledTask.LastExecution, scheduledTask => scheduledTask.LastStartTime); } } } diff --git a/src/NzbDrone.Core/Jobs/Scheduler.cs b/src/NzbDrone.Core/Jobs/Scheduler.cs index 4d854d508..dffe4dba7 100644 --- a/src/NzbDrone.Core/Jobs/Scheduler.cs +++ b/src/NzbDrone.Core/Jobs/Scheduler.cs @@ -39,7 +39,7 @@ namespace NzbDrone.Core.Jobs foreach (var task in tasks) { - _commandQueueManager.Push(task.TypeName, task.LastExecution, CommandPriority.Low, CommandTrigger.Scheduled); + _commandQueueManager.Push(task.TypeName, task.LastExecution, task.LastStartTime, CommandPriority.Low, CommandTrigger.Scheduled); } } finally diff --git a/src/NzbDrone.Core/Jobs/TaskManager.cs b/src/NzbDrone.Core/Jobs/TaskManager.cs index c7c0b6561..244b647df 100644 --- a/src/NzbDrone.Core/Jobs/TaskManager.cs +++ b/src/NzbDrone.Core/Jobs/TaskManager.cs @@ -145,7 +145,7 @@ namespace NzbDrone.Core.Jobs if (scheduledTask != null && message.Command.Body.UpdateScheduledTask) { _logger.Trace("Updating last run time for: {0}", scheduledTask.TypeName); - _scheduledTaskRepository.SetLastExecutionTime(scheduledTask.Id, DateTime.UtcNow); + _scheduledTaskRepository.SetLastExecutionTime(scheduledTask.Id, DateTime.UtcNow, message.Command.StartedAt.Value); } } diff --git a/src/NzbDrone.Core/Messaging/Commands/Command.cs b/src/NzbDrone.Core/Messaging/Commands/Command.cs index a4575ebe9..4402bfe6c 100644 --- a/src/NzbDrone.Core/Messaging/Commands/Command.cs +++ b/src/NzbDrone.Core/Messaging/Commands/Command.cs @@ -28,6 +28,7 @@ namespace NzbDrone.Core.Messaging.Commands public string Name { get; private set; } public DateTime? LastExecutionTime { get; set; } + public DateTime? LastStartTime { get; set; } public CommandTrigger Trigger { get; set; } public bool SuppressMessages { get; set; } diff --git a/src/NzbDrone.Core/Messaging/Commands/CommandQueueManager.cs b/src/NzbDrone.Core/Messaging/Commands/CommandQueueManager.cs index 5567fbd63..9a65e5abf 100644 --- a/src/NzbDrone.Core/Messaging/Commands/CommandQueueManager.cs +++ b/src/NzbDrone.Core/Messaging/Commands/CommandQueueManager.cs @@ -19,7 +19,7 @@ namespace NzbDrone.Core.Messaging.Commands where TCommand : Command; CommandModel Push(TCommand command, CommandPriority priority = CommandPriority.Normal, CommandTrigger trigger = CommandTrigger.Unspecified) where TCommand : Command; - CommandModel Push(string commandName, DateTime? lastExecutionTime, CommandPriority priority = CommandPriority.Normal, CommandTrigger trigger = CommandTrigger.Unspecified); + CommandModel Push(string commandName, DateTime? lastExecutionTime, DateTime? lastStartTime, CommandPriority priority = CommandPriority.Normal, CommandTrigger trigger = CommandTrigger.Unspecified); IEnumerable Queue(CancellationToken cancellationToken); List All(); CommandModel Get(int id); @@ -128,10 +128,11 @@ namespace NzbDrone.Core.Messaging.Commands return commandModel; } - public CommandModel Push(string commandName, DateTime? lastExecutionTime, CommandPriority priority = CommandPriority.Normal, CommandTrigger trigger = CommandTrigger.Unspecified) + public CommandModel Push(string commandName, DateTime? lastExecutionTime, DateTime? lastStartTime, CommandPriority priority = CommandPriority.Normal, CommandTrigger trigger = CommandTrigger.Unspecified) { dynamic command = GetCommand(commandName); command.LastExecutionTime = lastExecutionTime; + command.LastStartTime = lastStartTime; command.Trigger = trigger; return Push(command, priority, trigger);