|
|
|
@ -26,7 +26,7 @@ using Microsoft.Extensions.Logging;
|
|
|
|
|
namespace Emby.Server.Implementations.Updates
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Manages all install, uninstall and update operations (both plugins and system).
|
|
|
|
|
/// Manages all install, uninstall, and update operations for the system and individual plugins.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public class InstallationManager : IInstallationManager
|
|
|
|
|
{
|
|
|
|
@ -36,7 +36,7 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
public const string PluginManifestUrlKey = "InstallationManager:PluginManifestUrl";
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The _logger.
|
|
|
|
|
/// The logger.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private readonly ILogger _logger;
|
|
|
|
|
private readonly IApplicationPaths _appPaths;
|
|
|
|
@ -112,10 +112,10 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
public event EventHandler<GenericEventArgs<IPlugin>> PluginUninstalled;
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
|
public event EventHandler<GenericEventArgs<(IPlugin, PackageVersionInfo)>> PluginUpdated;
|
|
|
|
|
public event EventHandler<GenericEventArgs<(IPlugin, VersionInfo)>> PluginUpdated;
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
|
public event EventHandler<GenericEventArgs<PackageVersionInfo>> PluginInstalled;
|
|
|
|
|
public event EventHandler<GenericEventArgs<VersionInfo>> PluginInstalled;
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
|
public IEnumerable<InstallationInfo> CompletedInstallations => _completedInstallationsInternal;
|
|
|
|
@ -183,61 +183,56 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
|
public IEnumerable<PackageVersionInfo> GetCompatibleVersions(
|
|
|
|
|
IEnumerable<PackageVersionInfo> availableVersions,
|
|
|
|
|
Version minVersion = null,
|
|
|
|
|
PackageVersionClass classification = PackageVersionClass.Release)
|
|
|
|
|
public IEnumerable<VersionInfo> GetCompatibleVersions(
|
|
|
|
|
IEnumerable<VersionInfo> availableVersions,
|
|
|
|
|
Version minVersion = null)
|
|
|
|
|
{
|
|
|
|
|
var appVer = _applicationHost.ApplicationVersion;
|
|
|
|
|
availableVersions = availableVersions
|
|
|
|
|
.Where(x => x.classification == classification
|
|
|
|
|
&& Version.Parse(x.requiredVersionStr) <= appVer);
|
|
|
|
|
.Where(x => Version.Parse(x.targetAbi) <= appVer);
|
|
|
|
|
|
|
|
|
|
if (minVersion != null)
|
|
|
|
|
{
|
|
|
|
|
availableVersions = availableVersions.Where(x => x.Version >= minVersion);
|
|
|
|
|
availableVersions = availableVersions.Where(x => x.version >= minVersion);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return availableVersions.OrderByDescending(x => x.Version);
|
|
|
|
|
return availableVersions.OrderByDescending(x => x.version);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
|
public IEnumerable<PackageVersionInfo> GetCompatibleVersions(
|
|
|
|
|
public IEnumerable<VersionInfo> GetCompatibleVersions(
|
|
|
|
|
IEnumerable<PackageInfo> availablePackages,
|
|
|
|
|
string name = null,
|
|
|
|
|
Guid guid = default,
|
|
|
|
|
Version minVersion = null,
|
|
|
|
|
PackageVersionClass classification = PackageVersionClass.Release)
|
|
|
|
|
Version minVersion = null)
|
|
|
|
|
{
|
|
|
|
|
var package = FilterPackages(availablePackages, name, guid).FirstOrDefault();
|
|
|
|
|
|
|
|
|
|
// Package not found.
|
|
|
|
|
// Package not found in repository
|
|
|
|
|
if (package == null)
|
|
|
|
|
{
|
|
|
|
|
return Enumerable.Empty<PackageVersionInfo>();
|
|
|
|
|
return Enumerable.Empty<VersionInfo>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return GetCompatibleVersions(
|
|
|
|
|
package.versions,
|
|
|
|
|
minVersion,
|
|
|
|
|
classification);
|
|
|
|
|
minVersion);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
|
public async Task<IEnumerable<PackageVersionInfo>> GetAvailablePluginUpdates(CancellationToken cancellationToken = default)
|
|
|
|
|
public async Task<IEnumerable<VersionInfo>> GetAvailablePluginUpdates(CancellationToken cancellationToken = default)
|
|
|
|
|
{
|
|
|
|
|
var catalog = await GetAvailablePackages(cancellationToken).ConfigureAwait(false);
|
|
|
|
|
return GetAvailablePluginUpdates(catalog);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private IEnumerable<PackageVersionInfo> GetAvailablePluginUpdates(IReadOnlyList<PackageInfo> pluginCatalog)
|
|
|
|
|
private IEnumerable<VersionInfo> GetAvailablePluginUpdates(IReadOnlyList<PackageInfo> pluginCatalog)
|
|
|
|
|
{
|
|
|
|
|
foreach (var plugin in _applicationHost.Plugins)
|
|
|
|
|
{
|
|
|
|
|
var compatibleversions = GetCompatibleVersions(pluginCatalog, plugin.Name, plugin.Id, plugin.Version, _applicationHost.SystemUpdateLevel);
|
|
|
|
|
var version = compatibleversions.FirstOrDefault(y => y.Version > plugin.Version);
|
|
|
|
|
if (version != null
|
|
|
|
|
&& !CompletedInstallations.Any(x => string.Equals(x.AssemblyGuid, version.guid, StringComparison.OrdinalIgnoreCase)))
|
|
|
|
|
var compatibleversions = GetCompatibleVersions(pluginCatalog, plugin.Name, plugin.Id, plugin.Version);
|
|
|
|
|
var version = compatibleversions.FirstOrDefault(y => y.version > plugin.Version);
|
|
|
|
|
if (version != null && !CompletedInstallations.Any(x => string.Equals(x.Guid, version.guid, StringComparison.OrdinalIgnoreCase)))
|
|
|
|
|
{
|
|
|
|
|
yield return version;
|
|
|
|
|
}
|
|
|
|
@ -245,7 +240,7 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
|
public async Task InstallPackage(PackageVersionInfo package, CancellationToken cancellationToken)
|
|
|
|
|
public async Task InstallPackage(VersionInfo package, CancellationToken cancellationToken)
|
|
|
|
|
{
|
|
|
|
|
if (package == null)
|
|
|
|
|
{
|
|
|
|
@ -254,11 +249,9 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
|
|
|
|
|
var installationInfo = new InstallationInfo
|
|
|
|
|
{
|
|
|
|
|
Id = Guid.NewGuid(),
|
|
|
|
|
Guid = package.guid,
|
|
|
|
|
Name = package.name,
|
|
|
|
|
AssemblyGuid = package.guid,
|
|
|
|
|
UpdateClass = package.classification,
|
|
|
|
|
Version = package.versionStr
|
|
|
|
|
Version = package.version.ToString()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var innerCancellationTokenSource = new CancellationTokenSource();
|
|
|
|
@ -276,7 +269,7 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
var installationEventArgs = new InstallationEventArgs
|
|
|
|
|
{
|
|
|
|
|
InstallationInfo = installationInfo,
|
|
|
|
|
PackageVersionInfo = package
|
|
|
|
|
VersionInfo = package
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
PackageInstalling?.Invoke(this, installationEventArgs);
|
|
|
|
@ -301,7 +294,7 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
_currentInstallations.Remove(tuple);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_logger.LogInformation("Package installation cancelled: {0} {1}", package.name, package.versionStr);
|
|
|
|
|
_logger.LogInformation("Package installation cancelled: {0} {1}", package.name, package.version);
|
|
|
|
|
|
|
|
|
|
PackageInstallationCancelled?.Invoke(this, installationEventArgs);
|
|
|
|
|
|
|
|
|
@ -337,7 +330,7 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
/// <param name="package">The package.</param>
|
|
|
|
|
/// <param name="cancellationToken">The cancellation token.</param>
|
|
|
|
|
/// <returns><see cref="Task" />.</returns>
|
|
|
|
|
private async Task InstallPackageInternal(PackageVersionInfo package, CancellationToken cancellationToken)
|
|
|
|
|
private async Task InstallPackageInternal(VersionInfo package, CancellationToken cancellationToken)
|
|
|
|
|
{
|
|
|
|
|
// Set last update time if we were installed before
|
|
|
|
|
IPlugin plugin = _applicationHost.Plugins.FirstOrDefault(p => string.Equals(p.Id.ToString(), package.guid, StringComparison.OrdinalIgnoreCase))
|
|
|
|
@ -349,26 +342,26 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
// Do plugin-specific processing
|
|
|
|
|
if (plugin == null)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogInformation("New plugin installed: {0} {1} {2}", package.name, package.versionStr ?? string.Empty, package.classification);
|
|
|
|
|
_logger.LogInformation("New plugin installed: {0} {1} {2}", package.name, package.version);
|
|
|
|
|
|
|
|
|
|
PluginInstalled?.Invoke(this, new GenericEventArgs<PackageVersionInfo>(package));
|
|
|
|
|
PluginInstalled?.Invoke(this, new GenericEventArgs<VersionInfo>(package));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_logger.LogInformation("Plugin updated: {0} {1} {2}", package.name, package.versionStr ?? string.Empty, package.classification);
|
|
|
|
|
_logger.LogInformation("Plugin updated: {0} {1} {2}", package.name, package.version);
|
|
|
|
|
|
|
|
|
|
PluginUpdated?.Invoke(this, new GenericEventArgs<(IPlugin, PackageVersionInfo)>((plugin, package)));
|
|
|
|
|
PluginUpdated?.Invoke(this, new GenericEventArgs<(IPlugin, VersionInfo)>((plugin, package)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_applicationHost.NotifyPendingRestart();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task PerformPackageInstallation(PackageVersionInfo package, CancellationToken cancellationToken)
|
|
|
|
|
private async Task PerformPackageInstallation(VersionInfo package, CancellationToken cancellationToken)
|
|
|
|
|
{
|
|
|
|
|
var extension = Path.GetExtension(package.targetFilename);
|
|
|
|
|
var extension = Path.GetExtension(package.filename);
|
|
|
|
|
if (!string.Equals(extension, ".zip", StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
{
|
|
|
|
|
_logger.LogError("Only zip packages are supported. {Filename} is not a zip archive.", package.targetFilename);
|
|
|
|
|
_logger.LogError("Only zip packages are supported. {Filename} is not a zip archive.", package.filename);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -415,7 +408,7 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Uninstalls a plugin
|
|
|
|
|
/// Uninstalls a plugin.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="plugin">The plugin.</param>
|
|
|
|
|
public void UninstallPlugin(IPlugin plugin)
|
|
|
|
@ -473,7 +466,7 @@ namespace Emby.Server.Implementations.Updates
|
|
|
|
|
{
|
|
|
|
|
lock (_currentInstallationsLock)
|
|
|
|
|
{
|
|
|
|
|
var install = _currentInstallations.Find(x => x.info.Id == id);
|
|
|
|
|
var install = _currentInstallations.Find(x => x.info.Guid == id.ToString());
|
|
|
|
|
if (install == default((InstallationInfo, CancellationTokenSource)))
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|