/* * Copyright (c) 2021 ETH Zürich, Educational Development and Technology (LET) * * 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/. */ using System; using Microsoft.VisualStudio.TestTools.UnitTesting; using Moq; 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.Core.Contracts.OperationModel; using SafeExamBrowser.Core.Contracts.OperationModel.Events; using SafeExamBrowser.I18n.Contracts; using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Runtime.Operations.Events; using SafeExamBrowser.Settings; using SafeExamBrowser.Settings.Security; using SafeExamBrowser.Settings.Service; using SafeExamBrowser.UserInterface.Contracts; using SafeExamBrowser.UserInterface.Contracts.MessageBox; using SafeExamBrowser.UserInterface.Contracts.Windows; using SafeExamBrowser.UserInterface.Contracts.Windows.Data; using SafeExamBrowser.WindowsApi.Contracts; namespace SafeExamBrowser.Runtime.UnitTests { [TestClass] public class RuntimeControllerTests { private AppConfig appConfig; private Mock bootstrapSequence; private Mock clientProcess; private Mock clientProxy; private SessionConfiguration currentSession; private AppSettings currentSettings; private Mock logger; private Mock messageBox; private SessionConfiguration nextSession; private AppSettings nextSettings; private Mock runtimeHost; private Mock runtimeWindow; private Mock service; private SessionContext sessionContext; private Mock sessionSequence; private Mock shutdown; private Mock splashScreen; private Mock text; private Mock uiFactory; private RuntimeController sut; [TestInitialize] public void Initialize() { appConfig = new AppConfig(); bootstrapSequence = new Mock(); clientProcess = new Mock(); clientProxy = new Mock(); currentSession = new SessionConfiguration(); currentSettings = new AppSettings(); logger = new Mock(); messageBox = new Mock(); nextSession = new SessionConfiguration(); nextSettings = new AppSettings(); runtimeHost = new Mock(); runtimeWindow = new Mock(); service = new Mock(); sessionContext = new SessionContext(); sessionSequence = new Mock(); shutdown = new Mock(); splashScreen = new Mock(); text = new Mock(); uiFactory = new Mock(); currentSession.Settings = currentSettings; nextSession.Settings = nextSettings; sessionContext.ClientProcess = clientProcess.Object; sessionContext.ClientProxy = clientProxy.Object; sessionContext.Current = currentSession; sessionContext.Next = nextSession; uiFactory.Setup(u => u.CreateRuntimeWindow(It.IsAny())).Returns(new Mock().Object); uiFactory.Setup(u => u.CreateSplashScreen(It.IsAny())).Returns(new Mock().Object); sut = new RuntimeController( appConfig, logger.Object, messageBox.Object, bootstrapSequence.Object, sessionSequence.Object, runtimeHost.Object, runtimeWindow.Object, service.Object, sessionContext, shutdown.Object, splashScreen.Object, text.Object, uiFactory.Object); } [TestMethod] public void ClientProcess_MustShutdownWhenClientTerminated() { StartSession(); clientProcess.Raise(c => c.Terminated += null, -1); messageBox.Verify(m => m.Show( It.IsAny(), It.IsAny(), It.IsAny(), It.Is(i => i == MessageBoxIcon.Error), It.IsAny()), Times.Once); sessionSequence.Verify(s => s.TryRevert(), Times.Once); shutdown.Verify(s => s(), Times.Once); } [TestMethod] public void ClientProxy_MustShutdownWhenConnectionLost() { StartSession(); clientProxy.Raise(c => c.ConnectionLost += null); messageBox.Verify(m => m.Show( It.IsAny(), It.IsAny(), It.IsAny(), It.Is(i => i == MessageBoxIcon.Error), It.IsAny()), Times.Once); sessionSequence.Verify(s => s.TryRevert(), Times.Once); shutdown.Verify(s => s(), Times.Once); } [TestMethod] public void Communication_MustProvideClientConfigurationUponRequest() { var args = new ClientConfigurationEventArgs(); var nextAppConfig = new AppConfig(); var nextSessionId = Guid.NewGuid(); var nextSettings = new AppSettings(); nextSession.AppConfig = nextAppConfig; nextSession.SessionId = nextSessionId; nextSession.Settings = nextSettings; StartSession(); runtimeHost.Raise(r => r.ClientConfigurationNeeded += null, args); Assert.AreSame(nextAppConfig, args.ClientConfiguration.AppConfig); Assert.AreEqual(nextSessionId, args.ClientConfiguration.SessionId); Assert.AreSame(nextSettings, args.ClientConfiguration.Settings); } [TestMethod] public void Communication_MustStartNewSessionUponRequest() { var args = new ReconfigurationEventArgs { ConfigurationPath = "C:\\Some\\File\\Path.seb" }; StartSession(); bootstrapSequence.Reset(); sessionSequence.Reset(); sessionSequence.Setup(s => s.TryRepeat()).Returns(OperationResult.Success); runtimeHost.Raise(r => r.ReconfigurationRequested += null, args); bootstrapSequence.VerifyNoOtherCalls(); sessionSequence.Verify(s => s.TryPerform(), Times.Never); sessionSequence.Verify(s => s.TryRepeat(), Times.Once); sessionSequence.Verify(s => s.TryRevert(), Times.Never); Assert.AreEqual(sessionContext.ReconfigurationFilePath, args.ConfigurationPath); } [TestMethod] public void Communication_MustInformClientAboutAbortedReconfiguration() { StartSession(); sessionSequence.Reset(); sessionSequence.Setup(s => s.TryRepeat()).Returns(OperationResult.Aborted); runtimeHost.Raise(r => r.ReconfigurationRequested += null, new ReconfigurationEventArgs()); clientProxy.Verify(c => c.InformReconfigurationAborted(), Times.Once); } [TestMethod] public void Communication_MustShutdownUponRequest() { StartSession(); bootstrapSequence.Reset(); sessionSequence.Reset(); runtimeHost.Raise(r => r.ShutdownRequested += null); bootstrapSequence.VerifyNoOtherCalls(); sessionSequence.VerifyNoOtherCalls(); shutdown.Verify(s => s(), Times.Once); } [TestMethod] public void Operations_MustAllowToAbortStartupForClientConfiguration() { var args = new ConfigurationCompletedEventArgs(); messageBox.Setup(m => m.Show(It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny())).Returns(MessageBoxResult.Yes); sut.TryStart(); sessionSequence.Raise(s => s.ActionRequired += null, args); Assert.IsTrue(args.AbortStartup); } [TestMethod] public void Operations_MustRequestPasswordViaDialogOnDefaultDesktop() { var args = new PasswordRequiredEventArgs(); var passwordDialog = new Mock(); var result = new PasswordDialogResult { Password = "test1234", Success = true }; currentSettings.Security.KioskMode = KioskMode.DisableExplorerShell; passwordDialog.Setup(p => p.Show(It.IsAny())).Returns(result); uiFactory.Setup(u => u.CreatePasswordDialog(It.IsAny(), It.IsAny())).Returns(passwordDialog.Object); sut.TryStart(); sessionSequence.Raise(s => s.ActionRequired += null, args); clientProxy.VerifyNoOtherCalls(); passwordDialog.Verify(p => p.Show(It.IsAny()), Times.Once); uiFactory.Verify(u => u.CreatePasswordDialog(It.IsAny(), It.IsAny()), Times.Once); Assert.AreEqual(true, args.Success); Assert.AreEqual(result.Password, args.Password); } [TestMethod] public void Operations_MustRequestPasswordViaClientOnNewDesktop() { var args = new PasswordRequiredEventArgs(); var passwordReceived = new Action((p, id) => { runtimeHost.Raise(r => r.PasswordReceived += null, new PasswordReplyEventArgs { RequestId = id, Success = true }); }); currentSettings.Security.KioskMode = KioskMode.CreateNewDesktop; clientProxy.Setup(c => c.RequestPassword(It.IsAny(), It.IsAny())).Returns(new CommunicationResult(true)).Callback(passwordReceived); sut.TryStart(); sessionSequence.Raise(s => s.ActionRequired += null, args); clientProxy.Verify(c => c.RequestPassword(It.IsAny(), It.IsAny()), Times.Once); uiFactory.Verify(u => u.CreatePasswordDialog(It.IsAny(), It.IsAny()), Times.Never); } [TestMethod] public void Operations_MustAbortAskingForPasswordViaClientIfDecidedByUser() { var args = new PasswordRequiredEventArgs(); var passwordReceived = new Action((p, id) => { runtimeHost.Raise(r => r.PasswordReceived += null, new PasswordReplyEventArgs { RequestId = id, Success = false }); }); currentSettings.Security.KioskMode = KioskMode.CreateNewDesktop; clientProxy.Setup(c => c.RequestPassword(It.IsAny(), It.IsAny())).Returns(new CommunicationResult(true)).Callback(passwordReceived); sut.TryStart(); sessionSequence.Raise(s => s.ActionRequired += null, args); clientProxy.Verify(c => c.RequestPassword(It.IsAny(), It.IsAny()), Times.Once); uiFactory.Verify(u => u.CreatePasswordDialog(It.IsAny(), It.IsAny()), Times.Never); } [TestMethod] public void Operations_MustNotWaitForPasswordViaClientIfCommunicationHasFailed() { var args = new PasswordRequiredEventArgs(); currentSettings.Security.KioskMode = KioskMode.CreateNewDesktop; clientProxy.Setup(c => c.RequestPassword(It.IsAny(), It.IsAny())).Returns(new CommunicationResult(false)); sut.TryStart(); sessionSequence.Raise(s => s.ActionRequired += null, args); } [TestMethod] public void Operations_MustShowNormalMessageBoxOnDefaultDesktop() { var args = new MessageEventArgs { Icon = MessageBoxIcon.Question, Message = TextKey.MessageBox_ClientConfigurationQuestion, Title = TextKey.MessageBox_ClientConfigurationQuestionTitle }; StartSession(); currentSettings.Security.KioskMode = KioskMode.DisableExplorerShell; sessionSequence.Raise(s => s.ActionRequired += null, args); messageBox.Verify(m => m.Show( It.IsAny(), It.IsAny(), It.Is(a => a == MessageBoxAction.Ok), It.Is(i => i == args.Icon), It.IsAny()), Times.Once); clientProxy.VerifyAdd(p => p.ConnectionLost += It.IsAny()); clientProxy.VerifyNoOtherCalls(); } [TestMethod] public void Operations_MustShowMessageBoxViaClientOnNewDesktop() { var args = new MessageEventArgs { Icon = MessageBoxIcon.Question, Message = TextKey.MessageBox_ClientConfigurationQuestion, Title = TextKey.MessageBox_ClientConfigurationQuestionTitle }; var reply = new MessageBoxReplyEventArgs(); StartSession(); currentSettings.Security.KioskMode = KioskMode.CreateNewDesktop; clientProxy.Setup(c => c.ShowMessage( It.IsAny(), It.IsAny(), It.Is(a => a == (int) MessageBoxAction.Ok), It.IsAny(), It.IsAny())) .Callback((m, t, a, i, id) => { runtimeHost.Raise(r => r.MessageBoxReplyReceived += null, new MessageBoxReplyEventArgs { RequestId = id }); }) .Returns(new CommunicationResult(true)); sessionSequence.Raise(s => s.ActionRequired += null, args); messageBox.Verify(m => m.Show(It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny()), Times.Never); clientProxy.Verify(c => c.ShowMessage( It.IsAny(), It.IsAny(), It.Is(a => a == (int) MessageBoxAction.Ok), It.Is(i => i == (int) args.Icon), It.IsAny()), Times.Once); } [TestMethod] public void Operations_MustUpdateProgress() { var args = new ProgressChangedEventArgs { CurrentValue = 23, IsIndeterminate = true, MaxValue = 150, Progress = true, Regress = true }; sut.TryStart(); sessionSequence.Raise(o => o.ProgressChanged += null, args); runtimeWindow.Verify(s => s.SetValue(It.Is(i => i == args.CurrentValue)), Times.Once); runtimeWindow.Verify(s => s.SetIndeterminate(), Times.Once); runtimeWindow.Verify(s => s.SetMaxValue(It.Is(i => i == args.MaxValue)), Times.Once); runtimeWindow.Verify(s => s.Progress(), Times.Once); runtimeWindow.Verify(s => s.Regress(), Times.Once); } [TestMethod] public void Operations_MustUpdateStatus() { var key = TextKey.OperationStatus_EmptyClipboard; sut.TryStart(); sessionSequence.Raise(o => o.StatusChanged += null, key); runtimeWindow.Verify(s => s.UpdateStatus(It.Is(k => k == key), It.IsAny()), Times.Once); } [TestMethod] public void ServiceProxy_MustShutdownWhenConnectionLostAndMandatory() { currentSettings.Service.Policy = ServicePolicy.Mandatory; StartSession(); service.Raise(c => c.ConnectionLost += null); messageBox.Verify(m => m.Show( It.IsAny(), It.IsAny(), It.IsAny(), It.Is(i => i == MessageBoxIcon.Error), It.IsAny()), Times.Once); sessionSequence.Verify(s => s.TryRevert(), Times.Once); shutdown.Verify(s => s(), Times.Once); } [TestMethod] public void ServiceProxy_MustNotShutdownWhenConnectionLostAndNotMandatory() { currentSettings.Service.Policy = ServicePolicy.Optional; StartSession(); service.Raise(c => c.ConnectionLost += null); messageBox.Verify(m => m.Show( It.IsAny(), It.IsAny(), It.IsAny(), It.Is(i => i == MessageBoxIcon.Error), It.IsAny()), Times.Never); sessionSequence.Verify(s => s.TryRevert(), Times.Never); shutdown.Verify(s => s(), Times.Never); } [TestMethod] public void Shutdown_MustRevertSessionThenBootstrapSequence() { var order = 0; var bootstrap = 0; var session = 0; sut.TryStart(); bootstrapSequence.Reset(); sessionSequence.Reset(); bootstrapSequence.Setup(b => b.TryRevert()).Returns(OperationResult.Success).Callback(() => bootstrap = ++order); sessionSequence.Setup(b => b.TryRevert()).Returns(OperationResult.Success).Callback(() => session = ++order); sut.Terminate(); bootstrapSequence.Verify(b => b.TryPerform(), Times.Never); bootstrapSequence.Verify(b => b.TryRevert(), Times.Once); sessionSequence.Verify(b => b.TryPerform(), Times.Never); sessionSequence.Verify(b => b.TryRepeat(), Times.Never); sessionSequence.Verify(b => b.TryRevert(), Times.Once); Assert.AreEqual(1, session); Assert.AreEqual(2, bootstrap); } [TestMethod] public void Shutdown_MustOnlyRevertBootstrapSequenceIfNoSessionRunning() { var order = 0; var bootstrap = 0; var session = 0; sut.TryStart(); bootstrapSequence.Reset(); sessionSequence.Reset(); bootstrapSequence.Setup(b => b.TryRevert()).Returns(OperationResult.Success).Callback(() => bootstrap = ++order); sessionSequence.Setup(b => b.TryRevert()).Returns(OperationResult.Success).Callback(() => session = ++order); sessionContext.Current = null; sut.Terminate(); bootstrapSequence.Verify(b => b.TryPerform(), Times.Never); bootstrapSequence.Verify(b => b.TryRevert(), Times.Once); sessionSequence.Verify(b => b.TryPerform(), Times.Never); sessionSequence.Verify(b => b.TryRepeat(), Times.Never); sessionSequence.Verify(b => b.TryRevert(), Times.Never); Assert.AreEqual(0, session); Assert.AreEqual(1, bootstrap); } [TestMethod] public void Shutdown_MustIndicateFailureToUser() { var order = 0; var bootstrap = 0; var session = 0; sut.TryStart(); bootstrapSequence.Reset(); sessionSequence.Reset(); bootstrapSequence.Setup(b => b.TryRevert()).Returns(OperationResult.Failed).Callback(() => bootstrap = ++order); sessionSequence.Setup(b => b.TryRevert()).Returns(OperationResult.Success).Callback(() => session = ++order); sut.Terminate(); messageBox.Verify(m => m.Show(It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny()), Times.AtLeastOnce); } [TestMethod] public void Startup_MustPerformBootstrapThenSessionSequence() { var order = 0; var bootstrap = 0; var session = 0; bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success).Callback(() => bootstrap = ++order); sessionSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success).Callback(() => { session = ++order; sessionContext.Current = currentSession; }); sessionContext.Current = null; var success = sut.TryStart(); bootstrapSequence.Verify(b => b.TryPerform(), Times.Once); bootstrapSequence.Verify(b => b.TryRevert(), Times.Never); sessionSequence.Verify(b => b.TryPerform(), Times.Once); sessionSequence.Verify(b => b.TryRepeat(), Times.Never); sessionSequence.Verify(b => b.TryRevert(), Times.Never); Assert.IsTrue(success); Assert.AreEqual(1, bootstrap); Assert.AreEqual(2, session); } [TestMethod] public void Startup_MustNotPerformSessionSequenceIfBootstrapFails() { var order = 0; var bootstrap = 0; var session = 0; bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Failed).Callback(() => bootstrap = ++order); sessionSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success).Callback(() => { session = ++order; sessionContext.Current = currentSession; }); sessionContext.Current = null; var success = sut.TryStart(); bootstrapSequence.Verify(b => b.TryPerform(), Times.Once); bootstrapSequence.Verify(b => b.TryRevert(), Times.Never); sessionSequence.Verify(b => b.TryPerform(), Times.Never); sessionSequence.Verify(b => b.TryRepeat(), Times.Never); sessionSequence.Verify(b => b.TryRevert(), Times.Never); Assert.IsFalse(success); Assert.AreEqual(1, bootstrap); Assert.AreEqual(0, session); } [TestMethod] public void Startup_MustTerminateOnSessionStartFailure() { bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success); sessionSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Failed).Callback(() => sessionContext.Current = currentSession); sessionContext.Current = null; var success = sut.TryStart(); bootstrapSequence.Verify(b => b.TryPerform(), Times.Once); bootstrapSequence.Verify(b => b.TryRevert(), Times.Never); sessionSequence.Verify(b => b.TryPerform(), Times.Once); sessionSequence.Verify(b => b.TryRepeat(), Times.Never); sessionSequence.Verify(b => b.TryRevert(), Times.Once); shutdown.Verify(s => s(), Times.Once); } [TestMethod] public void Startup_MustNotTerminateOnSessionStartAbortion() { bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success); sessionSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Aborted).Callback(() => sessionContext.Current = currentSession); sessionContext.Current = null; var success = sut.TryStart(); bootstrapSequence.Verify(b => b.TryPerform(), Times.Once); bootstrapSequence.Verify(b => b.TryRevert(), Times.Never); sessionSequence.Verify(b => b.TryPerform(), Times.Once); sessionSequence.Verify(b => b.TryRepeat(), Times.Never); sessionSequence.Verify(b => b.TryRevert(), Times.Never); shutdown.Verify(s => s(), Times.Never); } private void StartSession() { bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success); sessionSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success).Callback(() => sessionContext.Current = currentSession); sessionContext.Current = null; sut.TryStart(); } } }