/* * Copyright (c) 2019 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.Contracts.Communication.Hosts; using SafeExamBrowser.Contracts.Communication.Proxies; using SafeExamBrowser.Contracts.Configuration; using SafeExamBrowser.Contracts.Configuration.Settings; using SafeExamBrowser.Contracts.Core.OperationModel; using SafeExamBrowser.Contracts.Logging; using SafeExamBrowser.Runtime.Operations; namespace SafeExamBrowser.Runtime.UnitTests.Operations { [TestClass] public class ServiceOperationTests { private Mock logger; private Mock runtimeHost; private Mock service; private Mock session; private SessionContext sessionContext; private Settings settings; private ServiceOperation sut; [TestInitialize] public void Initialize() { logger = new Mock(); runtimeHost = new Mock(); service = new Mock(); session = new Mock(); sessionContext = new SessionContext(); settings = new Settings(); sessionContext.Current = session.Object; sessionContext.Next = session.Object; session.SetupGet(s => s.Settings).Returns(settings); settings.ServicePolicy = ServicePolicy.Mandatory; sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, 0); } [TestMethod] public void Perform_MustConnectToService() { service.Setup(s => s.Connect(null, true)).Returns(true); settings.ServicePolicy = ServicePolicy.Mandatory; sut.Perform(); service.Setup(s => s.Connect(null, true)).Returns(true); settings.ServicePolicy = ServicePolicy.Optional; sut.Perform(); service.Verify(s => s.Connect(null, true), Times.Exactly(2)); } [TestMethod] public void Perform_MustStartSessionIfConnected() { service.SetupGet(s => s.IsConnected).Returns(true); service.Setup(s => s.Connect(null, true)).Returns(true); service .Setup(s => s.StartSession(It.IsAny(), It.IsAny())) .Returns(new CommunicationResult(true)) .Callback(() => runtimeHost.Raise(h => h.ServiceSessionStarted += null)); var result = sut.Perform(); service.Verify(s => s.StartSession(It.IsAny(), It.IsAny()), Times.Once); Assert.AreEqual(OperationResult.Success, result); } [TestMethod] public void Perform_MustFailIfSessionStartUnsuccessful() { service.SetupGet(s => s.IsConnected).Returns(true); service.Setup(s => s.Connect(null, true)).Returns(true); service .Setup(s => s.StartSession(It.IsAny(), It.IsAny())) .Returns(new CommunicationResult(true)) .Callback(() => runtimeHost.Raise(h => h.ServiceFailed += null)); var result = sut.Perform(); service.Verify(s => s.StartSession(It.IsAny(), It.IsAny()), Times.Once); Assert.AreEqual(OperationResult.Failed, result); } [TestMethod] public void Perform_MustFailIfSessionNotStartedWithinTimeout() { const int TIMEOUT = 50; var after = default(DateTime); var before = default(DateTime); service.SetupGet(s => s.IsConnected).Returns(true); service.Setup(s => s.Connect(null, true)).Returns(true); service.Setup(s => s.StartSession(It.IsAny(), It.IsAny())).Returns(new CommunicationResult(true)); sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, TIMEOUT); before = DateTime.Now; var result = sut.Perform(); after = DateTime.Now; service.Verify(s => s.StartSession(It.IsAny(), It.IsAny()), Times.Once); Assert.AreEqual(OperationResult.Failed, result); Assert.IsTrue(after - before >= new TimeSpan(0, 0, 0, 0, TIMEOUT)); } [TestMethod] public void Perform_MustNotStartSessionIfNotConnected() { service.SetupGet(s => s.IsConnected).Returns(false); service.Setup(s => s.Connect(null, true)).Returns(false); sut.Perform(); service.Verify(s => s.StartSession(It.IsAny(), It.IsAny()), Times.Never); } [TestMethod] public void Perform_MustFailIfServiceMandatoryAndNotAvailable() { service.SetupGet(s => s.IsConnected).Returns(false); service.Setup(s => s.Connect(null, true)).Returns(false); settings.ServicePolicy = ServicePolicy.Mandatory; var result = sut.Perform(); Assert.AreEqual(OperationResult.Failed, result); } [TestMethod] public void Perform_MustNotFailIfServiceOptionalAndNotAvailable() { service.SetupGet(s => s.IsConnected).Returns(false); service.Setup(s => s.Connect(null, true)).Returns(false); settings.ServicePolicy = ServicePolicy.Optional; var result = sut.Perform(); service.VerifySet(s => s.Ignore = true); Assert.AreEqual(OperationResult.Success, result); } [TestMethod] public void Repeat_MustStopCurrentAndStartNewSession() { service .Setup(s => s.StopSession(It.IsAny())) .Returns(new CommunicationResult(true)) .Callback(() => runtimeHost.Raise(h => h.ServiceSessionStopped += null)); PerformNormally(); var result = sut.Repeat(); service.Verify(s => s.Connect(It.IsAny(), It.IsAny()), Times.Once); service.Verify(s => s.StopSession(It.IsAny()), Times.Once); service.Verify(s => s.StartSession(It.IsAny(), It.IsAny()), Times.Exactly(2)); service.Verify(s => s.Disconnect(), Times.Never); Assert.AreEqual(OperationResult.Success, result); } [TestMethod] public void Repeat_MustFailIfCurrentSessionWasNotStoppedSuccessfully() { service.Setup(s => s.StopSession(It.IsAny())).Returns(new CommunicationResult(false)); PerformNormally(); var result = sut.Repeat(); service.Verify(s => s.StopSession(It.IsAny()), Times.Once); service.Verify(s => s.StartSession(It.IsAny(), It.IsAny()), Times.Once); service.Verify(s => s.Disconnect(), Times.Never); Assert.AreEqual(OperationResult.Failed, result); } [TestMethod] public void Revert_MustDisconnect() { service.Setup(s => s.Disconnect()).Returns(true).Callback(() => runtimeHost.Raise(h => h.ServiceDisconnected += null)); service .Setup(s => s.StopSession(It.IsAny())) .Returns(new CommunicationResult(true)) .Callback(() => runtimeHost.Raise(h => h.ServiceSessionStopped += null)); PerformNormally(); var result = sut.Revert(); service.Verify(s => s.Disconnect(), Times.Once); Assert.AreEqual(OperationResult.Success, result); } [TestMethod] public void Revert_MustFailIfServiceNotDisconnectedWithinTimeout() { const int TIMEOUT = 50; var after = default(DateTime); var before = default(DateTime); sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, TIMEOUT); service.Setup(s => s.Disconnect()).Returns(true); service .Setup(s => s.StopSession(It.IsAny())) .Returns(new CommunicationResult(true)) .Callback(() => runtimeHost.Raise(h => h.ServiceSessionStopped += null)); PerformNormally(); before = DateTime.Now; var result = sut.Revert(); after = DateTime.Now; service.Verify(s => s.Disconnect(), Times.Once); Assert.AreEqual(OperationResult.Failed, result); Assert.IsTrue(after - before >= new TimeSpan(0, 0, 0, 0, TIMEOUT)); } [TestMethod] public void Revert_MustStopSessionIfConnected() { service.Setup(s => s.Disconnect()).Returns(true).Callback(() => runtimeHost.Raise(h => h.ServiceDisconnected += null)); service .Setup(s => s.StopSession(It.IsAny())) .Returns(new CommunicationResult(true)) .Callback(() => runtimeHost.Raise(h => h.ServiceSessionStopped += null)); PerformNormally(); var result = sut.Revert(); service.Verify(s => s.StopSession(It.IsAny()), Times.Once); service.Verify(s => s.Disconnect(), Times.Once); Assert.AreEqual(OperationResult.Success, result); } [TestMethod] public void Revert_MustHandleCommunicationFailureWhenStoppingSession() { service.Setup(s => s.StopSession(It.IsAny())).Returns(new CommunicationResult(false)); PerformNormally(); var result = sut.Revert(); service.Verify(s => s.StopSession(It.IsAny()), Times.Once); service.Verify(s => s.Disconnect(), Times.Once); Assert.AreEqual(OperationResult.Failed, result); } [TestMethod] public void Revert_MustFailIfSessionNotStoppedWithinTimeout() { const int TIMEOUT = 50; var after = default(DateTime); var before = default(DateTime); service.Setup(s => s.StopSession(It.IsAny())).Returns(new CommunicationResult(true)); sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, TIMEOUT); PerformNormally(); before = DateTime.Now; var result = sut.Revert(); after = DateTime.Now; service.Verify(s => s.StopSession(It.IsAny()), Times.Once); service.Verify(s => s.Disconnect(), Times.Once); Assert.AreEqual(OperationResult.Failed, result); Assert.IsTrue(after - before >= new TimeSpan(0, 0, 0, 0, TIMEOUT)); } [TestMethod] public void Revert_MustNotStopSessionWhenNotConnected() { var result = sut.Revert(); service.Verify(s => s.StopSession(It.IsAny()), Times.Never); Assert.AreEqual(OperationResult.Success, result); } [TestMethod] public void Revert_MustNotDisconnnectIfNotConnected() { var result = sut.Revert(); service.Verify(s => s.Disconnect(), Times.Never); Assert.AreEqual(OperationResult.Success, result); } private void PerformNormally() { service.SetupGet(s => s.IsConnected).Returns(true); service.Setup(s => s.Connect(null, true)).Returns(true); service .Setup(s => s.StartSession(It.IsAny(), It.IsAny())) .Returns(new CommunicationResult(true)) .Callback(() => runtimeHost.Raise(h => h.ServiceSessionStarted += null)); sut.Perform(); } } }