parent
3b579900bb
commit
f9840c66f8
|
@ -10,9 +10,43 @@ import ModalFooter from 'Components/Modal/ModalFooter';
|
||||||
import UpdateChanges from 'System/Updates/UpdateChanges';
|
import UpdateChanges from 'System/Updates/UpdateChanges';
|
||||||
import styles from './AppUpdatedModalContent.css';
|
import styles from './AppUpdatedModalContent.css';
|
||||||
|
|
||||||
|
function mergeUpdates(items, version, prevVersion) {
|
||||||
|
let installedIndex = items.findIndex((u) => u.version === version);
|
||||||
|
let installedPreviouslyIndex = items.findIndex((u) => u.version === prevVersion);
|
||||||
|
|
||||||
|
if (installedIndex === -1) {
|
||||||
|
installedIndex = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (installedPreviouslyIndex === -1) {
|
||||||
|
installedPreviouslyIndex = items.length;
|
||||||
|
} else if (installedPreviouslyIndex === installedIndex && items.size()) {
|
||||||
|
installedPreviouslyIndex += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
const appliedUpdates = items.slice(installedIndex, installedPreviouslyIndex);
|
||||||
|
|
||||||
|
if (!appliedUpdates.length) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const appliedChanges = { new: [], fixed: [] };
|
||||||
|
appliedUpdates.forEach((u) => {
|
||||||
|
if (u.changes) {
|
||||||
|
appliedChanges.new.push(... u.changes.new);
|
||||||
|
appliedChanges.fixed.push(... u.changes.fixed);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const mergedUpdate = Object.assign({}, appliedUpdates[0], { changes: appliedChanges });
|
||||||
|
|
||||||
|
return mergedUpdate;
|
||||||
|
}
|
||||||
|
|
||||||
function AppUpdatedModalContent(props) {
|
function AppUpdatedModalContent(props) {
|
||||||
const {
|
const {
|
||||||
version,
|
version,
|
||||||
|
prevVersion,
|
||||||
isPopulated,
|
isPopulated,
|
||||||
error,
|
error,
|
||||||
items,
|
items,
|
||||||
|
@ -20,7 +54,7 @@ function AppUpdatedModalContent(props) {
|
||||||
onModalClose
|
onModalClose
|
||||||
} = props;
|
} = props;
|
||||||
|
|
||||||
const update = items[0];
|
const update = mergeUpdates(items, version, prevVersion);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ModalContent onModalClose={onModalClose}>
|
<ModalContent onModalClose={onModalClose}>
|
||||||
|
@ -30,7 +64,7 @@ function AppUpdatedModalContent(props) {
|
||||||
|
|
||||||
<ModalBody>
|
<ModalBody>
|
||||||
<div>
|
<div>
|
||||||
Version <span className={styles.version}>{version}</span> of Sonarr has been installed, in order to get the latest changes you'll need to reload Sonarr.
|
Sonarr has been updated to version <span className={styles.version}>{version}</span>, in order to get the latest changes you'll need to reload Sonarr.
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -88,6 +122,7 @@ function AppUpdatedModalContent(props) {
|
||||||
|
|
||||||
AppUpdatedModalContent.propTypes = {
|
AppUpdatedModalContent.propTypes = {
|
||||||
version: PropTypes.string.isRequired,
|
version: PropTypes.string.isRequired,
|
||||||
|
prevVersion: PropTypes.string,
|
||||||
isPopulated: PropTypes.bool.isRequired,
|
isPopulated: PropTypes.bool.isRequired,
|
||||||
error: PropTypes.object,
|
error: PropTypes.object,
|
||||||
items: PropTypes.arrayOf(PropTypes.object).isRequired,
|
items: PropTypes.arrayOf(PropTypes.object).isRequired,
|
||||||
|
|
|
@ -8,8 +8,9 @@ import AppUpdatedModalContent from './AppUpdatedModalContent';
|
||||||
function createMapStateToProps() {
|
function createMapStateToProps() {
|
||||||
return createSelector(
|
return createSelector(
|
||||||
(state) => state.app.version,
|
(state) => state.app.version,
|
||||||
|
(state) => state.app.prevVersion,
|
||||||
(state) => state.system.updates,
|
(state) => state.system.updates,
|
||||||
(version, updates) => {
|
(version, prevVersion, updates) => {
|
||||||
const {
|
const {
|
||||||
isPopulated,
|
isPopulated,
|
||||||
error,
|
error,
|
||||||
|
@ -18,6 +19,7 @@ function createMapStateToProps() {
|
||||||
|
|
||||||
return {
|
return {
|
||||||
version,
|
version,
|
||||||
|
prevVersion,
|
||||||
isPopulated,
|
isPopulated,
|
||||||
error,
|
error,
|
||||||
items
|
items
|
||||||
|
|
|
@ -117,6 +117,9 @@ export const reducers = createHandleActions({
|
||||||
};
|
};
|
||||||
|
|
||||||
if (state.version !== version) {
|
if (state.version !== version) {
|
||||||
|
if (!state.prevVersion) {
|
||||||
|
newState.prevVersion = state.version;
|
||||||
|
}
|
||||||
newState.isUpdated = true;
|
newState.isUpdated = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,6 +3,7 @@ import PropTypes from 'prop-types';
|
||||||
import React, { Component, Fragment } from 'react';
|
import React, { Component, Fragment } from 'react';
|
||||||
import { icons, kinds } from 'Helpers/Props';
|
import { icons, kinds } from 'Helpers/Props';
|
||||||
import formatDate from 'Utilities/Date/formatDate';
|
import formatDate from 'Utilities/Date/formatDate';
|
||||||
|
import formatDateTime from 'Utilities/Date/formatDateTime';
|
||||||
import LoadingIndicator from 'Components/Loading/LoadingIndicator';
|
import LoadingIndicator from 'Components/Loading/LoadingIndicator';
|
||||||
import SpinnerButton from 'Components/Link/SpinnerButton';
|
import SpinnerButton from 'Components/Link/SpinnerButton';
|
||||||
import InlineMarkdown from 'Components/Markdown/InlineMarkdown';
|
import InlineMarkdown from 'Components/Markdown/InlineMarkdown';
|
||||||
|
@ -30,6 +31,8 @@ class Updates extends Component {
|
||||||
updateMechanism,
|
updateMechanism,
|
||||||
updateMechanismMessage,
|
updateMechanismMessage,
|
||||||
shortDateFormat,
|
shortDateFormat,
|
||||||
|
longDateFormat,
|
||||||
|
timeFormat,
|
||||||
onInstallLatestPress
|
onInstallLatestPress
|
||||||
} = this.props;
|
} = this.props;
|
||||||
|
|
||||||
|
@ -134,7 +137,12 @@ class Updates extends Component {
|
||||||
<div className={styles.info}>
|
<div className={styles.info}>
|
||||||
<div className={styles.version}>{update.version}</div>
|
<div className={styles.version}>{update.version}</div>
|
||||||
<div className={styles.space}>—</div>
|
<div className={styles.space}>—</div>
|
||||||
<div className={styles.date}>{formatDate(update.releaseDate, shortDateFormat)}</div>
|
<div
|
||||||
|
className={styles.date}
|
||||||
|
title={formatDateTime(update.releaseDate, longDateFormat, timeFormat)}
|
||||||
|
>
|
||||||
|
{formatDate(update.releaseDate, shortDateFormat)}
|
||||||
|
</div>
|
||||||
|
|
||||||
{
|
{
|
||||||
update.branch === 'master' ?
|
update.branch === 'master' ?
|
||||||
|
@ -151,11 +159,24 @@ class Updates extends Component {
|
||||||
<Label
|
<Label
|
||||||
className={styles.label}
|
className={styles.label}
|
||||||
kind={kinds.SUCCESS}
|
kind={kinds.SUCCESS}
|
||||||
|
title={formatDateTime(update.installedOn, longDateFormat, timeFormat)}
|
||||||
>
|
>
|
||||||
Currently Installed
|
Currently Installed
|
||||||
</Label> :
|
</Label> :
|
||||||
null
|
null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
update.version !== currentVersion && update.installedOn ?
|
||||||
|
<Label
|
||||||
|
className={styles.label}
|
||||||
|
kind={kinds.INVERSE}
|
||||||
|
title={formatDateTime(update.installedOn, longDateFormat, timeFormat)}
|
||||||
|
>
|
||||||
|
Previously Installed
|
||||||
|
</Label> :
|
||||||
|
null
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -215,6 +236,8 @@ Updates.propTypes = {
|
||||||
updateMechanism: PropTypes.string,
|
updateMechanism: PropTypes.string,
|
||||||
updateMechanismMessage: PropTypes.string,
|
updateMechanismMessage: PropTypes.string,
|
||||||
shortDateFormat: PropTypes.string.isRequired,
|
shortDateFormat: PropTypes.string.isRequired,
|
||||||
|
longDateFormat: PropTypes.string.isRequired,
|
||||||
|
timeFormat: PropTypes.string.isRequired,
|
||||||
onInstallLatestPress: PropTypes.func.isRequired
|
onInstallLatestPress: PropTypes.func.isRequired
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -45,7 +45,9 @@ function createMapStateToProps() {
|
||||||
isInstallingUpdate,
|
isInstallingUpdate,
|
||||||
updateMechanism: generalSettings.item.updateMechanism,
|
updateMechanism: generalSettings.item.updateMechanism,
|
||||||
updateMechanismMessage: status.packageUpdateMechanismMessage,
|
updateMechanismMessage: status.packageUpdateMechanismMessage,
|
||||||
shortDateFormat: uiSettings.shortDateFormat
|
shortDateFormat: uiSettings.shortDateFormat,
|
||||||
|
longDateFormat: uiSettings.longDateFormat,
|
||||||
|
timeFormat: uiSettings.timeFormat
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
|
@ -44,7 +44,7 @@ namespace NzbDrone.Core.Test.UpdateTests
|
||||||
{
|
{
|
||||||
const string branch = "master";
|
const string branch = "master";
|
||||||
UseRealHttp();
|
UseRealHttp();
|
||||||
var recent = Subject.GetRecentUpdates(branch, new Version(2, 0));
|
var recent = Subject.GetRecentUpdates(branch, new Version(2, 0), null);
|
||||||
|
|
||||||
recent.Should().NotBeEmpty();
|
recent.Should().NotBeEmpty();
|
||||||
recent.Should().OnlyContain(c => c.Hash.IsNotNullOrWhiteSpace());
|
recent.Should().OnlyContain(c => c.Hash.IsNotNullOrWhiteSpace());
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
using System;
|
||||||
|
using Marr.Data.Converters;
|
||||||
|
using Marr.Data.Mapping;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Datastore.Converters
|
||||||
|
{
|
||||||
|
public class SystemVersionConverter : IConverter
|
||||||
|
{
|
||||||
|
public object FromDB(ConverterContext context)
|
||||||
|
{
|
||||||
|
if (context.DbValue is string version)
|
||||||
|
{
|
||||||
|
return Version.Parse(version);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public object FromDB(ColumnMap map, object dbValue)
|
||||||
|
{
|
||||||
|
if (dbValue is string version)
|
||||||
|
{
|
||||||
|
return Version.Parse(version);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public object ToDB(object clrValue)
|
||||||
|
{
|
||||||
|
if (clrValue is Version version)
|
||||||
|
{
|
||||||
|
return version.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
return DBNull.Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Type DbType => typeof(String);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,31 @@
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Data;
|
||||||
|
using FluentMigrator;
|
||||||
|
using NzbDrone.Common.EnvironmentInfo;
|
||||||
|
using NzbDrone.Common.Serializer;
|
||||||
|
using NzbDrone.Core.Datastore.Converters;
|
||||||
|
using NzbDrone.Core.Datastore.Migration.Framework;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Datastore.Migration
|
||||||
|
{
|
||||||
|
[Migration(141)]
|
||||||
|
public class add_update_history : NzbDroneMigrationBase
|
||||||
|
{
|
||||||
|
protected override void LogDbUpgrade()
|
||||||
|
{
|
||||||
|
Create.TableForModel("UpdateHistory")
|
||||||
|
.WithColumn("Date").AsDateTime().NotNullable().Indexed()
|
||||||
|
.WithColumn("Version").AsString().NotNullable()
|
||||||
|
.WithColumn("EventType").AsInt32().NotNullable();
|
||||||
|
|
||||||
|
Insert.IntoTable("UpdateHistory")
|
||||||
|
.Row(new
|
||||||
|
{
|
||||||
|
Date = new UtcConverter().ToDB(DateTime.UtcNow).ToString(),
|
||||||
|
Version = BuildInfo.Version.ToString(),
|
||||||
|
EventType = 2
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -39,6 +39,7 @@ using NzbDrone.Core.Messaging.Commands;
|
||||||
using NzbDrone.Core.Languages;
|
using NzbDrone.Core.Languages;
|
||||||
using NzbDrone.Core.Profiles.Languages;
|
using NzbDrone.Core.Profiles.Languages;
|
||||||
using NzbDrone.Core.Profiles.Releases;
|
using NzbDrone.Core.Profiles.Releases;
|
||||||
|
using NzbDrone.Core.Update.History;
|
||||||
|
|
||||||
namespace NzbDrone.Core.Datastore
|
namespace NzbDrone.Core.Datastore
|
||||||
{
|
{
|
||||||
|
@ -139,6 +140,8 @@ namespace NzbDrone.Core.Datastore
|
||||||
|
|
||||||
Mapper.Entity<DownloadHistory>().RegisterModel("DownloadHistory")
|
Mapper.Entity<DownloadHistory>().RegisterModel("DownloadHistory")
|
||||||
.AutoMapChildModels();
|
.AutoMapChildModels();
|
||||||
|
|
||||||
|
Mapper.Entity<UpdateHistory>().RegisterModel("UpdateHistory");
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void RegisterMappers()
|
private static void RegisterMappers()
|
||||||
|
@ -168,6 +171,7 @@ namespace NzbDrone.Core.Datastore
|
||||||
MapRepository.Instance.RegisterTypeConverter(typeof(Command), new CommandConverter());
|
MapRepository.Instance.RegisterTypeConverter(typeof(Command), new CommandConverter());
|
||||||
MapRepository.Instance.RegisterTypeConverter(typeof(TimeSpan), new TimeSpanConverter());
|
MapRepository.Instance.RegisterTypeConverter(typeof(TimeSpan), new TimeSpanConverter());
|
||||||
MapRepository.Instance.RegisterTypeConverter(typeof(TimeSpan?), new TimeSpanConverter());
|
MapRepository.Instance.RegisterTypeConverter(typeof(TimeSpan?), new TimeSpanConverter());
|
||||||
|
MapRepository.Instance.RegisterTypeConverter(typeof(Version), new SystemVersionConverter());
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void RegisterProviderSettingConverter()
|
private static void RegisterProviderSettingConverter()
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
using System;
|
||||||
|
using NzbDrone.Common.Messaging;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Update.History.Events
|
||||||
|
{
|
||||||
|
public class UpdateInstalledEvent : IEvent
|
||||||
|
{
|
||||||
|
public Version PreviousVerison { get; set; }
|
||||||
|
public Version NewVersion { get; set; }
|
||||||
|
|
||||||
|
public UpdateInstalledEvent(Version previousVersion, Version newVersion)
|
||||||
|
{
|
||||||
|
PreviousVerison = previousVersion;
|
||||||
|
NewVersion = newVersion;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,12 @@
|
||||||
|
using System;
|
||||||
|
using NzbDrone.Core.Datastore;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Update.History
|
||||||
|
{
|
||||||
|
public class UpdateHistory : ModelBase
|
||||||
|
{
|
||||||
|
public DateTime Date { get; set; }
|
||||||
|
public Version Version { get; set; }
|
||||||
|
public UpdateHistoryEventType EventType { get; set; }
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,9 @@
|
||||||
|
namespace NzbDrone.Core.Update.History
|
||||||
|
{
|
||||||
|
public enum UpdateHistoryEventType
|
||||||
|
{
|
||||||
|
Unknown = 0,
|
||||||
|
Initiated = 1,
|
||||||
|
Installed = 2
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,56 @@
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using NzbDrone.Core.Datastore;
|
||||||
|
using NzbDrone.Core.Messaging.Events;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Update.History
|
||||||
|
{
|
||||||
|
public interface IUpdateHistoryRepository : IBasicRepository<UpdateHistory>
|
||||||
|
{
|
||||||
|
UpdateHistory LastInstalled();
|
||||||
|
UpdateHistory PreviouslyInstalled();
|
||||||
|
List<UpdateHistory> InstalledSince(DateTime dateTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
public class UpdateHistoryRepository : BasicRepository<UpdateHistory>, IUpdateHistoryRepository
|
||||||
|
{
|
||||||
|
public UpdateHistoryRepository(ILogDatabase logDatabase, IEventAggregator eventAggregator)
|
||||||
|
: base(logDatabase, eventAggregator)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public UpdateHistory LastInstalled()
|
||||||
|
{
|
||||||
|
var history = Query.Where(v => v.EventType == UpdateHistoryEventType.Installed)
|
||||||
|
.OrderBy(v => v.Date)
|
||||||
|
.Take(1)
|
||||||
|
.FirstOrDefault();
|
||||||
|
|
||||||
|
return history;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UpdateHistory PreviouslyInstalled()
|
||||||
|
{
|
||||||
|
var history = Query.Where(v => v.EventType == UpdateHistoryEventType.Installed)
|
||||||
|
.OrderBy(v => v.Date)
|
||||||
|
.Skip(1)
|
||||||
|
.Take(1)
|
||||||
|
.FirstOrDefault();
|
||||||
|
|
||||||
|
return history;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<UpdateHistory> InstalledSince(DateTime dateTime)
|
||||||
|
{
|
||||||
|
var history = Query.Where(v => v.EventType == UpdateHistoryEventType.Installed && v.Date >= dateTime)
|
||||||
|
.OrderBy(v => v.Date)
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
return history;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,71 @@
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using NzbDrone.Common.EnvironmentInfo;
|
||||||
|
using NzbDrone.Core.Lifecycle;
|
||||||
|
using NzbDrone.Core.Messaging.Events;
|
||||||
|
using NzbDrone.Core.Update.History.Events;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Update.History
|
||||||
|
{
|
||||||
|
public interface IUpdateHistoryService
|
||||||
|
{
|
||||||
|
Version PreviouslyInstalled();
|
||||||
|
List<UpdateHistory> InstalledSince(DateTime dateTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
public class UpdateHistoryService : IUpdateHistoryService, IHandle<ApplicationStartedEvent>, IHandleAsync<ApplicationStartedEvent>
|
||||||
|
{
|
||||||
|
private readonly IUpdateHistoryRepository _repository;
|
||||||
|
private readonly IEventAggregator _eventAggregator;
|
||||||
|
private Version _prevVersion;
|
||||||
|
|
||||||
|
public UpdateHistoryService(IUpdateHistoryRepository repository, IEventAggregator eventAggregator)
|
||||||
|
{
|
||||||
|
_repository = repository;
|
||||||
|
_eventAggregator = eventAggregator;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Version PreviouslyInstalled()
|
||||||
|
{
|
||||||
|
var history = _repository.PreviouslyInstalled();
|
||||||
|
|
||||||
|
return history?.Version;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<UpdateHistory> InstalledSince(DateTime dateTime)
|
||||||
|
{
|
||||||
|
return _repository.InstalledSince(dateTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Handle(ApplicationStartedEvent message)
|
||||||
|
{
|
||||||
|
if (BuildInfo.Version.Major == 10)
|
||||||
|
{
|
||||||
|
// Don't save dev versions, they change constantly
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var history = _repository.LastInstalled();
|
||||||
|
|
||||||
|
if (history == null || history.Version != BuildInfo.Version)
|
||||||
|
{
|
||||||
|
_prevVersion = history.Version;
|
||||||
|
|
||||||
|
_repository.Insert(new UpdateHistory
|
||||||
|
{
|
||||||
|
Date = DateTime.UtcNow,
|
||||||
|
Version = BuildInfo.Version,
|
||||||
|
EventType = UpdateHistoryEventType.Installed
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void HandleAsync(ApplicationStartedEvent message)
|
||||||
|
{
|
||||||
|
if (_prevVersion != null)
|
||||||
|
{
|
||||||
|
_eventAggregator.PublishEvent(new UpdateInstalledEvent(_prevVersion, BuildInfo.Version));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,6 +1,8 @@
|
||||||
using System.Collections.Generic;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using NzbDrone.Common.EnvironmentInfo;
|
using NzbDrone.Common.EnvironmentInfo;
|
||||||
using NzbDrone.Core.Configuration;
|
using NzbDrone.Core.Configuration;
|
||||||
|
using NzbDrone.Core.Update.History;
|
||||||
|
|
||||||
namespace NzbDrone.Core.Update
|
namespace NzbDrone.Core.Update
|
||||||
{
|
{
|
||||||
|
@ -13,18 +15,23 @@ namespace NzbDrone.Core.Update
|
||||||
{
|
{
|
||||||
private readonly IConfigFileProvider _configFileProvider;
|
private readonly IConfigFileProvider _configFileProvider;
|
||||||
private readonly IUpdatePackageProvider _updatePackageProvider;
|
private readonly IUpdatePackageProvider _updatePackageProvider;
|
||||||
|
private readonly IUpdateHistoryService _updateHistoryService;
|
||||||
|
|
||||||
public RecentUpdateProvider(IConfigFileProvider configFileProvider,
|
public RecentUpdateProvider(IConfigFileProvider configFileProvider,
|
||||||
IUpdatePackageProvider updatePackageProvider)
|
IUpdatePackageProvider updatePackageProvider,
|
||||||
|
IUpdateHistoryService updateHistoryService)
|
||||||
{
|
{
|
||||||
_configFileProvider = configFileProvider;
|
_configFileProvider = configFileProvider;
|
||||||
_updatePackageProvider = updatePackageProvider;
|
_updatePackageProvider = updatePackageProvider;
|
||||||
|
_updateHistoryService = updateHistoryService;
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<UpdatePackage> GetRecentUpdatePackages()
|
public List<UpdatePackage> GetRecentUpdatePackages()
|
||||||
{
|
{
|
||||||
var branch = _configFileProvider.Branch;
|
var branch = _configFileProvider.Branch;
|
||||||
return _updatePackageProvider.GetRecentUpdates(branch, BuildInfo.Version);
|
var version = BuildInfo.Version;
|
||||||
|
var prevVersion = _updateHistoryService.PreviouslyInstalled();
|
||||||
|
return _updatePackageProvider.GetRecentUpdates(branch, version, prevVersion);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ namespace NzbDrone.Core.Update
|
||||||
public interface IUpdatePackageProvider
|
public interface IUpdatePackageProvider
|
||||||
{
|
{
|
||||||
UpdatePackage GetLatestUpdate(string branch, Version currentVersion);
|
UpdatePackage GetLatestUpdate(string branch, Version currentVersion);
|
||||||
List<UpdatePackage> GetRecentUpdates(string branch, Version currentVersion);
|
List<UpdatePackage> GetRecentUpdates(string branch, Version currentVersion, Version previousVersion = null);
|
||||||
}
|
}
|
||||||
|
|
||||||
public class UpdatePackageProvider : IUpdatePackageProvider
|
public class UpdatePackageProvider : IUpdatePackageProvider
|
||||||
|
@ -50,7 +50,7 @@ namespace NzbDrone.Core.Update
|
||||||
return update.UpdatePackage;
|
return update.UpdatePackage;
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<UpdatePackage> GetRecentUpdates(string branch, Version currentVersion)
|
public List<UpdatePackage> GetRecentUpdates(string branch, Version currentVersion, Version previousVersion)
|
||||||
{
|
{
|
||||||
var request = _requestBuilder.Create()
|
var request = _requestBuilder.Create()
|
||||||
.Resource("/update/{branch}/changes")
|
.Resource("/update/{branch}/changes")
|
||||||
|
@ -59,6 +59,11 @@ namespace NzbDrone.Core.Update
|
||||||
.AddQueryParam("runtimeVer", _platformInfo.Version)
|
.AddQueryParam("runtimeVer", _platformInfo.Version)
|
||||||
.SetSegment("branch", branch);
|
.SetSegment("branch", branch);
|
||||||
|
|
||||||
|
if (previousVersion != null && previousVersion != currentVersion)
|
||||||
|
{
|
||||||
|
request.AddQueryParam("prevVersion", previousVersion);
|
||||||
|
}
|
||||||
|
|
||||||
if (_analyticsService.IsEnabled)
|
if (_analyticsService.IsEnabled)
|
||||||
{
|
{
|
||||||
// Send if the system is active so we know which versions to deprecate/ignore
|
// Send if the system is active so we know which versions to deprecate/ignore
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using NzbDrone.Common.EnvironmentInfo;
|
using NzbDrone.Common.EnvironmentInfo;
|
||||||
|
using NzbDrone.Common.Extensions;
|
||||||
using NzbDrone.Core.Update;
|
using NzbDrone.Core.Update;
|
||||||
|
using NzbDrone.Core.Update.History;
|
||||||
using Sonarr.Http;
|
using Sonarr.Http;
|
||||||
|
|
||||||
namespace Sonarr.Api.V3.Update
|
namespace Sonarr.Api.V3.Update
|
||||||
|
@ -9,10 +11,12 @@ namespace Sonarr.Api.V3.Update
|
||||||
public class UpdateModule : SonarrRestModule<UpdateResource>
|
public class UpdateModule : SonarrRestModule<UpdateResource>
|
||||||
{
|
{
|
||||||
private readonly IRecentUpdateProvider _recentUpdateProvider;
|
private readonly IRecentUpdateProvider _recentUpdateProvider;
|
||||||
|
private readonly IUpdateHistoryService _updateHistoryService;
|
||||||
|
|
||||||
public UpdateModule(IRecentUpdateProvider recentUpdateProvider)
|
public UpdateModule(IRecentUpdateProvider recentUpdateProvider, IUpdateHistoryService updateHistoryService)
|
||||||
{
|
{
|
||||||
_recentUpdateProvider = recentUpdateProvider;
|
_recentUpdateProvider = recentUpdateProvider;
|
||||||
|
_updateHistoryService = updateHistoryService;
|
||||||
GetResourceAll = GetRecentUpdates;
|
GetResourceAll = GetRecentUpdates;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -38,6 +42,18 @@ namespace Sonarr.Api.V3.Update
|
||||||
{
|
{
|
||||||
installed.Installed = true;
|
installed.Installed = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var installDates = _updateHistoryService.InstalledSince(resources.Last().ReleaseDate)
|
||||||
|
.DistinctBy(v => v.Version)
|
||||||
|
.ToDictionary(v => v.Version);
|
||||||
|
|
||||||
|
foreach (var resource in resources)
|
||||||
|
{
|
||||||
|
if (installDates.TryGetValue(resource.Version, out var installDate))
|
||||||
|
{
|
||||||
|
resource.InstalledOn = installDate.Date;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return resources;
|
return resources;
|
||||||
|
|
|
@ -17,6 +17,7 @@ namespace Sonarr.Api.V3.Update
|
||||||
public string FileName { get; set; }
|
public string FileName { get; set; }
|
||||||
public string Url { get; set; }
|
public string Url { get; set; }
|
||||||
public bool Installed { get; set; }
|
public bool Installed { get; set; }
|
||||||
|
public DateTime? InstalledOn { get; set; }
|
||||||
public bool Installable { get; set; }
|
public bool Installable { get; set; }
|
||||||
public bool Latest { get; set; }
|
public bool Latest { get; set; }
|
||||||
public UpdateChanges Changes { get; set; }
|
public UpdateChanges Changes { get; set; }
|
||||||
|
|
Loading…
Reference in New Issue