Fixed: Install Update UI should now report an error if the application folder is not writable instead of failing silently.
This commit is contained in:
parent
11803afc39
commit
3a938e18fa
|
@ -13,7 +13,7 @@ namespace NzbDrone.Core.Test.HealthCheck.Checks
|
||||||
{
|
{
|
||||||
private const string DRONE_FACTORY_FOLDER = @"C:\Test\Unsorted";
|
private const string DRONE_FACTORY_FOLDER = @"C:\Test\Unsorted";
|
||||||
|
|
||||||
private void GivenDroneFactoryFolder(bool exists = false)
|
private void GivenDroneFactoryFolder(bool exists = false, bool writable = true)
|
||||||
{
|
{
|
||||||
Mocker.GetMock<IConfigService>()
|
Mocker.GetMock<IConfigService>()
|
||||||
.SetupGet(s => s.DownloadedEpisodesFolder)
|
.SetupGet(s => s.DownloadedEpisodesFolder)
|
||||||
|
@ -22,6 +22,10 @@ namespace NzbDrone.Core.Test.HealthCheck.Checks
|
||||||
Mocker.GetMock<IDiskProvider>()
|
Mocker.GetMock<IDiskProvider>()
|
||||||
.Setup(s => s.FolderExists(DRONE_FACTORY_FOLDER))
|
.Setup(s => s.FolderExists(DRONE_FACTORY_FOLDER))
|
||||||
.Returns(exists);
|
.Returns(exists);
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Setup(s => s.FolderWritable(It.IsAny<String>()))
|
||||||
|
.Returns(exists && writable);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
|
@ -35,11 +39,7 @@ namespace NzbDrone.Core.Test.HealthCheck.Checks
|
||||||
[Test]
|
[Test]
|
||||||
public void should_return_error_when_unable_to_write_to_drone_factory_folder()
|
public void should_return_error_when_unable_to_write_to_drone_factory_folder()
|
||||||
{
|
{
|
||||||
GivenDroneFactoryFolder(true);
|
GivenDroneFactoryFolder(true, false);
|
||||||
|
|
||||||
Mocker.GetMock<IDiskProvider>()
|
|
||||||
.Setup(s => s.WriteAllText(It.IsAny<String>(), It.IsAny<String>()))
|
|
||||||
.Throws<Exception>();
|
|
||||||
|
|
||||||
Subject.Check().ShouldBeError();
|
Subject.Check().ShouldBeError();
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,9 +21,9 @@ namespace NzbDrone.Core.Test.HealthCheck.Checks
|
||||||
.Setup(s => s.StartUpFolder)
|
.Setup(s => s.StartUpFolder)
|
||||||
.Returns(@"C:\NzbDrone");
|
.Returns(@"C:\NzbDrone");
|
||||||
|
|
||||||
Mocker.GetMock<IDiskProvider>()
|
Mocker.GetMock<NzbDrone.Common.Disk.IDiskProvider>()
|
||||||
.Setup(s => s.WriteAllText(It.IsAny<String>(), It.IsAny<String>()))
|
.Setup(c => c.FolderWritable(Moq.It.IsAny<string>()))
|
||||||
.Throws<Exception>();
|
.Returns(false);
|
||||||
|
|
||||||
Subject.Check().ShouldBeError();
|
Subject.Check().ShouldBeError();
|
||||||
}
|
}
|
||||||
|
@ -41,9 +41,9 @@ namespace NzbDrone.Core.Test.HealthCheck.Checks
|
||||||
.Setup(s => s.StartUpFolder)
|
.Setup(s => s.StartUpFolder)
|
||||||
.Returns(@"/opt/nzbdrone");
|
.Returns(@"/opt/nzbdrone");
|
||||||
|
|
||||||
Mocker.GetMock<IDiskProvider>()
|
Mocker.GetMock<NzbDrone.Common.Disk.IDiskProvider>()
|
||||||
.Setup(s => s.WriteAllText(It.IsAny<String>(), It.IsAny<String>()))
|
.Setup(c => c.FolderWritable(Moq.It.IsAny<string>()))
|
||||||
.Throws<Exception>();
|
.Returns(false);
|
||||||
|
|
||||||
Subject.Check().ShouldBeError();
|
Subject.Check().ShouldBeError();
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ using NzbDrone.Common.Http;
|
||||||
using NzbDrone.Common.Model;
|
using NzbDrone.Common.Model;
|
||||||
using NzbDrone.Common.Processes;
|
using NzbDrone.Common.Processes;
|
||||||
using NzbDrone.Core.Configuration;
|
using NzbDrone.Core.Configuration;
|
||||||
|
using NzbDrone.Core.Exceptions;
|
||||||
using NzbDrone.Core.Test.Framework;
|
using NzbDrone.Core.Test.Framework;
|
||||||
using NzbDrone.Core.Update;
|
using NzbDrone.Core.Update;
|
||||||
using NzbDrone.Core.Update.Commands;
|
using NzbDrone.Core.Update.Commands;
|
||||||
|
@ -59,6 +60,10 @@ namespace NzbDrone.Core.Test.UpdateTests
|
||||||
Mocker.GetMock<IProcessProvider>().Setup(c => c.GetCurrentProcess()).Returns(new ProcessInfo { Id = 12 });
|
Mocker.GetMock<IProcessProvider>().Setup(c => c.GetCurrentProcess()).Returns(new ProcessInfo { Id = 12 });
|
||||||
Mocker.GetMock<IRuntimeInfo>().Setup(c => c.ExecutingApplication).Returns(@"C:\Test\NzbDrone.exe");
|
Mocker.GetMock<IRuntimeInfo>().Setup(c => c.ExecutingApplication).Returns(@"C:\Test\NzbDrone.exe");
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Setup(c => c.FolderWritable(It.IsAny<string>()))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
_sandboxFolder = Mocker.GetMock<IAppFolderInfo>().Object.GetUpdateSandboxFolder();
|
_sandboxFolder = Mocker.GetMock<IAppFolderInfo>().Object.GetUpdateSandboxFolder();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -259,6 +264,36 @@ namespace NzbDrone.Core.Test.UpdateTests
|
||||||
ExceptionVerification.ExpectedErrors(1);
|
ExceptionVerification.ExpectedErrors(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_log_error_when_startup_folder_is_not_writable()
|
||||||
|
{
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Setup(c => c.FolderWritable(It.IsAny<string>()))
|
||||||
|
.Returns(false);
|
||||||
|
|
||||||
|
var updateArchive = Path.Combine(_sandboxFolder, _updatePackage.FileName);
|
||||||
|
|
||||||
|
Subject.Execute(new ApplicationUpdateCommand());
|
||||||
|
|
||||||
|
Mocker.GetMock<IHttpClient>().Verify(c => c.DownloadFile(_updatePackage.Url, updateArchive), Times.Never());
|
||||||
|
ExceptionVerification.ExpectedErrors(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_throw_when_install_cannot_be_started()
|
||||||
|
{
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Setup(c => c.FolderWritable(It.IsAny<string>()))
|
||||||
|
.Returns(false);
|
||||||
|
|
||||||
|
var updateArchive = Path.Combine(_sandboxFolder, _updatePackage.FileName);
|
||||||
|
|
||||||
|
Assert.Throws<NzbDroneClientException>(() => Subject.Execute(new InstallUpdateCommand() { UpdatePackage = _updatePackage }));
|
||||||
|
|
||||||
|
Mocker.GetMock<IHttpClient>().Verify(c => c.DownloadFile(_updatePackage.Url, updateArchive), Times.Never());
|
||||||
|
ExceptionVerification.ExpectedErrors(1);
|
||||||
|
}
|
||||||
|
|
||||||
[TearDown]
|
[TearDown]
|
||||||
public void TearDown()
|
public void TearDown()
|
||||||
{
|
{
|
||||||
|
|
|
@ -98,20 +98,17 @@ namespace NzbDrone.Core.Download
|
||||||
{
|
{
|
||||||
return new NzbDroneValidationFailure(propertyName, "Folder does not exist")
|
return new NzbDroneValidationFailure(propertyName, "Folder does not exist")
|
||||||
{
|
{
|
||||||
DetailedDescription = "The folder you specified does not exist or is inaccessible. Please verify the folder permissions for the user account that is used to execute NzbDrone."
|
DetailedDescription = string.Format("The folder you specified does not exist or is inaccessible. Please verify the folder permissions for the user account '{0}', which is used to execute Sonarr.", Environment.UserName)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mustBeWritable)
|
if (mustBeWritable && !_diskProvider.FolderWritable(folder))
|
||||||
{
|
{
|
||||||
if (!_diskProvider.FolderWritable(folder))
|
_logger.Error("Folder '{0}' is not writable.", folder);
|
||||||
|
return new NzbDroneValidationFailure(propertyName, "Unable to write to folder")
|
||||||
{
|
{
|
||||||
_logger.Error("Folder '{0}' is not writable.", folder);
|
DetailedDescription = string.Format("The folder you specified is not writable. Please verify the folder permissions for the user account '{0}', which is used to execute Sonarr.", Environment.UserName)
|
||||||
return new NzbDroneValidationFailure(propertyName, "Unable to write to folder")
|
};
|
||||||
{
|
|
||||||
DetailedDescription = "The folder you specified is not writable. Please verify the folder permissions for the user account that is used to execute NzbDrone."
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
|
|
|
@ -31,7 +31,7 @@ namespace NzbDrone.Core.HealthCheck.Checks
|
||||||
{
|
{
|
||||||
if (!_diskProvider.FolderWritable(_appFolderInfo.StartUpFolder))
|
if (!_diskProvider.FolderWritable(_appFolderInfo.StartUpFolder))
|
||||||
{
|
{
|
||||||
return new HealthCheck(GetType(), HealthCheckResult.Error, "Unable to update, running from write-protected folder");
|
return new HealthCheck(GetType(), HealthCheckResult.Error, string.Format("Cannot install update because startup folder '{0}' is not writable by the user '{1}'.", _appFolderInfo.StartUpFolder, Environment.UserName));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -10,6 +10,7 @@ using NzbDrone.Common.Instrumentation.Extensions;
|
||||||
using NzbDrone.Common.Processes;
|
using NzbDrone.Common.Processes;
|
||||||
using NzbDrone.Core.Backup;
|
using NzbDrone.Core.Backup;
|
||||||
using NzbDrone.Core.Configuration;
|
using NzbDrone.Core.Configuration;
|
||||||
|
using NzbDrone.Core.Exceptions;
|
||||||
using NzbDrone.Core.Messaging.Commands;
|
using NzbDrone.Core.Messaging.Commands;
|
||||||
using NzbDrone.Core.Update.Commands;
|
using NzbDrone.Core.Update.Commands;
|
||||||
|
|
||||||
|
@ -61,12 +62,20 @@ namespace NzbDrone.Core.Update
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void InstallUpdate(UpdatePackage updatePackage)
|
private bool InstallUpdate(UpdatePackage updatePackage)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
EnsureAppDataSafety();
|
EnsureAppDataSafety();
|
||||||
|
|
||||||
|
if (OsInfo.IsWindows || _configFileProvider.UpdateMechanism != UpdateMechanism.Script)
|
||||||
|
{
|
||||||
|
if (!_diskProvider.FolderWritable(_appFolderInfo.StartUpFolder))
|
||||||
|
{
|
||||||
|
throw new ApplicationException(string.Format("Cannot install update because startup folder '{0}' is not writable by the user '{1}'.", _appFolderInfo.StartUpFolder, Environment.UserName));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var updateSandboxFolder = _appFolderInfo.GetUpdateSandboxFolder();
|
var updateSandboxFolder = _appFolderInfo.GetUpdateSandboxFolder();
|
||||||
|
|
||||||
var packageDestination = Path.Combine(updateSandboxFolder, updatePackage.FileName);
|
var packageDestination = Path.Combine(updateSandboxFolder, updatePackage.FileName);
|
||||||
|
@ -100,7 +109,7 @@ namespace NzbDrone.Core.Update
|
||||||
if (OsInfo.IsNotWindows && _configFileProvider.UpdateMechanism == UpdateMechanism.Script)
|
if (OsInfo.IsNotWindows && _configFileProvider.UpdateMechanism == UpdateMechanism.Script)
|
||||||
{
|
{
|
||||||
InstallUpdateWithScript(updateSandboxFolder);
|
InstallUpdateWithScript(updateSandboxFolder);
|
||||||
return;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
_logger.Info("Preparing client");
|
_logger.Info("Preparing client");
|
||||||
|
@ -111,10 +120,13 @@ namespace NzbDrone.Core.Update
|
||||||
_logger.ProgressInfo("NzbDrone will restart shortly.");
|
_logger.ProgressInfo("NzbDrone will restart shortly.");
|
||||||
|
|
||||||
_processProvider.Start(_appFolderInfo.GetUpdateClientExePath(), GetUpdaterArgs(updateSandboxFolder));
|
_processProvider.Start(_appFolderInfo.GetUpdateClientExePath(), GetUpdaterArgs(updateSandboxFolder));
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.ErrorException("Update process failed", ex);
|
_logger.ErrorException("Update process failed", ex);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -170,7 +182,12 @@ namespace NzbDrone.Core.Update
|
||||||
|
|
||||||
public void Execute(InstallUpdateCommand message)
|
public void Execute(InstallUpdateCommand message)
|
||||||
{
|
{
|
||||||
InstallUpdate(message.UpdatePackage);
|
var success = InstallUpdate(message.UpdatePackage);
|
||||||
|
|
||||||
|
if (!success)
|
||||||
|
{
|
||||||
|
throw new NzbDroneClientException(System.Net.HttpStatusCode.Conflict, "Failed to install update");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue