2017-07-28 11:05:42 +02:00
|
|
|
|
/*
|
2021-02-03 00:45:33 +01:00
|
|
|
|
* Copyright (c) 2021 ETH Zürich, Educational Development and Technology (LET)
|
2017-07-28 11:05:42 +02:00
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
|
|
|
|
|
2019-02-12 13:57:22 +01:00
|
|
|
|
using System;
|
2019-12-09 16:25:03 +01:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
2017-07-28 11:05:42 +02:00
|
|
|
|
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
2019-02-12 13:57:22 +01:00
|
|
|
|
using Moq;
|
2019-11-20 15:30:53 +01:00
|
|
|
|
using SafeExamBrowser.Applications.Contracts;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Browser.Contracts;
|
2019-09-06 08:13:27 +02:00
|
|
|
|
using SafeExamBrowser.Browser.Contracts.Events;
|
2019-12-09 16:25:03 +01:00
|
|
|
|
using SafeExamBrowser.Client.Operations.Events;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Communication.Contracts.Data;
|
|
|
|
|
using SafeExamBrowser.Communication.Contracts.Events;
|
|
|
|
|
using SafeExamBrowser.Communication.Contracts.Hosts;
|
|
|
|
|
using SafeExamBrowser.Communication.Contracts.Proxies;
|
|
|
|
|
using SafeExamBrowser.Configuration.Contracts;
|
|
|
|
|
using SafeExamBrowser.Configuration.Contracts.Cryptography;
|
|
|
|
|
using SafeExamBrowser.Core.Contracts.OperationModel;
|
|
|
|
|
using SafeExamBrowser.Core.Contracts.OperationModel.Events;
|
|
|
|
|
using SafeExamBrowser.I18n.Contracts;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2019-10-01 11:30:53 +02:00
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
2019-09-05 09:00:41 +02:00
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.Display;
|
2020-06-29 19:29:48 +02:00
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.System;
|
2021-07-22 14:21:06 +02:00
|
|
|
|
using SafeExamBrowser.Server.Contracts;
|
|
|
|
|
using SafeExamBrowser.Server.Contracts.Data;
|
2019-10-01 11:30:53 +02:00
|
|
|
|
using SafeExamBrowser.Settings;
|
2021-06-01 13:07:55 +02:00
|
|
|
|
using SafeExamBrowser.Settings.Monitoring;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts;
|
2020-01-22 16:08:57 +01:00
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts.FileSystemDialog;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts.MessageBox;
|
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts.Shell;
|
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts.Windows;
|
2019-10-11 15:46:15 +02:00
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts.Windows.Data;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts;
|
2017-07-28 11:05:42 +02:00
|
|
|
|
|
2018-01-17 08:26:44 +01:00
|
|
|
|
namespace SafeExamBrowser.Client.UnitTests
|
2017-07-28 11:05:42 +02:00
|
|
|
|
{
|
|
|
|
|
[TestClass]
|
2018-01-17 08:26:44 +01:00
|
|
|
|
public class ClientControllerTests
|
2017-07-28 11:05:42 +02:00
|
|
|
|
{
|
2019-02-12 13:57:22 +01:00
|
|
|
|
private AppConfig appConfig;
|
2019-03-06 16:10:00 +01:00
|
|
|
|
private Mock<IActionCenter> actionCenter;
|
2019-10-01 11:30:53 +02:00
|
|
|
|
private Mock<IApplicationMonitor> applicationMonitor;
|
2019-12-02 15:48:06 +01:00
|
|
|
|
private Mock<IBrowserApplication> browser;
|
2019-02-12 13:57:22 +01:00
|
|
|
|
private Mock<IClientHost> clientHost;
|
2019-10-01 16:24:10 +02:00
|
|
|
|
private ClientContext context;
|
2019-02-12 13:57:22 +01:00
|
|
|
|
private Mock<IDisplayMonitor> displayMonitor;
|
|
|
|
|
private Mock<IExplorerShell> explorerShell;
|
2020-01-22 16:08:57 +01:00
|
|
|
|
private Mock<IFileSystemDialog> fileSystemDialog;
|
2019-02-12 13:57:22 +01:00
|
|
|
|
private Mock<IHashAlgorithm> hashAlgorithm;
|
|
|
|
|
private Mock<ILogger> logger;
|
|
|
|
|
private Mock<IMessageBox> messageBox;
|
|
|
|
|
private Mock<IOperationSequence> operationSequence;
|
|
|
|
|
private Mock<IRuntimeProxy> runtimeProxy;
|
2021-07-22 14:21:06 +02:00
|
|
|
|
private Mock<IServerProxy> server;
|
2019-02-12 13:57:22 +01:00
|
|
|
|
private Guid sessionId;
|
2019-10-01 11:30:53 +02:00
|
|
|
|
private AppSettings settings;
|
2019-02-12 13:57:22 +01:00
|
|
|
|
private Mock<Action> shutdown;
|
2020-02-14 14:43:08 +01:00
|
|
|
|
private Mock<ISplashScreen> splashScreen;
|
2020-06-29 19:29:48 +02:00
|
|
|
|
private Mock<ISystemMonitor> systemMonitor;
|
2019-02-12 13:57:22 +01:00
|
|
|
|
private Mock<ITaskbar> taskbar;
|
|
|
|
|
private Mock<IText> text;
|
|
|
|
|
private Mock<IUserInterfaceFactory> uiFactory;
|
|
|
|
|
|
|
|
|
|
private ClientController sut;
|
|
|
|
|
|
|
|
|
|
[TestInitialize]
|
|
|
|
|
public void Initialize()
|
|
|
|
|
{
|
|
|
|
|
appConfig = new AppConfig();
|
2019-03-06 16:10:00 +01:00
|
|
|
|
actionCenter = new Mock<IActionCenter>();
|
2019-10-01 11:30:53 +02:00
|
|
|
|
applicationMonitor = new Mock<IApplicationMonitor>();
|
2019-12-02 15:48:06 +01:00
|
|
|
|
browser = new Mock<IBrowserApplication>();
|
2019-02-12 13:57:22 +01:00
|
|
|
|
clientHost = new Mock<IClientHost>();
|
2019-10-01 16:24:10 +02:00
|
|
|
|
context = new ClientContext();
|
2019-02-12 13:57:22 +01:00
|
|
|
|
displayMonitor = new Mock<IDisplayMonitor>();
|
|
|
|
|
explorerShell = new Mock<IExplorerShell>();
|
2020-01-22 16:08:57 +01:00
|
|
|
|
fileSystemDialog = new Mock<IFileSystemDialog>();
|
2019-02-12 13:57:22 +01:00
|
|
|
|
hashAlgorithm = new Mock<IHashAlgorithm>();
|
|
|
|
|
logger = new Mock<ILogger>();
|
|
|
|
|
messageBox = new Mock<IMessageBox>();
|
|
|
|
|
operationSequence = new Mock<IOperationSequence>();
|
|
|
|
|
runtimeProxy = new Mock<IRuntimeProxy>();
|
2021-07-22 14:21:06 +02:00
|
|
|
|
server = new Mock<IServerProxy>();
|
2019-02-12 13:57:22 +01:00
|
|
|
|
sessionId = Guid.NewGuid();
|
2019-10-01 11:30:53 +02:00
|
|
|
|
settings = new AppSettings();
|
2019-02-12 13:57:22 +01:00
|
|
|
|
shutdown = new Mock<Action>();
|
2020-02-14 14:43:08 +01:00
|
|
|
|
splashScreen = new Mock<ISplashScreen>();
|
2020-06-29 19:29:48 +02:00
|
|
|
|
systemMonitor = new Mock<ISystemMonitor>();
|
2019-02-12 13:57:22 +01:00
|
|
|
|
taskbar = new Mock<ITaskbar>();
|
|
|
|
|
text = new Mock<IText>();
|
|
|
|
|
uiFactory = new Mock<IUserInterfaceFactory>();
|
|
|
|
|
|
|
|
|
|
operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
runtimeProxy.Setup(r => r.InformClientReady()).Returns(new CommunicationResult(true));
|
|
|
|
|
uiFactory.Setup(u => u.CreateSplashScreen(It.IsAny<AppConfig>())).Returns(new Mock<ISplashScreen>().Object);
|
|
|
|
|
|
|
|
|
|
sut = new ClientController(
|
2019-03-06 16:10:00 +01:00
|
|
|
|
actionCenter.Object,
|
2019-10-01 11:30:53 +02:00
|
|
|
|
applicationMonitor.Object,
|
2019-10-01 16:24:10 +02:00
|
|
|
|
context,
|
2019-02-12 13:57:22 +01:00
|
|
|
|
displayMonitor.Object,
|
|
|
|
|
explorerShell.Object,
|
2020-01-22 16:08:57 +01:00
|
|
|
|
fileSystemDialog.Object,
|
2019-02-12 13:57:22 +01:00
|
|
|
|
hashAlgorithm.Object,
|
|
|
|
|
logger.Object,
|
|
|
|
|
messageBox.Object,
|
|
|
|
|
operationSequence.Object,
|
|
|
|
|
runtimeProxy.Object,
|
|
|
|
|
shutdown.Object,
|
2020-02-14 14:43:08 +01:00
|
|
|
|
splashScreen.Object,
|
2020-06-29 19:29:48 +02:00
|
|
|
|
systemMonitor.Object,
|
2019-02-12 13:57:22 +01:00
|
|
|
|
taskbar.Object,
|
|
|
|
|
text.Object,
|
2019-10-01 11:30:53 +02:00
|
|
|
|
uiFactory.Object);
|
2019-02-12 13:57:22 +01:00
|
|
|
|
|
2019-10-01 16:24:10 +02:00
|
|
|
|
context.AppConfig = appConfig;
|
2019-12-02 15:48:06 +01:00
|
|
|
|
context.Browser = browser.Object;
|
2019-10-01 16:24:10 +02:00
|
|
|
|
context.ClientHost = clientHost.Object;
|
2021-07-22 14:21:06 +02:00
|
|
|
|
context.Server = server.Object;
|
2019-10-01 16:24:10 +02:00
|
|
|
|
context.SessionId = sessionId;
|
|
|
|
|
context.Settings = settings;
|
2019-02-14 16:05:58 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-01 11:30:53 +02:00
|
|
|
|
[TestMethod]
|
2020-02-21 09:38:39 +01:00
|
|
|
|
public void ApplicationMonitor_MustCorrectlyHandleExplorerStartWithTaskbar()
|
2019-10-01 11:30:53 +02:00
|
|
|
|
{
|
2020-02-21 09:38:39 +01:00
|
|
|
|
var boundsActionCenter = 0;
|
|
|
|
|
var boundsTaskbar = 0;
|
|
|
|
|
var height = 30;
|
2019-10-01 11:30:53 +02:00
|
|
|
|
var order = 0;
|
|
|
|
|
var shell = 0;
|
|
|
|
|
var workingArea = 0;
|
|
|
|
|
|
2020-02-21 09:38:39 +01:00
|
|
|
|
settings.Taskbar.EnableTaskbar = true;
|
|
|
|
|
|
|
|
|
|
actionCenter.Setup(a => a.InitializeBounds()).Callback(() => boundsActionCenter = ++order);
|
2019-10-01 11:30:53 +02:00
|
|
|
|
explorerShell.Setup(e => e.Terminate()).Callback(() => shell = ++order);
|
2020-02-21 09:38:39 +01:00
|
|
|
|
displayMonitor.Setup(w => w.InitializePrimaryDisplay(It.Is<int>(h => h == height))).Callback(() => workingArea = ++order);
|
|
|
|
|
taskbar.Setup(t => t.GetAbsoluteHeight()).Returns(height);
|
|
|
|
|
taskbar.Setup(t => t.InitializeBounds()).Callback(() => boundsTaskbar = ++order);
|
2019-10-01 11:30:53 +02:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
2019-12-09 16:25:03 +01:00
|
|
|
|
applicationMonitor.Raise(a => a.ExplorerStarted += null);
|
2019-10-01 11:30:53 +02:00
|
|
|
|
|
2020-02-21 09:38:39 +01:00
|
|
|
|
actionCenter.Verify(a => a.InitializeBounds(), Times.Once);
|
2019-12-09 16:25:03 +01:00
|
|
|
|
explorerShell.Verify(e => e.Terminate(), Times.Once);
|
2020-02-21 09:38:39 +01:00
|
|
|
|
displayMonitor.Verify(d => d.InitializePrimaryDisplay(It.Is<int>(h => h == 0)), Times.Never);
|
|
|
|
|
displayMonitor.Verify(d => d.InitializePrimaryDisplay(It.Is<int>(h => h == height)), Times.Once);
|
2019-10-01 11:30:53 +02:00
|
|
|
|
taskbar.Verify(t => t.InitializeBounds(), Times.Once);
|
2020-02-21 09:38:39 +01:00
|
|
|
|
taskbar.Verify(t => t.GetAbsoluteHeight(), Times.Once);
|
2019-10-01 11:30:53 +02:00
|
|
|
|
|
|
|
|
|
Assert.IsTrue(shell == 1);
|
|
|
|
|
Assert.IsTrue(workingArea == 2);
|
2020-02-21 09:38:39 +01:00
|
|
|
|
Assert.IsTrue(boundsActionCenter == 3);
|
|
|
|
|
Assert.IsTrue(boundsTaskbar == 4);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void ApplicationMonitor_MustCorrectlyHandleExplorerStartWithoutTaskbar()
|
|
|
|
|
{
|
|
|
|
|
var boundsActionCenter = 0;
|
|
|
|
|
var boundsTaskbar = 0;
|
|
|
|
|
var height = 30;
|
|
|
|
|
var order = 0;
|
|
|
|
|
var shell = 0;
|
|
|
|
|
var workingArea = 0;
|
|
|
|
|
|
|
|
|
|
settings.Taskbar.EnableTaskbar = false;
|
|
|
|
|
|
|
|
|
|
actionCenter.Setup(a => a.InitializeBounds()).Callback(() => boundsActionCenter = ++order);
|
|
|
|
|
explorerShell.Setup(e => e.Terminate()).Callback(() => shell = ++order);
|
|
|
|
|
displayMonitor.Setup(w => w.InitializePrimaryDisplay(It.Is<int>(h => h == 0))).Callback(() => workingArea = ++order);
|
|
|
|
|
taskbar.Setup(t => t.GetAbsoluteHeight()).Returns(height);
|
|
|
|
|
taskbar.Setup(t => t.InitializeBounds()).Callback(() => boundsTaskbar = ++order);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
applicationMonitor.Raise(a => a.ExplorerStarted += null);
|
|
|
|
|
|
|
|
|
|
actionCenter.Verify(a => a.InitializeBounds(), Times.Once);
|
|
|
|
|
explorerShell.Verify(e => e.Terminate(), Times.Once);
|
|
|
|
|
displayMonitor.Verify(d => d.InitializePrimaryDisplay(It.Is<int>(h => h == 0)), Times.Once);
|
|
|
|
|
displayMonitor.Verify(d => d.InitializePrimaryDisplay(It.Is<int>(h => h == height)), Times.Never);
|
|
|
|
|
taskbar.Verify(t => t.InitializeBounds(), Times.Once);
|
|
|
|
|
taskbar.Verify(t => t.GetAbsoluteHeight(), Times.Never);
|
|
|
|
|
|
|
|
|
|
Assert.IsTrue(shell == 1);
|
|
|
|
|
Assert.IsTrue(workingArea == 2);
|
|
|
|
|
Assert.IsTrue(boundsActionCenter == 3);
|
|
|
|
|
Assert.IsTrue(boundsTaskbar == 4);
|
2019-10-01 11:30:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
2021-07-22 14:21:06 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void ApplicationMonitor_MustPermitApplicationIfChosenByUserAfterFailedTermination()
|
|
|
|
|
{
|
|
|
|
|
var lockScreen = new Mock<ILockScreen>();
|
|
|
|
|
var result = new LockScreenResult();
|
|
|
|
|
|
|
|
|
|
lockScreen.Setup(l => l.WaitForResult()).Returns(result);
|
|
|
|
|
runtimeProxy.Setup(p => p.RequestShutdown()).Returns(new CommunicationResult(true));
|
|
|
|
|
uiFactory
|
|
|
|
|
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
|
|
|
|
|
.Returns(lockScreen.Object)
|
|
|
|
|
.Callback<string, string, IEnumerable<LockScreenOption>>((m, t, o) => result.OptionId = o.First().Id);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
applicationMonitor.Raise(m => m.TerminationFailed += null, new List<RunningApplication>());
|
|
|
|
|
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Never);
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-09 16:25:03 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void ApplicationMonitor_MustRequestShutdownIfChosenByUserAfterFailedTermination()
|
|
|
|
|
{
|
|
|
|
|
var lockScreen = new Mock<ILockScreen>();
|
|
|
|
|
var result = new LockScreenResult();
|
|
|
|
|
|
|
|
|
|
lockScreen.Setup(l => l.WaitForResult()).Returns(result);
|
|
|
|
|
runtimeProxy.Setup(p => p.RequestShutdown()).Returns(new CommunicationResult(true));
|
|
|
|
|
uiFactory
|
|
|
|
|
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
|
|
|
|
|
.Returns(lockScreen.Object)
|
|
|
|
|
.Callback<string, string, IEnumerable<LockScreenOption>>((m, t, o) => result.OptionId = o.Last().Id);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
applicationMonitor.Raise(m => m.TerminationFailed += null, new List<RunningApplication>());
|
|
|
|
|
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void ApplicationMonitor_MustShowLockScreenIfTerminationFailed()
|
|
|
|
|
{
|
|
|
|
|
var activator1 = new Mock<IActivator>();
|
|
|
|
|
var activator2 = new Mock<IActivator>();
|
|
|
|
|
var activator3 = new Mock<IActivator>();
|
|
|
|
|
var lockScreen = new Mock<ILockScreen>();
|
|
|
|
|
var result = new LockScreenResult();
|
|
|
|
|
var order = 0;
|
|
|
|
|
var pause = 0;
|
|
|
|
|
var show = 0;
|
|
|
|
|
var wait = 0;
|
|
|
|
|
var close = 0;
|
|
|
|
|
var resume = 0;
|
|
|
|
|
|
|
|
|
|
activator1.Setup(a => a.Pause()).Callback(() => pause = ++order);
|
|
|
|
|
activator1.Setup(a => a.Resume()).Callback(() => resume = ++order);
|
|
|
|
|
context.Activators.Add(activator1.Object);
|
|
|
|
|
context.Activators.Add(activator2.Object);
|
|
|
|
|
context.Activators.Add(activator3.Object);
|
|
|
|
|
lockScreen.Setup(l => l.Show()).Callback(() => show = ++order);
|
|
|
|
|
lockScreen.Setup(l => l.WaitForResult()).Callback(() => wait = ++order).Returns(result);
|
|
|
|
|
lockScreen.Setup(l => l.Close()).Callback(() => close = ++order);
|
|
|
|
|
uiFactory
|
|
|
|
|
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
|
|
|
|
|
.Returns(lockScreen.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
applicationMonitor.Raise(m => m.TerminationFailed += null, new List<RunningApplication>());
|
|
|
|
|
|
|
|
|
|
activator1.Verify(a => a.Pause(), Times.Once);
|
|
|
|
|
activator1.Verify(a => a.Resume(), Times.Once);
|
|
|
|
|
activator2.Verify(a => a.Pause(), Times.Once);
|
|
|
|
|
activator2.Verify(a => a.Resume(), Times.Once);
|
|
|
|
|
activator3.Verify(a => a.Pause(), Times.Once);
|
|
|
|
|
activator3.Verify(a => a.Resume(), Times.Once);
|
|
|
|
|
lockScreen.Verify(l => l.Show(), Times.Once);
|
|
|
|
|
lockScreen.Verify(l => l.WaitForResult(), Times.Once);
|
|
|
|
|
lockScreen.Verify(l => l.Close(), Times.Once);
|
|
|
|
|
|
|
|
|
|
Assert.IsTrue(pause == 1);
|
|
|
|
|
Assert.IsTrue(show == 2);
|
|
|
|
|
Assert.IsTrue(wait == 3);
|
|
|
|
|
Assert.IsTrue(close == 4);
|
|
|
|
|
Assert.IsTrue(resume == 5);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void ApplicationMonitor_MustValidateQuitPasswordIfTerminationFailed()
|
|
|
|
|
{
|
|
|
|
|
var hash = "12345";
|
|
|
|
|
var lockScreen = new Mock<ILockScreen>();
|
|
|
|
|
var result = new LockScreenResult { Password = "test" };
|
|
|
|
|
var attempt = 0;
|
|
|
|
|
var correct = new Random().Next(1, 50);
|
|
|
|
|
var lockScreenResult = new Func<LockScreenResult>(() => ++attempt == correct ? result : new LockScreenResult());
|
|
|
|
|
|
2019-12-20 11:37:07 +01:00
|
|
|
|
context.Settings.Security.QuitPasswordHash = hash;
|
2019-12-09 16:25:03 +01:00
|
|
|
|
hashAlgorithm.Setup(a => a.GenerateHashFor(It.Is<string>(p => p == result.Password))).Returns(hash);
|
|
|
|
|
lockScreen.Setup(l => l.WaitForResult()).Returns(lockScreenResult);
|
|
|
|
|
uiFactory
|
|
|
|
|
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
|
|
|
|
|
.Returns(lockScreen.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
applicationMonitor.Raise(m => m.TerminationFailed += null, new List<RunningApplication>());
|
|
|
|
|
|
|
|
|
|
hashAlgorithm.Verify(a => a.GenerateHashFor(It.Is<string>(p => p == result.Password)), Times.Once);
|
|
|
|
|
hashAlgorithm.Verify(a => a.GenerateHashFor(It.Is<string>(p => p != result.Password)), Times.Exactly(attempt - 1));
|
|
|
|
|
messageBox.Verify(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.Is<IWindow>(w => w == lockScreen.Object)), Times.Exactly(attempt - 1));
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-22 14:21:06 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Browser_MustHandleSessionIdentifierDetection()
|
|
|
|
|
{
|
|
|
|
|
var counter = 0;
|
|
|
|
|
var identifier = "abc123";
|
|
|
|
|
|
|
|
|
|
settings.SessionMode = SessionMode.Server;
|
|
|
|
|
server.Setup(s => s.SendSessionIdentifier(It.IsAny<string>())).Returns(() => new ServerResponse(++counter == 3));
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
browser.Raise(b => b.SessionIdentifierDetected += null, identifier);
|
|
|
|
|
|
|
|
|
|
server.Verify(s => s.SendSessionIdentifier(It.Is<string>(id => id == identifier)), Times.Exactly(3));
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-19 15:02:40 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Browser_MustTerminateIfRequested()
|
|
|
|
|
{
|
|
|
|
|
runtimeProxy.Setup(p => p.RequestShutdown()).Returns(new CommunicationResult(true));
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
browser.Raise(b => b.TerminationRequested += null);
|
|
|
|
|
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-22 14:21:06 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustCorrectlyHandleExamSelection()
|
|
|
|
|
{
|
|
|
|
|
var args = new ExamSelectionRequestEventArgs
|
|
|
|
|
{
|
|
|
|
|
Exams = new List<(string id, string lms, string name, string url)> { ("", "", "", "") },
|
|
|
|
|
RequestId = Guid.NewGuid()
|
|
|
|
|
};
|
|
|
|
|
var dialog = new Mock<IExamSelectionDialog>();
|
|
|
|
|
|
|
|
|
|
dialog.Setup(d => d.Show(It.IsAny<IWindow>())).Returns(new ExamSelectionDialogResult { Success = true });
|
|
|
|
|
uiFactory.Setup(f => f.CreateExamSelectionDialog(It.IsAny<IEnumerable<Exam>>())).Returns(dialog.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
clientHost.Raise(c => c.ExamSelectionRequested += null, args);
|
|
|
|
|
|
|
|
|
|
runtimeProxy.Verify(p => p.SubmitExamSelectionResult(It.Is<Guid>(g => g == args.RequestId), true, null), Times.Once);
|
|
|
|
|
uiFactory.Verify(f => f.CreateExamSelectionDialog(It.IsAny<IEnumerable<Exam>>()), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-14 16:05:58 +01:00
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Communication_MustCorrectlyHandleMessageBoxRequest()
|
2019-02-14 16:05:58 +01:00
|
|
|
|
{
|
2019-02-15 09:06:21 +01:00
|
|
|
|
var args = new MessageBoxRequestEventArgs
|
|
|
|
|
{
|
2019-08-30 14:02:36 +02:00
|
|
|
|
Action = (int) MessageBoxAction.YesNo,
|
|
|
|
|
Icon = (int) MessageBoxIcon.Question,
|
2019-02-15 09:06:21 +01:00
|
|
|
|
Message = "Some question to be answered",
|
|
|
|
|
RequestId = Guid.NewGuid(),
|
|
|
|
|
Title = "A Title"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.Is<string>(s => s == args.Message),
|
2019-08-30 14:02:36 +02:00
|
|
|
|
It.Is<string>(s => s == args.Title),
|
|
|
|
|
It.Is<MessageBoxAction>(a => a == (MessageBoxAction) args.Action),
|
|
|
|
|
It.Is<MessageBoxIcon>(i => i == (MessageBoxIcon) args.Icon),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.No);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
sut.TryStart();
|
|
|
|
|
clientHost.Raise(c => c.MessageBoxRequested += null, args);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
runtimeProxy.Verify(p => p.SubmitMessageBoxResult(
|
2019-08-30 14:02:36 +02:00
|
|
|
|
It.Is<Guid>(g => g == args.RequestId),
|
|
|
|
|
It.Is<int>(r => r == (int) MessageBoxResult.No)), Times.Once);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Communication_MustCorrectlyHandlePasswordRequest()
|
2019-02-14 16:05:58 +01:00
|
|
|
|
{
|
2019-02-15 09:06:21 +01:00
|
|
|
|
var args = new PasswordRequestEventArgs
|
|
|
|
|
{
|
|
|
|
|
Purpose = PasswordRequestPurpose.LocalSettings,
|
|
|
|
|
RequestId = Guid.NewGuid()
|
|
|
|
|
};
|
|
|
|
|
var dialog = new Mock<IPasswordDialog>();
|
2019-10-11 15:46:15 +02:00
|
|
|
|
var result = new PasswordDialogResult { Password = "blubb", Success = true };
|
2019-02-15 09:06:21 +01:00
|
|
|
|
|
2019-10-11 15:46:15 +02:00
|
|
|
|
dialog.Setup(d => d.Show(It.IsAny<IWindow>())).Returns(result);
|
2019-02-15 09:06:21 +01:00
|
|
|
|
uiFactory.Setup(f => f.CreatePasswordDialog(It.IsAny<string>(), It.IsAny<string>())).Returns(dialog.Object);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
sut.TryStart();
|
|
|
|
|
clientHost.Raise(c => c.PasswordRequested += null, args);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
runtimeProxy.Verify(p => p.SubmitPassword(
|
|
|
|
|
It.Is<Guid>(g => g == args.RequestId),
|
2019-10-11 15:46:15 +02:00
|
|
|
|
It.Is<bool>(b => b == result.Success),
|
|
|
|
|
It.Is<string>(s => s == result.Password)), Times.Once);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-10 16:47:50 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustCorrectlyHandleAbortedReconfiguration()
|
|
|
|
|
{
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
clientHost.Raise(c => c.ReconfigurationAborted += null);
|
|
|
|
|
|
2020-02-14 14:43:08 +01:00
|
|
|
|
splashScreen.Verify(s => s.Hide(), Times.AtLeastOnce);
|
2020-02-10 16:47:50 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-14 16:05:58 +01:00
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Communication_MustInformUserAboutDeniedReconfiguration()
|
2019-02-14 16:05:58 +01:00
|
|
|
|
{
|
2019-02-15 09:06:21 +01:00
|
|
|
|
var args = new ReconfigurationEventArgs
|
|
|
|
|
{
|
|
|
|
|
ConfigurationPath = @"C:\Some\File\Path.seb"
|
|
|
|
|
};
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
sut.TryStart();
|
|
|
|
|
clientHost.Raise(c => c.ReconfigurationDenied += null, args);
|
|
|
|
|
|
|
|
|
|
messageBox.Verify(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>()), Times.Once);
|
|
|
|
|
}
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
2021-07-22 14:21:06 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustCorrectlyHandleServerCommunicationFailure()
|
|
|
|
|
{
|
|
|
|
|
var args = new ServerFailureActionRequestEventArgs { RequestId = Guid.NewGuid() };
|
|
|
|
|
var dialog = new Mock<IServerFailureDialog>();
|
|
|
|
|
|
|
|
|
|
dialog.Setup(d => d.Show(It.IsAny<IWindow>())).Returns(new ServerFailureDialogResult());
|
|
|
|
|
uiFactory.Setup(f => f.CreateServerFailureDialog(It.IsAny<string>(), It.IsAny<bool>())).Returns(dialog.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
clientHost.Raise(c => c.ServerFailureActionRequested += null, args);
|
|
|
|
|
|
|
|
|
|
runtimeProxy.Verify(r => r.SubmitServerFailureActionResult(It.Is<Guid>(g => g == args.RequestId), It.IsAny<bool>(), It.IsAny<bool>(), It.IsAny<bool>()), Times.Once);
|
|
|
|
|
uiFactory.Verify(f => f.CreateServerFailureDialog(It.IsAny<string>(), It.IsAny<bool>()), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustCorrectlyInitiateShutdown()
|
|
|
|
|
{
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
clientHost.Raise(c => c.Shutdown += null);
|
|
|
|
|
|
|
|
|
|
shutdown.Verify(s => s(), Times.Once);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Communication_MustShutdownOnLostConnection()
|
2019-02-14 16:05:58 +01:00
|
|
|
|
{
|
2019-02-15 09:06:21 +01:00
|
|
|
|
sut.TryStart();
|
|
|
|
|
runtimeProxy.Raise(p => p.ConnectionLost += null);
|
|
|
|
|
|
|
|
|
|
messageBox.Verify(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>()), Times.Once);
|
|
|
|
|
shutdown.Verify(s => s(), Times.Once);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2020-02-21 09:38:39 +01:00
|
|
|
|
public void DisplayMonitor_MustCorrectlyHandleDisplayChangeWithTaskbar()
|
|
|
|
|
{
|
|
|
|
|
var boundsActionCenter = 0;
|
|
|
|
|
var boundsTaskbar = 0;
|
|
|
|
|
var height = 25;
|
|
|
|
|
var order = 0;
|
|
|
|
|
var workingArea = 0;
|
|
|
|
|
|
|
|
|
|
settings.Taskbar.EnableTaskbar = true;
|
|
|
|
|
|
|
|
|
|
actionCenter.Setup(t => t.InitializeBounds()).Callback(() => boundsActionCenter = ++order);
|
2021-06-01 13:07:55 +02:00
|
|
|
|
displayMonitor.Setup(m => m.InitializePrimaryDisplay(It.Is<int>(h => h == height))).Callback(() => workingArea = ++order);
|
2021-06-29 17:34:05 +02:00
|
|
|
|
displayMonitor.Setup(m => m.ValidateConfiguration(It.IsAny<DisplaySettings>())).Returns(new ValidationResult { IsAllowed = true });
|
2020-02-21 09:38:39 +01:00
|
|
|
|
taskbar.Setup(t => t.GetAbsoluteHeight()).Returns(height);
|
|
|
|
|
taskbar.Setup(t => t.InitializeBounds()).Callback(() => boundsTaskbar = ++order);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
displayMonitor.Raise(d => d.DisplayChanged += null);
|
|
|
|
|
|
|
|
|
|
actionCenter.Verify(a => a.InitializeBounds(), Times.Once);
|
|
|
|
|
displayMonitor.Verify(d => d.InitializePrimaryDisplay(It.Is<int>(h => h == 0)), Times.Never);
|
|
|
|
|
displayMonitor.Verify(d => d.InitializePrimaryDisplay(It.Is<int>(h => h == height)), Times.Once);
|
|
|
|
|
taskbar.Verify(t => t.GetAbsoluteHeight(), Times.Once);
|
|
|
|
|
taskbar.Verify(t => t.InitializeBounds(), Times.Once);
|
|
|
|
|
|
|
|
|
|
Assert.IsTrue(workingArea == 1);
|
|
|
|
|
Assert.IsTrue(boundsActionCenter == 2);
|
|
|
|
|
Assert.IsTrue(boundsTaskbar == 3);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void DisplayMonitor_MustCorrectlyHandleDisplayChangeWithoutTaskbar()
|
2019-02-14 16:05:58 +01:00
|
|
|
|
{
|
2020-02-21 09:38:39 +01:00
|
|
|
|
var boundsActionCenter = 0;
|
|
|
|
|
var boundsTaskbar = 0;
|
|
|
|
|
var height = 25;
|
2019-02-15 09:06:21 +01:00
|
|
|
|
var order = 0;
|
|
|
|
|
var workingArea = 0;
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
2020-02-21 09:38:39 +01:00
|
|
|
|
settings.Taskbar.EnableTaskbar = false;
|
|
|
|
|
|
|
|
|
|
actionCenter.Setup(t => t.InitializeBounds()).Callback(() => boundsActionCenter = ++order);
|
|
|
|
|
displayMonitor.Setup(w => w.InitializePrimaryDisplay(It.Is<int>(h => h == 0))).Callback(() => workingArea = ++order);
|
2021-06-29 17:34:05 +02:00
|
|
|
|
displayMonitor.Setup(m => m.ValidateConfiguration(It.IsAny<DisplaySettings>())).Returns(new ValidationResult { IsAllowed = true });
|
2020-02-21 09:38:39 +01:00
|
|
|
|
taskbar.Setup(t => t.GetAbsoluteHeight()).Returns(height);
|
|
|
|
|
taskbar.Setup(t => t.InitializeBounds()).Callback(() => boundsTaskbar = ++order);
|
2019-02-15 09:06:21 +01:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
displayMonitor.Raise(d => d.DisplayChanged += null);
|
|
|
|
|
|
2020-02-21 09:38:39 +01:00
|
|
|
|
actionCenter.Verify(a => a.InitializeBounds(), Times.Once);
|
|
|
|
|
displayMonitor.Verify(d => d.InitializePrimaryDisplay(It.Is<int>(h => h == 0)), Times.Once);
|
|
|
|
|
displayMonitor.Verify(d => d.InitializePrimaryDisplay(It.Is<int>(h => h == height)), Times.Never);
|
|
|
|
|
taskbar.Verify(t => t.GetAbsoluteHeight(), Times.Never);
|
|
|
|
|
taskbar.Verify(t => t.InitializeBounds(), Times.Once);
|
2019-02-15 09:06:21 +01:00
|
|
|
|
|
|
|
|
|
Assert.IsTrue(workingArea == 1);
|
2020-02-21 09:38:39 +01:00
|
|
|
|
Assert.IsTrue(boundsActionCenter == 2);
|
|
|
|
|
Assert.IsTrue(boundsTaskbar == 3);
|
2019-02-15 09:06:21 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-07-22 14:21:06 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void DisplayMonitor_MustShowLockScreenOnDisplayChange()
|
|
|
|
|
{
|
|
|
|
|
var lockScreen = new Mock<ILockScreen>();
|
|
|
|
|
|
|
|
|
|
displayMonitor.Setup(m => m.ValidateConfiguration(It.IsAny<DisplaySettings>())).Returns(new ValidationResult { IsAllowed = false });
|
|
|
|
|
lockScreen.Setup(l => l.WaitForResult()).Returns(new LockScreenResult());
|
|
|
|
|
uiFactory.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>())).Returns(lockScreen.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
displayMonitor.Raise(d => d.DisplayChanged += null);
|
|
|
|
|
|
|
|
|
|
lockScreen.Verify(l => l.Show(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-09 16:25:03 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Operations_MustAskForAutomaticApplicationTermination()
|
|
|
|
|
{
|
|
|
|
|
var args = new ApplicationTerminationEventArgs(Enumerable.Empty<RunningApplication>());
|
|
|
|
|
|
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.Yes);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
operationSequence.Raise(s => s.ActionRequired += null, args);
|
|
|
|
|
|
|
|
|
|
Assert.IsTrue(args.TerminateProcesses);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Operations_MustAbortAskingForAutomaticApplicationTermination()
|
|
|
|
|
{
|
|
|
|
|
var args = new ApplicationTerminationEventArgs(Enumerable.Empty<RunningApplication>());
|
|
|
|
|
|
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.No);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
operationSequence.Raise(s => s.ActionRequired += null, args);
|
|
|
|
|
|
|
|
|
|
Assert.IsFalse(args.TerminateProcesses);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Operations_MustAskForApplicationPath()
|
|
|
|
|
{
|
|
|
|
|
var args = new ApplicationNotFoundEventArgs(default(string), default(string));
|
2020-01-22 16:08:57 +01:00
|
|
|
|
var result = new FileSystemDialogResult { FullPath = @"C:\Some\random\path\", Success = true };
|
2019-12-09 16:25:03 +01:00
|
|
|
|
|
2020-01-22 16:08:57 +01:00
|
|
|
|
fileSystemDialog.Setup(d => d.Show(
|
|
|
|
|
It.IsAny<FileSystemElement>(),
|
|
|
|
|
It.IsAny<FileSystemOperation>(),
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<string>(),
|
2020-08-05 22:55:38 +02:00
|
|
|
|
It.IsAny<IWindow>(),
|
|
|
|
|
It.IsAny<bool>())).Returns(result);
|
2019-12-09 16:25:03 +01:00
|
|
|
|
text.SetReturnsDefault(string.Empty);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
operationSequence.Raise(s => s.ActionRequired += null, args);
|
|
|
|
|
|
2020-01-22 16:08:57 +01:00
|
|
|
|
Assert.AreEqual(result.FullPath, args.CustomPath);
|
2019-12-09 16:25:03 +01:00
|
|
|
|
Assert.IsTrue(args.Success);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Operations_MustAbortAskingForApplicationPath()
|
|
|
|
|
{
|
|
|
|
|
var args = new ApplicationNotFoundEventArgs(default(string), default(string));
|
2020-01-22 16:08:57 +01:00
|
|
|
|
var result = new FileSystemDialogResult { Success = false };
|
2019-12-09 16:25:03 +01:00
|
|
|
|
|
2020-01-22 16:08:57 +01:00
|
|
|
|
fileSystemDialog.Setup(d => d.Show(
|
|
|
|
|
It.IsAny<FileSystemElement>(),
|
|
|
|
|
It.IsAny<FileSystemOperation>(),
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<string>(),
|
2020-08-05 22:55:38 +02:00
|
|
|
|
It.IsAny<IWindow>(),
|
|
|
|
|
It.IsAny<bool>())).Returns(result);
|
2019-12-09 16:25:03 +01:00
|
|
|
|
text.SetReturnsDefault(string.Empty);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
operationSequence.Raise(s => s.ActionRequired += null, args);
|
|
|
|
|
|
|
|
|
|
Assert.IsNull(args.CustomPath);
|
|
|
|
|
Assert.IsFalse(args.Success);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Operations_MustInformAboutFailedApplicationInitialization()
|
|
|
|
|
{
|
|
|
|
|
var args = new ApplicationInitializationFailedEventArgs(default(string), default(string), FactoryResult.NotFound);
|
|
|
|
|
|
|
|
|
|
text.SetReturnsDefault(string.Empty);
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
operationSequence.Raise(s => s.ActionRequired += null, args);
|
|
|
|
|
|
|
|
|
|
messageBox.Verify(m => m.Show(
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>()), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Operations_MustInformAboutFailedApplicationTermination()
|
|
|
|
|
{
|
|
|
|
|
var args = new ApplicationTerminationFailedEventArgs(Enumerable.Empty<RunningApplication>());
|
|
|
|
|
|
|
|
|
|
text.SetReturnsDefault(string.Empty);
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
operationSequence.Raise(s => s.ActionRequired += null, args);
|
|
|
|
|
|
|
|
|
|
messageBox.Verify(m => m.Show(
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<string>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>()), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Operations_MustUpdateProgress()
|
|
|
|
|
{
|
|
|
|
|
var args = new ProgressChangedEventArgs
|
|
|
|
|
{
|
|
|
|
|
CurrentValue = 23,
|
|
|
|
|
IsIndeterminate = true,
|
|
|
|
|
MaxValue = 150,
|
|
|
|
|
Progress = true,
|
|
|
|
|
Regress = true
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
operationSequence.Raise(o => o.ProgressChanged += null, args);
|
|
|
|
|
|
|
|
|
|
splashScreen.Verify(s => s.SetValue(It.Is<int>(i => i == args.CurrentValue)), Times.Once);
|
|
|
|
|
splashScreen.Verify(s => s.SetIndeterminate(), Times.Once);
|
|
|
|
|
splashScreen.Verify(s => s.SetMaxValue(It.Is<int>(i => i == args.MaxValue)), Times.Once);
|
|
|
|
|
splashScreen.Verify(s => s.Progress(), Times.Once);
|
|
|
|
|
splashScreen.Verify(s => s.Regress(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Operations_MustUpdateStatus()
|
|
|
|
|
{
|
|
|
|
|
var key = TextKey.OperationStatus_EmptyClipboard;
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
operationSequence.Raise(o => o.StatusChanged += null, key);
|
|
|
|
|
|
|
|
|
|
splashScreen.Verify(s => s.UpdateStatus(It.Is<TextKey>(k => k == key), It.IsAny<bool>()), Times.Once);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
[TestMethod]
|
2020-09-10 12:35:58 +02:00
|
|
|
|
public void Reconfiguration_MustAllowIfNoQuitPasswordSet()
|
2019-02-14 16:05:58 +01:00
|
|
|
|
{
|
2020-09-10 12:35:58 +02:00
|
|
|
|
var args = new DownloadEventArgs();
|
|
|
|
|
|
|
|
|
|
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Setup(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>())).Returns(new CommunicationResult(true));
|
2020-09-10 12:35:58 +02:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
|
|
|
|
|
args.Callback(true, string.Empty);
|
|
|
|
|
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Once);
|
2020-09-10 12:35:58 +02:00
|
|
|
|
Assert.IsTrue(args.AllowDownload);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2021-04-23 15:34:12 +02:00
|
|
|
|
public void Reconfiguration_MustNotAllowWithQuitPasswordAndNoUrl()
|
2020-09-10 12:35:58 +02:00
|
|
|
|
{
|
|
|
|
|
var args = new DownloadEventArgs();
|
|
|
|
|
|
|
|
|
|
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
|
|
|
|
|
settings.Security.AllowReconfiguration = true;
|
|
|
|
|
settings.Security.QuitPasswordHash = "abc123";
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Setup(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>())).Returns(new CommunicationResult(true));
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
2020-09-10 12:35:58 +02:00
|
|
|
|
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
|
|
|
|
|
|
2021-04-23 15:34:12 +02:00
|
|
|
|
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
|
|
|
|
|
Assert.IsFalse(args.AllowDownload);
|
2020-09-10 12:35:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Reconfiguration_MustAllowIfUrlMatches()
|
|
|
|
|
{
|
|
|
|
|
var args = new DownloadEventArgs { Url = "sebs://www.somehost.org/some/path/some_configuration.seb?query=123" };
|
|
|
|
|
|
|
|
|
|
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
|
|
|
|
|
settings.Security.AllowReconfiguration = true;
|
|
|
|
|
settings.Security.QuitPasswordHash = "abc123";
|
|
|
|
|
settings.Security.ReconfigurationUrl = "sebs://www.somehost.org/some/path/*.seb?query=123";
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Setup(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>())).Returns(new CommunicationResult(true));
|
2020-09-10 12:35:58 +02:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
|
|
|
|
|
args.Callback(true, string.Empty);
|
|
|
|
|
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Once);
|
2020-09-10 12:35:58 +02:00
|
|
|
|
Assert.IsTrue(args.AllowDownload);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Reconfiguration_MustDenyIfNotAllowed()
|
|
|
|
|
{
|
|
|
|
|
var args = new DownloadEventArgs();
|
|
|
|
|
|
|
|
|
|
settings.Security.AllowReconfiguration = false;
|
|
|
|
|
settings.Security.QuitPasswordHash = "abc123";
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
|
|
|
|
|
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
|
2020-09-10 12:35:58 +02:00
|
|
|
|
Assert.IsFalse(args.AllowDownload);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Reconfiguration_MustDenyIfUrlDoesNotMatch()
|
|
|
|
|
{
|
|
|
|
|
var args = new DownloadEventArgs { Url = "sebs://www.somehost.org/some/path/some_configuration.seb?query=123" };
|
|
|
|
|
|
|
|
|
|
settings.Security.AllowReconfiguration = false;
|
|
|
|
|
settings.Security.QuitPasswordHash = "abc123";
|
|
|
|
|
settings.Security.ReconfigurationUrl = "sebs://www.somehost.org/some/path/other_configuration.seb?query=123";
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
|
|
|
|
|
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
|
2020-09-10 12:35:58 +02:00
|
|
|
|
Assert.IsFalse(args.AllowDownload);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Reconfiguration_MustCorrectlyHandleDownload()
|
2019-02-14 16:05:58 +01:00
|
|
|
|
{
|
|
|
|
|
var downloadPath = @"C:\Folder\Does\Not\Exist\filepath.seb";
|
2020-09-24 12:55:20 +02:00
|
|
|
|
var downloadUrl = @"https://www.host.abc/someresource.seb";
|
2019-02-14 16:05:58 +01:00
|
|
|
|
var filename = "filepath.seb";
|
|
|
|
|
var args = new DownloadEventArgs();
|
|
|
|
|
|
2020-01-29 10:07:28 +01:00
|
|
|
|
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
|
2020-02-25 10:41:55 +01:00
|
|
|
|
settings.Security.AllowReconfiguration = true;
|
2019-02-15 09:06:21 +01:00
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.Yes);
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Setup(r => r.RequestReconfiguration(
|
|
|
|
|
It.Is<string>(p => p == downloadPath),
|
|
|
|
|
It.Is<string>(u => u == downloadUrl))).Returns(new CommunicationResult(true));
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
2019-12-02 15:48:06 +01:00
|
|
|
|
browser.Raise(b => b.ConfigurationDownloadRequested += null, filename, args);
|
2020-09-24 12:55:20 +02:00
|
|
|
|
args.Callback(true, downloadUrl, downloadPath);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Verify(r => r.RequestReconfiguration(It.Is<string>(p => p == downloadPath), It.Is<string>(u => u == downloadUrl)), Times.Once);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
|
|
|
|
Assert.AreEqual(downloadPath, args.DownloadPath);
|
|
|
|
|
Assert.IsTrue(args.AllowDownload);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Reconfiguration_MustCorrectlyHandleFailedDownload()
|
2019-02-14 16:05:58 +01:00
|
|
|
|
{
|
|
|
|
|
var downloadPath = @"C:\Folder\Does\Not\Exist\filepath.seb";
|
2020-09-24 12:55:20 +02:00
|
|
|
|
var downloadUrl = @"https://www.host.abc/someresource.seb";
|
2019-02-14 16:05:58 +01:00
|
|
|
|
var filename = "filepath.seb";
|
|
|
|
|
var args = new DownloadEventArgs();
|
|
|
|
|
|
2020-01-29 10:07:28 +01:00
|
|
|
|
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
|
2020-02-25 10:41:55 +01:00
|
|
|
|
settings.Security.AllowReconfiguration = true;
|
2019-02-15 09:06:21 +01:00
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.Yes);
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Setup(r => r.RequestReconfiguration(
|
|
|
|
|
It.Is<string>(p => p == downloadPath),
|
|
|
|
|
It.Is<string>(u => u == downloadUrl))).Returns(new CommunicationResult(true));
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
2019-12-02 15:48:06 +01:00
|
|
|
|
browser.Raise(b => b.ConfigurationDownloadRequested += null, filename, args);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
args.Callback(false, downloadPath);
|
|
|
|
|
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Reconfiguration_MustCorrectlyHandleFailedRequest()
|
2019-02-14 16:05:58 +01:00
|
|
|
|
{
|
|
|
|
|
var downloadPath = @"C:\Folder\Does\Not\Exist\filepath.seb";
|
2020-09-24 12:55:20 +02:00
|
|
|
|
var downloadUrl = @"https://www.host.abc/someresource.seb";
|
2019-02-14 16:05:58 +01:00
|
|
|
|
var filename = "filepath.seb";
|
|
|
|
|
var args = new DownloadEventArgs();
|
|
|
|
|
|
2020-01-29 10:07:28 +01:00
|
|
|
|
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
|
2020-02-25 10:41:55 +01:00
|
|
|
|
settings.Security.AllowReconfiguration = true;
|
2019-02-15 09:06:21 +01:00
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.Yes);
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Setup(r => r.RequestReconfiguration(
|
|
|
|
|
It.Is<string>(p => p == downloadPath),
|
|
|
|
|
It.Is<string>(u => u == downloadUrl))).Returns(new CommunicationResult(false));
|
2019-02-12 13:57:22 +01:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
2019-12-02 15:48:06 +01:00
|
|
|
|
browser.Raise(b => b.ConfigurationDownloadRequested += null, filename, args);
|
2020-09-24 12:55:20 +02:00
|
|
|
|
args.Callback(true, downloadUrl, downloadPath);
|
2019-02-14 16:05:58 +01:00
|
|
|
|
|
2020-09-24 12:55:20 +02:00
|
|
|
|
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Once);
|
2019-02-15 09:06:21 +01:00
|
|
|
|
messageBox.Verify(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.Is<MessageBoxIcon>(i => i == MessageBoxIcon.Error),
|
|
|
|
|
It.IsAny<IWindow>()), Times.Once);
|
2019-02-12 13:57:22 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-07-22 14:21:06 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Server_MustInitiateShutdownOnEvent()
|
|
|
|
|
{
|
|
|
|
|
runtimeProxy.Setup(r => r.RequestShutdown()).Returns(new CommunicationResult(true));
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
server.Raise(s => s.TerminationRequested += null);
|
|
|
|
|
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-12 13:57:22 +01:00
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Shutdown_MustAskUserToConfirm()
|
2019-02-12 13:57:22 +01:00
|
|
|
|
{
|
2019-02-15 09:06:21 +01:00
|
|
|
|
var args = new System.ComponentModel.CancelEventArgs();
|
2017-07-28 11:05:42 +02:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.Yes);
|
|
|
|
|
runtimeProxy.Setup(r => r.RequestShutdown()).Returns(new CommunicationResult(true));
|
2019-02-12 13:57:22 +01:00
|
|
|
|
|
2019-02-14 16:05:58 +01:00
|
|
|
|
sut.TryStart();
|
2019-02-15 09:06:21 +01:00
|
|
|
|
taskbar.Raise(t => t.QuitButtonClicked += null, args as object);
|
2019-02-12 13:57:22 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once);
|
|
|
|
|
Assert.IsFalse(args.Cancel);
|
|
|
|
|
}
|
2019-02-12 13:57:22 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Shutdown_MustNotInitiateIfNotWishedByUser()
|
|
|
|
|
{
|
|
|
|
|
var args = new System.ComponentModel.CancelEventArgs();
|
|
|
|
|
|
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.No);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
taskbar.Raise(t => t.QuitButtonClicked += null, args as object);
|
|
|
|
|
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Never);
|
|
|
|
|
Assert.IsTrue(args.Cancel);
|
2019-02-12 13:57:22 +01:00
|
|
|
|
}
|
2017-07-28 11:05:42 +02:00
|
|
|
|
|
2019-03-06 16:16:50 +01:00
|
|
|
|
[TestMethod]
|
2019-11-14 14:03:43 +01:00
|
|
|
|
public void Shutdown_MustCloseActionCenterAndTaskbarIfEnabled()
|
2019-03-06 16:16:50 +01:00
|
|
|
|
{
|
2019-11-14 14:03:43 +01:00
|
|
|
|
settings.ActionCenter.EnableActionCenter = true;
|
|
|
|
|
settings.Taskbar.EnableTaskbar = true;
|
|
|
|
|
|
2019-03-06 16:16:50 +01:00
|
|
|
|
sut.Terminate();
|
2019-11-14 14:03:43 +01:00
|
|
|
|
|
2019-03-06 16:16:50 +01:00
|
|
|
|
actionCenter.Verify(a => a.Close(), Times.Once);
|
|
|
|
|
taskbar.Verify(o => o.Close(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-14 14:03:43 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Shutdown_MustNotCloseActionCenterAndTaskbarIfNotEnabled()
|
|
|
|
|
{
|
|
|
|
|
settings.ActionCenter.EnableActionCenter = false;
|
|
|
|
|
settings.Taskbar.EnableTaskbar = false;
|
|
|
|
|
|
|
|
|
|
sut.Terminate();
|
|
|
|
|
|
|
|
|
|
actionCenter.Verify(a => a.Close(), Times.Never);
|
|
|
|
|
taskbar.Verify(o => o.Close(), Times.Never);
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-28 11:05:42 +02:00
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Shutdown_MustShowErrorMessageOnCommunicationFailure()
|
2017-07-28 11:05:42 +02:00
|
|
|
|
{
|
2019-02-15 09:06:21 +01:00
|
|
|
|
var args = new System.ComponentModel.CancelEventArgs();
|
2019-02-12 13:57:22 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.Yes);
|
|
|
|
|
runtimeProxy.Setup(r => r.RequestShutdown()).Returns(new CommunicationResult(false));
|
2019-02-12 13:57:22 +01:00
|
|
|
|
|
2019-02-14 16:05:58 +01:00
|
|
|
|
sut.TryStart();
|
2019-02-15 09:06:21 +01:00
|
|
|
|
taskbar.Raise(t => t.QuitButtonClicked += null, args as object);
|
|
|
|
|
|
|
|
|
|
messageBox.Verify(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.Is<MessageBoxIcon>(i => i == MessageBoxIcon.Error),
|
|
|
|
|
It.IsAny<IWindow>()), Times.Once);
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once);
|
|
|
|
|
}
|
2019-02-12 13:57:22 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Shutdown_MustAskUserForQuitPassword()
|
|
|
|
|
{
|
|
|
|
|
var args = new System.ComponentModel.CancelEventArgs();
|
|
|
|
|
var dialog = new Mock<IPasswordDialog>();
|
2019-10-11 15:46:15 +02:00
|
|
|
|
var dialogResult = new PasswordDialogResult { Password = "blobb", Success = true };
|
2019-02-15 09:06:21 +01:00
|
|
|
|
|
2019-12-20 11:37:07 +01:00
|
|
|
|
settings.Security.QuitPasswordHash = "1234";
|
2019-10-11 15:46:15 +02:00
|
|
|
|
dialog.Setup(d => d.Show(It.IsAny<IWindow>())).Returns(dialogResult);
|
2019-12-20 11:37:07 +01:00
|
|
|
|
hashAlgorithm.Setup(h => h.GenerateHashFor(It.Is<string>(s => s == dialogResult.Password))).Returns(settings.Security.QuitPasswordHash);
|
2019-02-15 09:06:21 +01:00
|
|
|
|
runtimeProxy.Setup(r => r.RequestShutdown()).Returns(new CommunicationResult(true));
|
|
|
|
|
uiFactory.Setup(u => u.CreatePasswordDialog(It.IsAny<TextKey>(), It.IsAny<TextKey>())).Returns(dialog.Object);
|
2019-02-12 13:57:22 +01:00
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
sut.TryStart();
|
|
|
|
|
taskbar.Raise(t => t.QuitButtonClicked += null, args as object);
|
|
|
|
|
|
|
|
|
|
uiFactory.Verify(u => u.CreatePasswordDialog(It.IsAny<TextKey>(), It.IsAny<TextKey>()), Times.Once);
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once);
|
|
|
|
|
|
|
|
|
|
Assert.IsFalse(args.Cancel);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-04 16:06:05 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Shutdown_MustAbortAskingUserForQuitPassword()
|
|
|
|
|
{
|
|
|
|
|
var args = new System.ComponentModel.CancelEventArgs();
|
|
|
|
|
var dialog = new Mock<IPasswordDialog>();
|
2019-10-11 15:46:15 +02:00
|
|
|
|
var dialogResult = new PasswordDialogResult { Success = false };
|
2019-04-04 16:06:05 +02:00
|
|
|
|
|
2019-12-20 11:37:07 +01:00
|
|
|
|
settings.Security.QuitPasswordHash = "1234";
|
2019-10-11 15:46:15 +02:00
|
|
|
|
dialog.Setup(d => d.Show(It.IsAny<IWindow>())).Returns(dialogResult);
|
2019-04-04 16:06:05 +02:00
|
|
|
|
runtimeProxy.Setup(r => r.RequestShutdown()).Returns(new CommunicationResult(true));
|
|
|
|
|
uiFactory.Setup(u => u.CreatePasswordDialog(It.IsAny<TextKey>(), It.IsAny<TextKey>())).Returns(dialog.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
taskbar.Raise(t => t.QuitButtonClicked += null, args as object);
|
|
|
|
|
|
|
|
|
|
uiFactory.Verify(u => u.CreatePasswordDialog(It.IsAny<TextKey>(), It.IsAny<TextKey>()), Times.Once);
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Never);
|
|
|
|
|
|
|
|
|
|
Assert.IsTrue(args.Cancel);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-15 09:06:21 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Shutdown_MustNotInitiateIfQuitPasswordIncorrect()
|
|
|
|
|
{
|
|
|
|
|
var args = new System.ComponentModel.CancelEventArgs();
|
|
|
|
|
var dialog = new Mock<IPasswordDialog>();
|
2019-10-11 15:46:15 +02:00
|
|
|
|
var dialogResult = new PasswordDialogResult { Password = "blobb", Success = true };
|
2019-02-15 09:06:21 +01:00
|
|
|
|
|
2019-12-20 11:37:07 +01:00
|
|
|
|
settings.Security.QuitPasswordHash = "1234";
|
2019-10-11 15:46:15 +02:00
|
|
|
|
dialog.Setup(d => d.Show(It.IsAny<IWindow>())).Returns(dialogResult);
|
2019-02-15 09:06:21 +01:00
|
|
|
|
hashAlgorithm.Setup(h => h.GenerateHashFor(It.IsAny<string>())).Returns("9876");
|
|
|
|
|
uiFactory.Setup(u => u.CreatePasswordDialog(It.IsAny<TextKey>(), It.IsAny<TextKey>())).Returns(dialog.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
taskbar.Raise(t => t.QuitButtonClicked += null, args as object);
|
|
|
|
|
|
|
|
|
|
messageBox.Verify(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.Is<MessageBoxIcon>(i => i == MessageBoxIcon.Warning),
|
|
|
|
|
It.IsAny<IWindow>()), Times.Once);
|
|
|
|
|
uiFactory.Verify(u => u.CreatePasswordDialog(It.IsAny<TextKey>(), It.IsAny<TextKey>()), Times.Once);
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Never);
|
|
|
|
|
|
|
|
|
|
Assert.IsTrue(args.Cancel);
|
2017-07-28 11:05:42 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-12 13:57:22 +01:00
|
|
|
|
[TestMethod]
|
2019-02-15 09:06:21 +01:00
|
|
|
|
public void Shutdown_MustRevertOperations()
|
|
|
|
|
{
|
|
|
|
|
operationSequence.Setup(o => o.TryRevert()).Returns(OperationResult.Success);
|
|
|
|
|
sut.Terminate();
|
|
|
|
|
operationSequence.Verify(o => o.TryRevert(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Shutdown_MustNotFailIfDependenciesAreNull()
|
|
|
|
|
{
|
2020-08-06 14:06:29 +02:00
|
|
|
|
context.AppConfig = null;
|
2019-10-01 16:24:10 +02:00
|
|
|
|
context.Browser = null;
|
|
|
|
|
context.ClientHost = null;
|
2020-08-06 14:06:29 +02:00
|
|
|
|
context.Server = null;
|
|
|
|
|
context.Settings = null;
|
2019-02-15 09:06:21 +01:00
|
|
|
|
|
|
|
|
|
sut.Terminate();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Startup_MustPerformOperations()
|
|
|
|
|
{
|
|
|
|
|
var success = sut.TryStart();
|
|
|
|
|
|
|
|
|
|
operationSequence.Verify(o => o.TryPerform(), Times.Once);
|
|
|
|
|
Assert.IsTrue(success);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Startup_MustHandleCommunicationError()
|
|
|
|
|
{
|
|
|
|
|
runtimeProxy.Setup(r => r.InformClientReady()).Returns(new CommunicationResult(false));
|
|
|
|
|
|
|
|
|
|
var success = sut.TryStart();
|
|
|
|
|
|
|
|
|
|
Assert.IsFalse(success);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Startup_MustHandleFailure()
|
|
|
|
|
{
|
|
|
|
|
var success = true;
|
|
|
|
|
|
|
|
|
|
operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Failed);
|
|
|
|
|
success = sut.TryStart();
|
|
|
|
|
|
|
|
|
|
Assert.IsFalse(success);
|
|
|
|
|
|
|
|
|
|
operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Aborted);
|
|
|
|
|
success = sut.TryStart();
|
|
|
|
|
|
|
|
|
|
Assert.IsFalse(success);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Startup_MustUpdateAppConfigForSplashScreen()
|
|
|
|
|
{
|
|
|
|
|
sut.TryStart();
|
2019-10-01 16:24:10 +02:00
|
|
|
|
sut.UpdateAppConfig();
|
2019-02-15 09:06:21 +01:00
|
|
|
|
|
|
|
|
|
splashScreen.VerifySet(s => s.AppConfig = appConfig, Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-06 16:10:00 +01:00
|
|
|
|
[TestMethod]
|
2019-12-09 16:25:03 +01:00
|
|
|
|
public void Startup_MustCorrectlyShowTaskbar()
|
2019-03-06 16:10:00 +01:00
|
|
|
|
{
|
2019-03-06 16:16:50 +01:00
|
|
|
|
settings.Taskbar.EnableTaskbar = true;
|
2019-03-06 16:10:00 +01:00
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
|
|
|
|
taskbar.Verify(t => t.Show(), Times.Once);
|
|
|
|
|
|
|
|
|
|
taskbar.Reset();
|
|
|
|
|
operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Aborted);
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
|
|
|
|
taskbar.Verify(t => t.Show(), Times.Never);
|
2019-12-09 16:25:03 +01:00
|
|
|
|
|
|
|
|
|
taskbar.Reset();
|
|
|
|
|
settings.Taskbar.EnableTaskbar = false;
|
|
|
|
|
operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
|
|
|
|
taskbar.Verify(t => t.Show(), Times.Never);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Startup_MustCorrectlyShowActionCenter()
|
|
|
|
|
{
|
|
|
|
|
settings.ActionCenter.EnableActionCenter = true;
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
2020-04-01 13:49:32 +02:00
|
|
|
|
actionCenter.Verify(t => t.Promote(), Times.Once);
|
|
|
|
|
actionCenter.Verify(t => t.Show(), Times.Never);
|
2019-12-09 16:25:03 +01:00
|
|
|
|
|
|
|
|
|
actionCenter.Reset();
|
|
|
|
|
operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Aborted);
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
2020-04-01 13:49:32 +02:00
|
|
|
|
actionCenter.Verify(t => t.Promote(), Times.Never);
|
2019-12-09 16:25:03 +01:00
|
|
|
|
actionCenter.Verify(t => t.Show(), Times.Never);
|
|
|
|
|
|
|
|
|
|
actionCenter.Reset();
|
|
|
|
|
settings.ActionCenter.EnableActionCenter = false;
|
|
|
|
|
operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
2020-04-01 13:49:32 +02:00
|
|
|
|
actionCenter.Verify(t => t.Promote(), Times.Never);
|
2019-12-09 16:25:03 +01:00
|
|
|
|
actionCenter.Verify(t => t.Show(), Times.Never);
|
2019-03-06 16:10:00 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-11-20 15:30:53 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Startup_MustAutoStartApplications()
|
|
|
|
|
{
|
|
|
|
|
var application1 = new Mock<IApplication>();
|
|
|
|
|
var application2 = new Mock<IApplication>();
|
|
|
|
|
var application3 = new Mock<IApplication>();
|
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
application1.SetupGet(a => a.AutoStart).Returns(true);
|
|
|
|
|
application2.SetupGet(a => a.AutoStart).Returns(false);
|
|
|
|
|
application3.SetupGet(a => a.AutoStart).Returns(true);
|
2019-11-20 15:30:53 +01:00
|
|
|
|
context.Applications.Add(application1.Object);
|
|
|
|
|
context.Applications.Add(application2.Object);
|
|
|
|
|
context.Applications.Add(application3.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
|
|
|
|
application1.Verify(a => a.Start(), Times.Once);
|
|
|
|
|
application2.Verify(a => a.Start(), Times.Never);
|
|
|
|
|
application3.Verify(a => a.Start(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Startup_MustAutoStartBrowser()
|
|
|
|
|
{
|
2020-01-10 10:25:51 +01:00
|
|
|
|
settings.Browser.EnableBrowser = true;
|
2019-12-02 15:48:06 +01:00
|
|
|
|
browser.SetupGet(b => b.AutoStart).Returns(true);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
|
|
|
|
browser.Verify(b => b.Start(), Times.Once);
|
|
|
|
|
browser.Reset();
|
|
|
|
|
browser.SetupGet(b => b.AutoStart).Returns(false);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
|
|
|
|
browser.Verify(b => b.Start(), Times.Never);
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-10 10:25:51 +01:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Startup_MustNotAutoStartBrowserIfNotEnabled()
|
|
|
|
|
{
|
|
|
|
|
settings.Browser.EnableBrowser = false;
|
|
|
|
|
browser.SetupGet(b => b.AutoStart).Returns(true);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
|
|
|
|
|
browser.Verify(b => b.Start(), Times.Never);
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-22 14:21:06 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void SystemMonitor_MustShowLockScreenOnSessionSwitch()
|
|
|
|
|
{
|
|
|
|
|
var lockScreen = new Mock<ILockScreen>();
|
|
|
|
|
|
|
|
|
|
settings.Service.IgnoreService = true;
|
|
|
|
|
lockScreen.Setup(l => l.WaitForResult()).Returns(new LockScreenResult());
|
|
|
|
|
uiFactory.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>())).Returns(lockScreen.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
systemMonitor.Raise(m => m.SessionSwitched += null);
|
|
|
|
|
|
|
|
|
|
lockScreen.Verify(l => l.Show(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void SystemMonitor_MustTerminateIfRequestedByUser()
|
|
|
|
|
{
|
|
|
|
|
var lockScreen = new Mock<ILockScreen>();
|
|
|
|
|
var result = new LockScreenResult();
|
|
|
|
|
|
|
|
|
|
settings.Service.IgnoreService = true;
|
|
|
|
|
lockScreen.Setup(l => l.WaitForResult()).Returns(result);
|
|
|
|
|
runtimeProxy.Setup(r => r.RequestShutdown()).Returns(new CommunicationResult(true));
|
|
|
|
|
uiFactory
|
|
|
|
|
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
|
|
|
|
|
.Callback(new Action<string, string, IEnumerable<LockScreenOption>>((message, title, options) => result.OptionId = options.Last().Id))
|
|
|
|
|
.Returns(lockScreen.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
systemMonitor.Raise(m => m.SessionSwitched += null);
|
|
|
|
|
|
|
|
|
|
lockScreen.Verify(l => l.Show(), Times.Once);
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void SystemMonitor_MustDoNothingIfSessionSwitchAllowed()
|
|
|
|
|
{
|
|
|
|
|
var lockScreen = new Mock<ILockScreen>();
|
|
|
|
|
|
|
|
|
|
settings.Service.IgnoreService = false;
|
|
|
|
|
settings.Service.DisableUserLock = false;
|
|
|
|
|
settings.Service.DisableUserSwitch = false;
|
|
|
|
|
lockScreen.Setup(l => l.WaitForResult()).Returns(new LockScreenResult());
|
|
|
|
|
uiFactory.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>())).Returns(lockScreen.Object);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
systemMonitor.Raise(m => m.SessionSwitched += null);
|
|
|
|
|
|
|
|
|
|
lockScreen.Verify(l => l.Show(), Times.Never);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-04 16:06:05 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void TerminationActivator_MustCorrectlyInitiateShutdown()
|
|
|
|
|
{
|
|
|
|
|
var order = 0;
|
|
|
|
|
var pause = 0;
|
|
|
|
|
var resume = 0;
|
2019-12-09 16:25:03 +01:00
|
|
|
|
var terminationActivator = new Mock<ITerminationActivator>();
|
2019-04-04 16:06:05 +02:00
|
|
|
|
|
2019-12-09 16:25:03 +01:00
|
|
|
|
context.Activators.Add(terminationActivator.Object);
|
2019-04-04 16:06:05 +02:00
|
|
|
|
messageBox.Setup(m => m.Show(
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<TextKey>(),
|
|
|
|
|
It.IsAny<MessageBoxAction>(),
|
|
|
|
|
It.IsAny<MessageBoxIcon>(),
|
|
|
|
|
It.IsAny<IWindow>())).Returns(MessageBoxResult.Yes);
|
|
|
|
|
runtimeProxy.Setup(r => r.RequestShutdown()).Returns(new CommunicationResult(true));
|
|
|
|
|
terminationActivator.Setup(t => t.Pause()).Callback(() => pause = ++order);
|
|
|
|
|
terminationActivator.Setup(t => t.Resume()).Callback(() => resume = ++order);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
terminationActivator.Raise(t => t.Activated += null);
|
|
|
|
|
|
|
|
|
|
Assert.AreEqual(1, pause);
|
|
|
|
|
Assert.AreEqual(2, resume);
|
|
|
|
|
terminationActivator.Verify(t => t.Pause(), Times.Once);
|
|
|
|
|
terminationActivator.Verify(t => t.Resume(), Times.Once);
|
|
|
|
|
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once);
|
|
|
|
|
}
|
2017-07-28 11:05:42 +02:00
|
|
|
|
}
|
|
|
|
|
}
|