2019-06-07 15:26:03 +02:00
|
|
|
|
/*
|
2023-03-08 00:30:20 +01:00
|
|
|
|
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
2019-06-07 15:26:03 +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-06-19 15:40:21 +02:00
|
|
|
|
using System;
|
2019-07-04 09:12:28 +02:00
|
|
|
|
using System.Threading;
|
2019-06-07 15:26:03 +02:00
|
|
|
|
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
|
|
|
|
using Moq;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Communication.Contracts.Events;
|
|
|
|
|
using SafeExamBrowser.Communication.Contracts.Hosts;
|
|
|
|
|
using SafeExamBrowser.Configuration.Contracts;
|
|
|
|
|
using SafeExamBrowser.Core.Contracts.OperationModel;
|
|
|
|
|
using SafeExamBrowser.Lockdown.Contracts;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2019-09-06 09:39:28 +02:00
|
|
|
|
using SafeExamBrowser.Settings;
|
|
|
|
|
using SafeExamBrowser.Settings.Logging;
|
2019-06-07 15:26:03 +02:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Service.UnitTests
|
|
|
|
|
{
|
|
|
|
|
[TestClass]
|
|
|
|
|
public class ServiceControllerTests
|
|
|
|
|
{
|
2019-06-18 10:18:56 +02:00
|
|
|
|
private Mock<ILogger> logger;
|
2019-07-03 08:59:27 +02:00
|
|
|
|
private Mock<Func<string, ILogObserver>> logWriterFactory;
|
2019-06-07 15:26:03 +02:00
|
|
|
|
private Mock<IOperationSequence> bootstrapSequence;
|
|
|
|
|
private SessionContext sessionContext;
|
2019-06-19 15:40:21 +02:00
|
|
|
|
private Mock<IOperationSequence> sessionSequence;
|
2019-06-07 15:26:03 +02:00
|
|
|
|
private Mock<IServiceHost> serviceHost;
|
2019-07-04 09:12:28 +02:00
|
|
|
|
private Mock<ISystemConfigurationUpdate> systemConfigurationUpdate;
|
2019-06-07 15:26:03 +02:00
|
|
|
|
private ServiceController sut;
|
|
|
|
|
|
|
|
|
|
[TestInitialize]
|
|
|
|
|
public void Initialize()
|
|
|
|
|
{
|
2019-06-18 10:18:56 +02:00
|
|
|
|
logger = new Mock<ILogger>();
|
2019-07-03 08:59:27 +02:00
|
|
|
|
logWriterFactory = new Mock<Func<string, ILogObserver>>();
|
2019-06-07 15:26:03 +02:00
|
|
|
|
bootstrapSequence = new Mock<IOperationSequence>();
|
|
|
|
|
sessionContext = new SessionContext();
|
2019-06-19 15:40:21 +02:00
|
|
|
|
sessionSequence = new Mock<IOperationSequence>();
|
2019-06-07 15:26:03 +02:00
|
|
|
|
serviceHost = new Mock<IServiceHost>();
|
2019-07-04 09:12:28 +02:00
|
|
|
|
systemConfigurationUpdate = new Mock<ISystemConfigurationUpdate>();
|
2019-06-07 15:26:03 +02:00
|
|
|
|
|
2019-07-03 08:59:27 +02:00
|
|
|
|
logWriterFactory.Setup(f => f.Invoke(It.IsAny<string>())).Returns(new Mock<ILogObserver>().Object);
|
|
|
|
|
|
2019-07-04 09:12:28 +02:00
|
|
|
|
sut = new ServiceController(
|
|
|
|
|
logger.Object,
|
|
|
|
|
logWriterFactory.Object,
|
|
|
|
|
bootstrapSequence.Object,
|
|
|
|
|
sessionSequence.Object,
|
|
|
|
|
serviceHost.Object,
|
|
|
|
|
sessionContext,
|
|
|
|
|
systemConfigurationUpdate.Object);
|
2019-06-07 15:26:03 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-06-19 15:40:21 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustStartNewSessionUponRequest()
|
|
|
|
|
{
|
|
|
|
|
var args = new SessionStartEventArgs { Configuration = new ServiceConfiguration { SessionId = Guid.NewGuid() } };
|
|
|
|
|
|
|
|
|
|
bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
sessionSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
serviceHost.Raise(h => h.SessionStartRequested += null, args);
|
|
|
|
|
|
|
|
|
|
sessionSequence.Verify(s => s.TryPerform(), Times.Once);
|
|
|
|
|
|
|
|
|
|
Assert.IsNotNull(sessionContext.Configuration);
|
|
|
|
|
Assert.AreEqual(args.Configuration.SessionId, sessionContext.Configuration.SessionId);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-03 08:59:27 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustInitializeSessionLogging()
|
|
|
|
|
{
|
|
|
|
|
var args = new SessionStartEventArgs
|
|
|
|
|
{
|
|
|
|
|
Configuration = new ServiceConfiguration
|
|
|
|
|
{
|
|
|
|
|
AppConfig = new AppConfig { ServiceLogFilePath = "Test.log" },
|
|
|
|
|
SessionId = Guid.NewGuid(),
|
2019-10-01 11:30:53 +02:00
|
|
|
|
Settings = new AppSettings { LogLevel = LogLevel.Warning }
|
2019-07-03 08:59:27 +02:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
sessionSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
serviceHost.Raise(h => h.SessionStartRequested += null, args);
|
|
|
|
|
|
|
|
|
|
logger.VerifySet(l => l.LogLevel = It.Is<LogLevel>(ll => ll == args.Configuration.Settings.LogLevel), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-19 15:40:21 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustNotAllowNewSessionDuringActiveSession()
|
|
|
|
|
{
|
|
|
|
|
var args = new SessionStartEventArgs { Configuration = new ServiceConfiguration { SessionId = Guid.NewGuid() } };
|
|
|
|
|
|
|
|
|
|
bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
sessionContext.Configuration = new ServiceConfiguration { SessionId = Guid.NewGuid() };
|
2019-06-27 08:32:37 +02:00
|
|
|
|
sessionContext.IsRunning = true;
|
2019-06-19 15:40:21 +02:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
serviceHost.Raise(h => h.SessionStartRequested += null, args);
|
|
|
|
|
|
|
|
|
|
sessionSequence.Verify(s => s.TryPerform(), Times.Never);
|
|
|
|
|
|
|
|
|
|
Assert.AreNotEqual(args.Configuration.SessionId, sessionContext.Configuration.SessionId);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-03 08:59:27 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustNotFailIfNoValidSessionData()
|
|
|
|
|
{
|
|
|
|
|
var args = new SessionStartEventArgs { Configuration = null };
|
|
|
|
|
|
|
|
|
|
bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
sessionSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
serviceHost.Raise(h => h.SessionStartRequested += null, args);
|
|
|
|
|
|
|
|
|
|
sessionSequence.Verify(s => s.TryPerform(), Times.Once);
|
|
|
|
|
|
|
|
|
|
Assert.IsNull(sessionContext.Configuration);
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-19 15:40:21 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustStopActiveSessionUponRequest()
|
|
|
|
|
{
|
|
|
|
|
var args = new SessionStopEventArgs { SessionId = Guid.NewGuid() };
|
|
|
|
|
|
|
|
|
|
bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
sessionContext.Configuration = new ServiceConfiguration { SessionId = args.SessionId };
|
2019-06-27 08:32:37 +02:00
|
|
|
|
sessionContext.IsRunning = true;
|
2019-06-19 15:40:21 +02:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
serviceHost.Raise(h => h.SessionStopRequested += null, args);
|
|
|
|
|
|
|
|
|
|
sessionSequence.Verify(s => s.TryRevert(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustNotStopSessionWithWrongId()
|
|
|
|
|
{
|
|
|
|
|
var args = new SessionStopEventArgs { SessionId = Guid.NewGuid() };
|
|
|
|
|
|
|
|
|
|
bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
sessionContext.Configuration = new ServiceConfiguration { SessionId = Guid.NewGuid() };
|
2019-06-27 10:47:41 +02:00
|
|
|
|
sessionContext.IsRunning = true;
|
2019-06-19 15:40:21 +02:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
serviceHost.Raise(h => h.SessionStopRequested += null, args);
|
|
|
|
|
|
|
|
|
|
sessionSequence.Verify(s => s.TryRevert(), Times.Never);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustNotStopSessionWithoutActiveSession()
|
|
|
|
|
{
|
|
|
|
|
var args = new SessionStopEventArgs { SessionId = Guid.NewGuid() };
|
|
|
|
|
|
|
|
|
|
bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
2019-06-27 08:32:37 +02:00
|
|
|
|
sessionContext.IsRunning = false;
|
2019-06-19 15:40:21 +02:00
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
serviceHost.Raise(h => h.SessionStopRequested += null, args);
|
|
|
|
|
|
|
|
|
|
sessionSequence.Verify(s => s.TryRevert(), Times.Never);
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-04 09:12:28 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Communication_MustStartSystemConfigurationUpdate()
|
|
|
|
|
{
|
|
|
|
|
var sync = new AutoResetEvent(false);
|
|
|
|
|
|
|
|
|
|
bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
systemConfigurationUpdate.Setup(u => u.ExecuteAsync()).Callback(() => sync.Set());
|
|
|
|
|
|
|
|
|
|
sut.TryStart();
|
|
|
|
|
serviceHost.Raise(h => h.SystemConfigurationUpdateRequested += null);
|
|
|
|
|
|
|
|
|
|
sync.WaitOne();
|
|
|
|
|
|
|
|
|
|
systemConfigurationUpdate.Verify(u => u.Execute(), Times.Never);
|
|
|
|
|
systemConfigurationUpdate.Verify(u => u.ExecuteAsync(), Times.Once);
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-07 15:26:03 +02:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Start_MustOnlyPerformBootstrapSequence()
|
|
|
|
|
{
|
|
|
|
|
bootstrapSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
|
|
|
|
sessionSequence.Setup(b => b.TryPerform()).Returns(OperationResult.Success);
|
2019-06-18 10:18:56 +02:00
|
|
|
|
sessionContext.Configuration = null;
|
2019-06-07 15:26:03 +02:00
|
|
|
|
|
|
|
|
|
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.TryRevert(), Times.Never);
|
|
|
|
|
|
|
|
|
|
Assert.IsTrue(success);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Stop_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);
|
2019-06-18 10:18:56 +02:00
|
|
|
|
sessionContext.Configuration = new ServiceConfiguration();
|
2019-06-27 08:32:37 +02:00
|
|
|
|
sessionContext.IsRunning = true;
|
2019-06-07 15:26:03 +02:00
|
|
|
|
|
|
|
|
|
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.TryRevert(), Times.Once);
|
|
|
|
|
|
|
|
|
|
Assert.AreEqual(1, session);
|
|
|
|
|
Assert.AreEqual(2, bootstrap);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Stop_MustNotRevertSessionSequenceIfNoSessionRunning()
|
|
|
|
|
{
|
|
|
|
|
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);
|
2019-06-18 10:18:56 +02:00
|
|
|
|
sessionContext.Configuration = null;
|
2019-06-07 15:26:03 +02:00
|
|
|
|
|
|
|
|
|
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.TryRevert(), Times.Never);
|
|
|
|
|
|
|
|
|
|
Assert.AreEqual(0, session);
|
|
|
|
|
Assert.AreEqual(1, bootstrap);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|