2018-03-21 15:28:59 +01:00
|
|
|
|
/*
|
2019-01-09 11:25:21 +01:00
|
|
|
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
2018-03-21 15:28:59 +01: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/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
using System;
|
2019-03-22 15:41:25 +01:00
|
|
|
|
using System.Threading.Tasks;
|
2018-03-21 15:28:59 +01:00
|
|
|
|
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
|
|
|
|
using Moq;
|
|
|
|
|
using SafeExamBrowser.Contracts.Communication.Data;
|
|
|
|
|
using SafeExamBrowser.Contracts.Communication.Hosts;
|
|
|
|
|
using SafeExamBrowser.Contracts.Communication.Proxies;
|
|
|
|
|
using SafeExamBrowser.Contracts.Configuration;
|
2019-02-12 10:34:39 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.Settings;
|
2018-08-31 10:06:27 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.Core.OperationModel;
|
2018-03-21 15:28:59 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Logging;
|
|
|
|
|
using SafeExamBrowser.Contracts.WindowsApi;
|
2018-08-31 10:06:27 +02:00
|
|
|
|
using SafeExamBrowser.Runtime.Operations;
|
2018-03-21 15:28:59 +01:00
|
|
|
|
|
2018-08-31 10:06:27 +02:00
|
|
|
|
namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
2018-03-21 15:28:59 +01:00
|
|
|
|
{
|
|
|
|
|
[TestClass]
|
|
|
|
|
public class ClientOperationTests
|
|
|
|
|
{
|
|
|
|
|
private Action clientReady;
|
|
|
|
|
private Action terminated;
|
2018-06-29 09:50:20 +02:00
|
|
|
|
private AppConfig appConfig;
|
2018-03-21 15:28:59 +01:00
|
|
|
|
private Mock<IClientProxy> proxy;
|
|
|
|
|
private Mock<ILogger> logger;
|
|
|
|
|
private Mock<IProcess> process;
|
|
|
|
|
private Mock<IProcessFactory> processFactory;
|
|
|
|
|
private Mock<IProxyFactory> proxyFactory;
|
|
|
|
|
private Mock<IRuntimeHost> runtimeHost;
|
2019-06-11 09:53:33 +02:00
|
|
|
|
private SessionConfiguration session;
|
2018-10-12 15:23:43 +02:00
|
|
|
|
private SessionContext sessionContext;
|
2019-02-12 10:34:39 +01:00
|
|
|
|
private Settings settings;
|
2018-03-21 15:28:59 +01:00
|
|
|
|
private ClientOperation sut;
|
|
|
|
|
|
|
|
|
|
[TestInitialize]
|
|
|
|
|
public void Initialize()
|
|
|
|
|
{
|
2018-06-29 09:50:20 +02:00
|
|
|
|
appConfig = new AppConfig();
|
2018-03-21 15:28:59 +01:00
|
|
|
|
clientReady = new Action(() => runtimeHost.Raise(h => h.ClientReady += null));
|
|
|
|
|
logger = new Mock<ILogger>();
|
|
|
|
|
process = new Mock<IProcess>();
|
|
|
|
|
processFactory = new Mock<IProcessFactory>();
|
|
|
|
|
proxy = new Mock<IClientProxy>();
|
|
|
|
|
proxyFactory = new Mock<IProxyFactory>();
|
|
|
|
|
runtimeHost = new Mock<IRuntimeHost>();
|
2019-06-11 09:53:33 +02:00
|
|
|
|
session = new SessionConfiguration();
|
2018-10-12 15:23:43 +02:00
|
|
|
|
sessionContext = new SessionContext();
|
2019-02-12 10:34:39 +01:00
|
|
|
|
settings = new Settings();
|
2018-03-21 15:28:59 +01:00
|
|
|
|
terminated = new Action(() =>
|
|
|
|
|
{
|
|
|
|
|
runtimeHost.Raise(h => h.ClientDisconnected += null);
|
|
|
|
|
process.Raise(p => p.Terminated += null, 0);
|
|
|
|
|
});
|
|
|
|
|
|
2019-06-11 09:53:33 +02:00
|
|
|
|
session.AppConfig = appConfig;
|
|
|
|
|
session.Settings = settings;
|
|
|
|
|
sessionContext.Current = session;
|
|
|
|
|
sessionContext.Next = session;
|
2018-03-21 15:28:59 +01:00
|
|
|
|
proxyFactory.Setup(f => f.CreateClientProxy(It.IsAny<string>())).Returns(proxy.Object);
|
|
|
|
|
|
2019-03-29 16:02:05 +01:00
|
|
|
|
sut = new ClientOperation(logger.Object, processFactory.Object, proxyFactory.Object, runtimeHost.Object, sessionContext, 0);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-03-22 15:41:25 +01:00
|
|
|
|
public void Perform_MustStartClient()
|
2018-03-21 15:28:59 +01:00
|
|
|
|
{
|
|
|
|
|
var result = default(OperationResult);
|
|
|
|
|
var response = new AuthenticationResponse { ProcessId = 1234 };
|
2018-08-10 13:23:24 +02:00
|
|
|
|
var communication = new CommunicationResult<AuthenticationResponse>(true, response);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
|
|
|
|
|
process.SetupGet(p => p.Id).Returns(response.ProcessId);
|
|
|
|
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process.Object).Callback(clientReady);
|
2018-08-10 13:23:24 +02:00
|
|
|
|
proxy.Setup(p => p.RequestAuthentication()).Returns(communication);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
proxy.Setup(p => p.Connect(It.IsAny<Guid>(), true)).Returns(true);
|
|
|
|
|
|
|
|
|
|
result = sut.Perform();
|
|
|
|
|
|
2018-10-12 15:23:43 +02:00
|
|
|
|
Assert.AreEqual(process.Object, sessionContext.ClientProcess);
|
|
|
|
|
Assert.AreEqual(proxy.Object, sessionContext.ClientProxy);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
Assert.AreEqual(OperationResult.Success, result);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-03-22 15:41:25 +01:00
|
|
|
|
public void Perform_MustFailStartupIfClientNotStartedWithinTimeout()
|
2018-03-21 15:28:59 +01:00
|
|
|
|
{
|
|
|
|
|
var result = default(OperationResult);
|
|
|
|
|
|
2019-03-22 15:41:25 +01:00
|
|
|
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process.Object);
|
|
|
|
|
|
2018-03-21 15:28:59 +01:00
|
|
|
|
result = sut.Perform();
|
|
|
|
|
|
2018-10-12 15:23:43 +02:00
|
|
|
|
Assert.IsNull(sessionContext.ClientProxy);
|
2019-03-22 15:41:25 +01:00
|
|
|
|
Assert.AreEqual(process.Object, sessionContext.ClientProcess);
|
|
|
|
|
Assert.AreEqual(OperationResult.Failed, result);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void Perform_MustFailStartupImmediatelyIfClientTerminates()
|
|
|
|
|
{
|
|
|
|
|
const int ONE_SECOND = 1000;
|
|
|
|
|
|
|
|
|
|
var after = default(DateTime);
|
|
|
|
|
var before = default(DateTime);
|
|
|
|
|
var result = default(OperationResult);
|
|
|
|
|
var terminateClient = new Action(() => Task.Delay(100).ContinueWith(_ => process.Raise(p => p.Terminated += null, 0)));
|
|
|
|
|
|
|
|
|
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process.Object).Callback(terminateClient);
|
2019-03-29 16:02:05 +01:00
|
|
|
|
sut = new ClientOperation(logger.Object, processFactory.Object, proxyFactory.Object, runtimeHost.Object, sessionContext, ONE_SECOND);
|
2019-03-22 15:41:25 +01:00
|
|
|
|
|
|
|
|
|
before = DateTime.Now;
|
|
|
|
|
result = sut.Perform();
|
|
|
|
|
after = DateTime.Now;
|
|
|
|
|
|
|
|
|
|
Assert.IsTrue(after - before < new TimeSpan(0, 0, ONE_SECOND));
|
2018-03-21 15:28:59 +01:00
|
|
|
|
Assert.AreEqual(OperationResult.Failed, result);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-03-22 15:41:25 +01:00
|
|
|
|
public void Perform_MustFailStartupIfConnectionToClientNotEstablished()
|
2018-03-21 15:28:59 +01:00
|
|
|
|
{
|
|
|
|
|
var result = default(OperationResult);
|
|
|
|
|
|
|
|
|
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process.Object).Callback(clientReady);
|
|
|
|
|
proxy.Setup(p => p.Connect(It.IsAny<Guid>(), true)).Returns(false);
|
|
|
|
|
|
|
|
|
|
result = sut.Perform();
|
|
|
|
|
|
2018-10-12 15:23:43 +02:00
|
|
|
|
Assert.IsNotNull(sessionContext.ClientProcess);
|
|
|
|
|
Assert.IsNotNull(sessionContext.ClientProxy);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
Assert.AreEqual(OperationResult.Failed, result);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-03-22 15:41:25 +01:00
|
|
|
|
public void Perform_MustFailStartupIfAuthenticationNotSuccessful()
|
2018-03-21 15:28:59 +01:00
|
|
|
|
{
|
|
|
|
|
var result = default(OperationResult);
|
|
|
|
|
var response = new AuthenticationResponse { ProcessId = -1 };
|
2018-08-10 13:23:24 +02:00
|
|
|
|
var communication = new CommunicationResult<AuthenticationResponse>(true, response);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
|
|
|
|
|
process.SetupGet(p => p.Id).Returns(1234);
|
|
|
|
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process.Object).Callback(clientReady);
|
2018-08-10 13:23:24 +02:00
|
|
|
|
proxy.Setup(p => p.RequestAuthentication()).Returns(communication);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
proxy.Setup(p => p.Connect(It.IsAny<Guid>(), true)).Returns(true);
|
|
|
|
|
|
|
|
|
|
result = sut.Perform();
|
|
|
|
|
|
2018-10-12 15:23:43 +02:00
|
|
|
|
Assert.IsNotNull(sessionContext.ClientProcess);
|
|
|
|
|
Assert.IsNotNull(sessionContext.ClientProxy);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
Assert.AreEqual(OperationResult.Failed, result);
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-10 09:19:03 +02:00
|
|
|
|
[TestMethod]
|
2019-03-22 15:41:25 +01:00
|
|
|
|
public void Repeat_MustStartClient()
|
2018-10-10 09:19:03 +02:00
|
|
|
|
{
|
|
|
|
|
var result = default(OperationResult);
|
|
|
|
|
var response = new AuthenticationResponse { ProcessId = 1234 };
|
|
|
|
|
var communication = new CommunicationResult<AuthenticationResponse>(true, response);
|
|
|
|
|
|
|
|
|
|
process.SetupGet(p => p.Id).Returns(response.ProcessId);
|
|
|
|
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process.Object).Callback(clientReady);
|
|
|
|
|
proxy.Setup(p => p.RequestAuthentication()).Returns(communication);
|
|
|
|
|
proxy.Setup(p => p.Connect(It.IsAny<Guid>(), true)).Returns(true);
|
|
|
|
|
|
|
|
|
|
result = sut.Repeat();
|
|
|
|
|
|
2018-10-12 15:23:43 +02:00
|
|
|
|
Assert.AreEqual(process.Object, sessionContext.ClientProcess);
|
|
|
|
|
Assert.AreEqual(proxy.Object, sessionContext.ClientProxy);
|
2018-10-10 09:19:03 +02:00
|
|
|
|
Assert.AreEqual(OperationResult.Success, result);
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-21 15:28:59 +01:00
|
|
|
|
[TestMethod]
|
2019-03-22 15:41:25 +01:00
|
|
|
|
public void Revert_MustStopClient()
|
2018-03-21 15:28:59 +01:00
|
|
|
|
{
|
|
|
|
|
proxy.Setup(p => p.Disconnect()).Callback(terminated);
|
|
|
|
|
|
|
|
|
|
PerformNormally();
|
|
|
|
|
sut.Revert();
|
|
|
|
|
|
|
|
|
|
proxy.Verify(p => p.InitiateShutdown(), Times.Once);
|
|
|
|
|
proxy.Verify(p => p.Disconnect(), Times.Once);
|
|
|
|
|
process.Verify(p => p.Kill(), Times.Never);
|
2018-10-12 15:23:43 +02:00
|
|
|
|
|
|
|
|
|
Assert.IsNull(sessionContext.ClientProcess);
|
|
|
|
|
Assert.IsNull(sessionContext.ClientProxy);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-03-22 15:41:25 +01:00
|
|
|
|
public void Revert_MustKillClientIfStoppingFailed()
|
2018-03-21 15:28:59 +01:00
|
|
|
|
{
|
|
|
|
|
process.Setup(p => p.Kill()).Callback(() => process.SetupGet(p => p.HasTerminated).Returns(true));
|
|
|
|
|
|
|
|
|
|
PerformNormally();
|
|
|
|
|
sut.Revert();
|
|
|
|
|
|
|
|
|
|
process.Verify(p => p.Kill(), Times.AtLeastOnce);
|
2018-10-12 15:23:43 +02:00
|
|
|
|
|
|
|
|
|
Assert.IsNull(sessionContext.ClientProcess);
|
|
|
|
|
Assert.IsNull(sessionContext.ClientProxy);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-03-22 15:41:25 +01:00
|
|
|
|
public void Revert_MustAttemptToKillFiveTimesThenAbort()
|
2018-03-21 15:28:59 +01:00
|
|
|
|
{
|
|
|
|
|
PerformNormally();
|
|
|
|
|
sut.Revert();
|
|
|
|
|
|
|
|
|
|
process.Verify(p => p.Kill(), Times.Exactly(5));
|
2018-10-12 15:23:43 +02:00
|
|
|
|
|
|
|
|
|
Assert.IsNotNull(sessionContext.ClientProcess);
|
|
|
|
|
Assert.IsNotNull(sessionContext.ClientProxy);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
2019-03-22 15:41:25 +01:00
|
|
|
|
public void Revert_MustNotStopClientIfAlreadyTerminated()
|
2018-03-21 15:28:59 +01:00
|
|
|
|
{
|
|
|
|
|
process.SetupGet(p => p.HasTerminated).Returns(true);
|
|
|
|
|
|
|
|
|
|
sut.Revert();
|
|
|
|
|
|
|
|
|
|
proxy.Verify(p => p.InitiateShutdown(), Times.Never);
|
|
|
|
|
proxy.Verify(p => p.Disconnect(), Times.Never);
|
|
|
|
|
process.Verify(p => p.Kill(), Times.Never);
|
2018-10-12 15:23:43 +02:00
|
|
|
|
|
|
|
|
|
Assert.IsNull(sessionContext.ClientProcess);
|
|
|
|
|
Assert.IsNull(sessionContext.ClientProxy);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void PerformNormally()
|
|
|
|
|
{
|
|
|
|
|
var response = new AuthenticationResponse { ProcessId = 1234 };
|
2018-08-10 13:23:24 +02:00
|
|
|
|
var communication = new CommunicationResult<AuthenticationResponse>(true, response);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
|
|
|
|
|
process.SetupGet(p => p.Id).Returns(response.ProcessId);
|
|
|
|
|
processFactory.Setup(f => f.StartNew(It.IsAny<string>(), It.IsAny<string[]>())).Returns(process.Object).Callback(clientReady);
|
2018-08-10 13:23:24 +02:00
|
|
|
|
proxy.Setup(p => p.RequestAuthentication()).Returns(communication);
|
2018-03-21 15:28:59 +01:00
|
|
|
|
proxy.Setup(p => p.Connect(It.IsAny<Guid>(), true)).Returns(true);
|
|
|
|
|
|
|
|
|
|
sut.Perform();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|