SEBWIN-319: Implemented system configuration update for service operation and auto-restore mechanism.
This commit is contained in:
parent
2f510096d0
commit
39b63218fb
13 changed files with 182 additions and 43 deletions
|
@ -91,6 +91,7 @@ namespace SafeExamBrowser.Contracts.I18n
|
||||||
OperationStatus_StopWindowMonitoring,
|
OperationStatus_StopWindowMonitoring,
|
||||||
OperationStatus_TerminateBrowser,
|
OperationStatus_TerminateBrowser,
|
||||||
OperationStatus_TerminateShell,
|
OperationStatus_TerminateShell,
|
||||||
|
OperationStatus_UpdateSystemConfiguration,
|
||||||
OperationStatus_WaitExplorerStartup,
|
OperationStatus_WaitExplorerStartup,
|
||||||
OperationStatus_WaitExplorerTermination,
|
OperationStatus_WaitExplorerTermination,
|
||||||
OperationStatus_WaitRuntimeDisconnection,
|
OperationStatus_WaitRuntimeDisconnection,
|
||||||
|
|
|
@ -0,0 +1,26 @@
|
||||||
|
/*
|
||||||
|
* 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Contracts.Lockdown
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Provides functionality to update and enforce the system configuration.
|
||||||
|
/// </summary>
|
||||||
|
public interface ISystemConfigurationUpdate
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Executes the update synchronously.
|
||||||
|
/// </summary>
|
||||||
|
void Execute();
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Executes the udpate asynchronously.
|
||||||
|
/// </summary>
|
||||||
|
void ExecuteAsync();
|
||||||
|
}
|
||||||
|
}
|
|
@ -96,6 +96,7 @@
|
||||||
<Compile Include="Lockdown\IFeatureConfiguration.cs" />
|
<Compile Include="Lockdown\IFeatureConfiguration.cs" />
|
||||||
<Compile Include="Lockdown\IFeatureConfigurationBackup.cs" />
|
<Compile Include="Lockdown\IFeatureConfigurationBackup.cs" />
|
||||||
<Compile Include="Lockdown\IFeatureConfigurationFactory.cs" />
|
<Compile Include="Lockdown\IFeatureConfigurationFactory.cs" />
|
||||||
|
<Compile Include="Lockdown\ISystemConfigurationUpdate.cs" />
|
||||||
<Compile Include="Runtime\IRuntimeController.cs" />
|
<Compile Include="Runtime\IRuntimeController.cs" />
|
||||||
<Compile Include="Core\OperationModel\Events\ActionRequiredEventArgs.cs" />
|
<Compile Include="Core\OperationModel\Events\ActionRequiredEventArgs.cs" />
|
||||||
<Compile Include="Core\OperationModel\Events\ActionRequiredEventHandler.cs" />
|
<Compile Include="Core\OperationModel\Events\ActionRequiredEventHandler.cs" />
|
||||||
|
|
|
@ -231,6 +231,9 @@
|
||||||
<Entry key="OperationStatus_TerminateShell">
|
<Entry key="OperationStatus_TerminateShell">
|
||||||
Terminating user interface
|
Terminating user interface
|
||||||
</Entry>
|
</Entry>
|
||||||
|
<Entry key="OperationStatus_UpdateSystemConfiguration">
|
||||||
|
Updating system configuration
|
||||||
|
</Entry>
|
||||||
<Entry key="OperationStatus_WaitExplorerStartup">
|
<Entry key="OperationStatus_WaitExplorerStartup">
|
||||||
Waiting for Windows explorer to start up
|
Waiting for Windows explorer to start up
|
||||||
</Entry>
|
</Entry>
|
||||||
|
|
|
@ -21,6 +21,7 @@ namespace SafeExamBrowser.Lockdown.UnitTests
|
||||||
{
|
{
|
||||||
private Mock<IFeatureConfigurationBackup> backup;
|
private Mock<IFeatureConfigurationBackup> backup;
|
||||||
private Mock<ILogger> logger;
|
private Mock<ILogger> logger;
|
||||||
|
private Mock<ISystemConfigurationUpdate> systemConfigurationUpdate;
|
||||||
private AutoRestoreMechanism sut;
|
private AutoRestoreMechanism sut;
|
||||||
|
|
||||||
[TestInitialize]
|
[TestInitialize]
|
||||||
|
@ -28,8 +29,9 @@ namespace SafeExamBrowser.Lockdown.UnitTests
|
||||||
{
|
{
|
||||||
backup = new Mock<IFeatureConfigurationBackup>();
|
backup = new Mock<IFeatureConfigurationBackup>();
|
||||||
logger = new Mock<ILogger>();
|
logger = new Mock<ILogger>();
|
||||||
|
systemConfigurationUpdate = new Mock<ISystemConfigurationUpdate>();
|
||||||
|
|
||||||
sut = new AutoRestoreMechanism(backup.Object, logger.Object, 0);
|
sut = new AutoRestoreMechanism(backup.Object, logger.Object, systemConfigurationUpdate.Object, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
|
@ -78,17 +80,20 @@ namespace SafeExamBrowser.Lockdown.UnitTests
|
||||||
var list = new List<IFeatureConfiguration> { configuration.Object };
|
var list = new List<IFeatureConfiguration> { configuration.Object };
|
||||||
var sync = new AutoResetEvent(false);
|
var sync = new AutoResetEvent(false);
|
||||||
|
|
||||||
backup.Setup(b => b.GetAllConfigurations()).Returns(list).Callback(() => counter++);
|
backup.Setup(b => b.GetAllConfigurations()).Returns(() => new List<IFeatureConfiguration>(list)).Callback(() => counter++);
|
||||||
backup.Setup(b => b.Delete(It.IsAny<IFeatureConfiguration>())).Callback(() => { list.Clear(); sync.Set(); });
|
backup.Setup(b => b.Delete(It.IsAny<IFeatureConfiguration>())).Callback(() => list.Clear());
|
||||||
configuration.Setup(c => c.Restore()).Returns(() => counter == limit);
|
configuration.Setup(c => c.Restore()).Returns(() => counter >= limit);
|
||||||
|
systemConfigurationUpdate.Setup(u => u.ExecuteAsync()).Callback(() => sync.Set());
|
||||||
|
|
||||||
sut.Start();
|
sut.Start();
|
||||||
sync.WaitOne();
|
sync.WaitOne();
|
||||||
sut.Stop();
|
sut.Stop();
|
||||||
|
|
||||||
backup.Verify(b => b.GetAllConfigurations(), Times.Exactly(limit));
|
backup.Verify(b => b.GetAllConfigurations(), Times.Exactly(limit + 1));
|
||||||
backup.Verify(b => b.Delete(It.Is<IFeatureConfiguration>(c => c == configuration.Object)), Times.Once);
|
backup.Verify(b => b.Delete(It.Is<IFeatureConfiguration>(c => c == configuration.Object)), Times.Once);
|
||||||
configuration.Verify(c => c.Restore(), Times.Exactly(limit));
|
configuration.Verify(c => c.Restore(), Times.Exactly(limit));
|
||||||
|
systemConfigurationUpdate.Verify(u => u.Execute(), Times.Never);
|
||||||
|
systemConfigurationUpdate.Verify(u => u.ExecuteAsync(), Times.Once);
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
|
@ -103,7 +108,7 @@ namespace SafeExamBrowser.Lockdown.UnitTests
|
||||||
var list = new List<IFeatureConfiguration> { configuration.Object };
|
var list = new List<IFeatureConfiguration> { configuration.Object };
|
||||||
var sync = new AutoResetEvent(false);
|
var sync = new AutoResetEvent(false);
|
||||||
|
|
||||||
sut = new AutoRestoreMechanism(backup.Object, logger.Object, TIMEOUT);
|
sut = new AutoRestoreMechanism(backup.Object, logger.Object, systemConfigurationUpdate.Object, TIMEOUT);
|
||||||
|
|
||||||
backup.Setup(b => b.GetAllConfigurations()).Returns(list).Callback(() =>
|
backup.Setup(b => b.GetAllConfigurations()).Returns(list).Callback(() =>
|
||||||
{
|
{
|
||||||
|
@ -144,9 +149,10 @@ namespace SafeExamBrowser.Lockdown.UnitTests
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
|
[ExpectedException(typeof(ArgumentException))]
|
||||||
public void MustValidateTimeout()
|
public void MustValidateTimeout()
|
||||||
{
|
{
|
||||||
Assert.ThrowsException<ArgumentException>(() => new AutoRestoreMechanism(backup.Object, logger.Object, new Random().Next(int.MinValue, -1)));
|
new AutoRestoreMechanism(backup.Object, logger.Object, systemConfigurationUpdate.Object, new Random().Next(int.MinValue, -1));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,10 +20,15 @@ namespace SafeExamBrowser.Lockdown
|
||||||
|
|
||||||
private IFeatureConfigurationBackup backup;
|
private IFeatureConfigurationBackup backup;
|
||||||
private ILogger logger;
|
private ILogger logger;
|
||||||
|
private ISystemConfigurationUpdate systemConfigurationUpdate;
|
||||||
private bool running;
|
private bool running;
|
||||||
private int timeout_ms;
|
private int timeout_ms;
|
||||||
|
|
||||||
public AutoRestoreMechanism(IFeatureConfigurationBackup backup, ILogger logger, int timeout_ms)
|
public AutoRestoreMechanism(
|
||||||
|
IFeatureConfigurationBackup backup,
|
||||||
|
ILogger logger,
|
||||||
|
ISystemConfigurationUpdate systemConfigurationUpdate,
|
||||||
|
int timeout_ms)
|
||||||
{
|
{
|
||||||
if (timeout_ms < 0)
|
if (timeout_ms < 0)
|
||||||
{
|
{
|
||||||
|
@ -32,6 +37,7 @@ namespace SafeExamBrowser.Lockdown
|
||||||
|
|
||||||
this.backup = backup;
|
this.backup = backup;
|
||||||
this.logger = logger;
|
this.logger = logger;
|
||||||
|
this.systemConfigurationUpdate = systemConfigurationUpdate;
|
||||||
this.timeout_ms = timeout_ms;
|
this.timeout_ms = timeout_ms;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,15 +77,11 @@ namespace SafeExamBrowser.Lockdown
|
||||||
private void RestoreAll()
|
private void RestoreAll()
|
||||||
{
|
{
|
||||||
var configurations = backup.GetAllConfigurations();
|
var configurations = backup.GetAllConfigurations();
|
||||||
|
var all = configurations.Count;
|
||||||
|
var restored = 0;
|
||||||
|
|
||||||
if (!configurations.Any())
|
if (configurations.Any())
|
||||||
{
|
{
|
||||||
running = false;
|
|
||||||
logger.Info("Nothing to restore, stopped auto-restore mechanism.");
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
logger.Info($"Attempting to restore {configurations.Count} items...");
|
logger.Info($"Attempting to restore {configurations.Count} items...");
|
||||||
|
|
||||||
foreach (var configuration in configurations)
|
foreach (var configuration in configurations)
|
||||||
|
@ -89,6 +91,7 @@ namespace SafeExamBrowser.Lockdown
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
backup.Delete(configuration);
|
backup.Delete(configuration);
|
||||||
|
restored++;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -106,7 +109,21 @@ namespace SafeExamBrowser.Lockdown
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (all == restored)
|
||||||
|
{
|
||||||
|
systemConfigurationUpdate.ExecuteAsync();
|
||||||
|
}
|
||||||
|
|
||||||
Task.Delay(timeout_ms).ContinueWith((_) => RestoreAll());
|
Task.Delay(timeout_ms).ContinueWith((_) => RestoreAll());
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
lock (@lock)
|
||||||
|
{
|
||||||
|
running = false;
|
||||||
|
logger.Info("Nothing to restore, stopped auto-restore mechanism.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -75,6 +75,7 @@
|
||||||
<Compile Include="FeatureConfigurations\RegistryConfigurations\UserHive\VmwareOverlayConfiguration.cs" />
|
<Compile Include="FeatureConfigurations\RegistryConfigurations\UserHive\VmwareOverlayConfiguration.cs" />
|
||||||
<Compile Include="FeatureConfigurations\WindowsUpdateConfiguration.cs" />
|
<Compile Include="FeatureConfigurations\WindowsUpdateConfiguration.cs" />
|
||||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||||
|
<Compile Include="SystemConfigurationUpdate.cs" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="..\SafeExamBrowser.Contracts\SafeExamBrowser.Contracts.csproj">
|
<ProjectReference Include="..\SafeExamBrowser.Contracts\SafeExamBrowser.Contracts.csproj">
|
||||||
|
|
59
SafeExamBrowser.Lockdown/SystemConfigurationUpdate.cs
Normal file
59
SafeExamBrowser.Lockdown/SystemConfigurationUpdate.cs
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
/*
|
||||||
|
* 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 System.Diagnostics;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using SafeExamBrowser.Contracts.Lockdown;
|
||||||
|
using SafeExamBrowser.Contracts.Logging;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Lockdown
|
||||||
|
{
|
||||||
|
public class SystemConfigurationUpdate : ISystemConfigurationUpdate
|
||||||
|
{
|
||||||
|
private ILogger logger;
|
||||||
|
|
||||||
|
public SystemConfigurationUpdate(ILogger logger)
|
||||||
|
{
|
||||||
|
this.logger = logger;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Execute()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
logger.Info("Starting system configuration update...");
|
||||||
|
|
||||||
|
var process = Process.Start(new ProcessStartInfo("cmd.exe", "/c \"gpupdate /force\"")
|
||||||
|
{
|
||||||
|
CreateNoWindow = true,
|
||||||
|
RedirectStandardOutput = true,
|
||||||
|
UseShellExecute = false
|
||||||
|
});
|
||||||
|
|
||||||
|
logger.Info("Waiting for update to complete...");
|
||||||
|
process.WaitForExit();
|
||||||
|
|
||||||
|
var output = process.StandardOutput.ReadToEnd();
|
||||||
|
var lines = output.Split(new [] { Environment.NewLine, "\r", "\n" }, StringSplitOptions.RemoveEmptyEntries);
|
||||||
|
|
||||||
|
logger.Info($"Update has completed: {String.Join(" ", lines.Skip(1))}");
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
logger.Error("Failed to update system configuration!", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ExecuteAsync()
|
||||||
|
{
|
||||||
|
Task.Run(new Action(Execute));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -15,6 +15,7 @@ using SafeExamBrowser.Contracts.Communication.Proxies;
|
||||||
using SafeExamBrowser.Contracts.Configuration;
|
using SafeExamBrowser.Contracts.Configuration;
|
||||||
using SafeExamBrowser.Contracts.Configuration.Settings;
|
using SafeExamBrowser.Contracts.Configuration.Settings;
|
||||||
using SafeExamBrowser.Contracts.Core.OperationModel;
|
using SafeExamBrowser.Contracts.Core.OperationModel;
|
||||||
|
using SafeExamBrowser.Contracts.Lockdown;
|
||||||
using SafeExamBrowser.Contracts.Logging;
|
using SafeExamBrowser.Contracts.Logging;
|
||||||
using SafeExamBrowser.Contracts.SystemComponents;
|
using SafeExamBrowser.Contracts.SystemComponents;
|
||||||
using SafeExamBrowser.Contracts.UserInterface.MessageBox;
|
using SafeExamBrowser.Contracts.UserInterface.MessageBox;
|
||||||
|
@ -34,6 +35,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
private SessionConfiguration session;
|
private SessionConfiguration session;
|
||||||
private SessionContext sessionContext;
|
private SessionContext sessionContext;
|
||||||
private Settings settings;
|
private Settings settings;
|
||||||
|
private Mock<ISystemConfigurationUpdate> systemConfigurationUpdate;
|
||||||
private Mock<IUserInfo> userInfo;
|
private Mock<IUserInfo> userInfo;
|
||||||
private ServiceOperation sut;
|
private ServiceOperation sut;
|
||||||
|
|
||||||
|
@ -50,6 +52,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
session = new SessionConfiguration();
|
session = new SessionConfiguration();
|
||||||
sessionContext = new SessionContext();
|
sessionContext = new SessionContext();
|
||||||
settings = new Settings();
|
settings = new Settings();
|
||||||
|
systemConfigurationUpdate = new Mock<ISystemConfigurationUpdate>();
|
||||||
userInfo = new Mock<IUserInfo>();
|
userInfo = new Mock<IUserInfo>();
|
||||||
|
|
||||||
appConfig.ServiceEventName = serviceEventName;
|
appConfig.ServiceEventName = serviceEventName;
|
||||||
|
@ -60,7 +63,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
session.Settings = settings;
|
session.Settings = settings;
|
||||||
settings.Service.Policy = ServicePolicy.Mandatory;
|
settings.Service.Policy = ServicePolicy.Mandatory;
|
||||||
|
|
||||||
sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, 0, userInfo.Object);
|
sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, systemConfigurationUpdate.Object, 0, userInfo.Object);
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
|
@ -89,6 +92,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
var result = sut.Perform();
|
var result = sut.Perform();
|
||||||
|
|
||||||
service.Verify(s => s.StartSession(It.IsAny<ServiceConfiguration>()), Times.Once);
|
service.Verify(s => s.StartSession(It.IsAny<ServiceConfiguration>()), Times.Once);
|
||||||
|
systemConfigurationUpdate.Verify(u => u.Execute(), Times.Once);
|
||||||
userInfo.Verify(u => u.GetUserName(), Times.Once);
|
userInfo.Verify(u => u.GetUserName(), Times.Once);
|
||||||
userInfo.Verify(u => u.GetUserSid(), Times.Once);
|
userInfo.Verify(u => u.GetUserSid(), Times.Once);
|
||||||
|
|
||||||
|
@ -121,7 +125,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
service.Setup(s => s.Connect(null, true)).Returns(true);
|
service.Setup(s => s.Connect(null, true)).Returns(true);
|
||||||
service.Setup(s => s.StartSession(It.IsAny<ServiceConfiguration>())).Returns(new CommunicationResult(true));
|
service.Setup(s => s.StartSession(It.IsAny<ServiceConfiguration>())).Returns(new CommunicationResult(true));
|
||||||
|
|
||||||
sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, TIMEOUT, userInfo.Object);
|
sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, systemConfigurationUpdate.Object, TIMEOUT, userInfo.Object);
|
||||||
|
|
||||||
before = DateTime.Now;
|
before = DateTime.Now;
|
||||||
var result = sut.Perform();
|
var result = sut.Perform();
|
||||||
|
@ -154,6 +158,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
var result = sut.Perform();
|
var result = sut.Perform();
|
||||||
|
|
||||||
service.Verify(s => s.StartSession(It.IsAny<ServiceConfiguration>()), Times.Once);
|
service.Verify(s => s.StartSession(It.IsAny<ServiceConfiguration>()), Times.Once);
|
||||||
|
systemConfigurationUpdate.Verify(u => u.Execute(), Times.Never);
|
||||||
|
|
||||||
Assert.AreEqual(OperationResult.Failed, result);
|
Assert.AreEqual(OperationResult.Failed, result);
|
||||||
}
|
}
|
||||||
|
@ -207,8 +212,8 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
public void Repeat_MustStopCurrentAndStartNewSession()
|
public void Repeat_MustStopCurrentAndStartNewSession()
|
||||||
{
|
{
|
||||||
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(true)).Callback(() => serviceEvent.Set());
|
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(true)).Callback(() => serviceEvent.Set());
|
||||||
|
|
||||||
PerformNormally();
|
PerformNormally();
|
||||||
|
systemConfigurationUpdate.Reset();
|
||||||
|
|
||||||
var result = sut.Repeat();
|
var result = sut.Repeat();
|
||||||
|
|
||||||
|
@ -216,6 +221,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
service.Verify(s => s.StopSession(It.IsAny<Guid>()), Times.Once);
|
service.Verify(s => s.StopSession(It.IsAny<Guid>()), Times.Once);
|
||||||
service.Verify(s => s.StartSession(It.IsAny<ServiceConfiguration>()), Times.Exactly(2));
|
service.Verify(s => s.StartSession(It.IsAny<ServiceConfiguration>()), Times.Exactly(2));
|
||||||
service.Verify(s => s.Disconnect(), Times.Never);
|
service.Verify(s => s.Disconnect(), Times.Never);
|
||||||
|
systemConfigurationUpdate.Verify(u => u.Execute(), Times.Exactly(2));
|
||||||
|
|
||||||
Assert.AreEqual(OperationResult.Success, result);
|
Assert.AreEqual(OperationResult.Success, result);
|
||||||
}
|
}
|
||||||
|
@ -262,7 +268,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
var before = default(DateTime);
|
var before = default(DateTime);
|
||||||
|
|
||||||
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(true));
|
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(true));
|
||||||
sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, TIMEOUT, userInfo.Object);
|
sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, systemConfigurationUpdate.Object, TIMEOUT, userInfo.Object);
|
||||||
|
|
||||||
PerformNormally();
|
PerformNormally();
|
||||||
|
|
||||||
|
@ -297,13 +303,14 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
{
|
{
|
||||||
service.Setup(s => s.Disconnect()).Returns(true);
|
service.Setup(s => s.Disconnect()).Returns(true);
|
||||||
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(true)).Callback(() => serviceEvent.Set());
|
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(true)).Callback(() => serviceEvent.Set());
|
||||||
|
|
||||||
PerformNormally();
|
PerformNormally();
|
||||||
|
systemConfigurationUpdate.Reset();
|
||||||
|
|
||||||
var result = sut.Revert();
|
var result = sut.Revert();
|
||||||
|
|
||||||
service.Verify(s => s.StopSession(It.IsAny<Guid>()), Times.Once);
|
service.Verify(s => s.StopSession(It.IsAny<Guid>()), Times.Once);
|
||||||
service.Verify(s => s.Disconnect(), Times.Once);
|
service.Verify(s => s.Disconnect(), Times.Once);
|
||||||
|
systemConfigurationUpdate.Verify(u => u.Execute(), Times.Once);
|
||||||
|
|
||||||
Assert.AreEqual(OperationResult.Success, result);
|
Assert.AreEqual(OperationResult.Success, result);
|
||||||
}
|
}
|
||||||
|
@ -312,13 +319,14 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
public void Revert_MustHandleCommunicationFailureWhenStoppingSession()
|
public void Revert_MustHandleCommunicationFailureWhenStoppingSession()
|
||||||
{
|
{
|
||||||
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(false));
|
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(false));
|
||||||
|
|
||||||
PerformNormally();
|
PerformNormally();
|
||||||
|
systemConfigurationUpdate.Reset();
|
||||||
|
|
||||||
var result = sut.Revert();
|
var result = sut.Revert();
|
||||||
|
|
||||||
service.Verify(s => s.StopSession(It.IsAny<Guid>()), Times.Once);
|
service.Verify(s => s.StopSession(It.IsAny<Guid>()), Times.Once);
|
||||||
service.Verify(s => s.Disconnect(), Times.Once);
|
service.Verify(s => s.Disconnect(), Times.Once);
|
||||||
|
systemConfigurationUpdate.Verify(u => u.Execute(), Times.Never);
|
||||||
|
|
||||||
Assert.AreEqual(OperationResult.Failed, result);
|
Assert.AreEqual(OperationResult.Failed, result);
|
||||||
}
|
}
|
||||||
|
@ -344,7 +352,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
|
||||||
var before = default(DateTime);
|
var before = default(DateTime);
|
||||||
|
|
||||||
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(true));
|
service.Setup(s => s.StopSession(It.IsAny<Guid>())).Returns(new CommunicationResult(true));
|
||||||
sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, TIMEOUT, userInfo.Object);
|
sut = new ServiceOperation(logger.Object, runtimeHost.Object, service.Object, sessionContext, systemConfigurationUpdate.Object, TIMEOUT, userInfo.Object);
|
||||||
|
|
||||||
PerformNormally();
|
PerformNormally();
|
||||||
|
|
||||||
|
|
|
@ -27,6 +27,7 @@ using SafeExamBrowser.Contracts.SystemComponents;
|
||||||
using SafeExamBrowser.Core.OperationModel;
|
using SafeExamBrowser.Core.OperationModel;
|
||||||
using SafeExamBrowser.Core.Operations;
|
using SafeExamBrowser.Core.Operations;
|
||||||
using SafeExamBrowser.I18n;
|
using SafeExamBrowser.I18n;
|
||||||
|
using SafeExamBrowser.Lockdown;
|
||||||
using SafeExamBrowser.Logging;
|
using SafeExamBrowser.Logging;
|
||||||
using SafeExamBrowser.Runtime.Communication;
|
using SafeExamBrowser.Runtime.Communication;
|
||||||
using SafeExamBrowser.Runtime.Operations;
|
using SafeExamBrowser.Runtime.Operations;
|
||||||
|
@ -70,6 +71,7 @@ namespace SafeExamBrowser.Runtime
|
||||||
var runtimeHost = new RuntimeHost(appConfig.RuntimeAddress, new HostObjectFactory(), ModuleLogger(nameof(RuntimeHost)), FIVE_SECONDS);
|
var runtimeHost = new RuntimeHost(appConfig.RuntimeAddress, new HostObjectFactory(), ModuleLogger(nameof(RuntimeHost)), FIVE_SECONDS);
|
||||||
var serviceProxy = new ServiceProxy(appConfig.ServiceAddress, new ProxyObjectFactory(), ModuleLogger(nameof(ServiceProxy)), Interlocutor.Runtime);
|
var serviceProxy = new ServiceProxy(appConfig.ServiceAddress, new ProxyObjectFactory(), ModuleLogger(nameof(ServiceProxy)), Interlocutor.Runtime);
|
||||||
var sessionContext = new SessionContext();
|
var sessionContext = new SessionContext();
|
||||||
|
var systemConfigurationUpdate = new SystemConfigurationUpdate(ModuleLogger(nameof(SystemConfigurationUpdate)));
|
||||||
var uiFactory = new UserInterfaceFactory(text);
|
var uiFactory = new UserInterfaceFactory(text);
|
||||||
var userInfo = new UserInfo();
|
var userInfo = new UserInfo();
|
||||||
|
|
||||||
|
@ -81,7 +83,7 @@ namespace SafeExamBrowser.Runtime
|
||||||
|
|
||||||
sessionOperations.Enqueue(new SessionInitializationOperation(configuration, logger, runtimeHost, sessionContext));
|
sessionOperations.Enqueue(new SessionInitializationOperation(configuration, logger, runtimeHost, sessionContext));
|
||||||
sessionOperations.Enqueue(new ConfigurationOperation(args, configuration, new HashAlgorithm(), logger, sessionContext));
|
sessionOperations.Enqueue(new ConfigurationOperation(args, configuration, new HashAlgorithm(), logger, sessionContext));
|
||||||
sessionOperations.Enqueue(new ServiceOperation(logger, runtimeHost, serviceProxy, sessionContext, THIRTY_SECONDS, userInfo));
|
sessionOperations.Enqueue(new ServiceOperation(logger, runtimeHost, serviceProxy, sessionContext, systemConfigurationUpdate, THIRTY_SECONDS, userInfo));
|
||||||
sessionOperations.Enqueue(new ClientTerminationOperation(logger, processFactory, proxyFactory, runtimeHost, sessionContext, THIRTY_SECONDS));
|
sessionOperations.Enqueue(new ClientTerminationOperation(logger, processFactory, proxyFactory, runtimeHost, sessionContext, THIRTY_SECONDS));
|
||||||
sessionOperations.Enqueue(new KioskModeOperation(desktopFactory, explorerShell, logger, processFactory, sessionContext));
|
sessionOperations.Enqueue(new KioskModeOperation(desktopFactory, explorerShell, logger, processFactory, sessionContext));
|
||||||
sessionOperations.Enqueue(new ClientOperation(logger, processFactory, proxyFactory, runtimeHost, sessionContext, THIRTY_SECONDS));
|
sessionOperations.Enqueue(new ClientOperation(logger, processFactory, proxyFactory, runtimeHost, sessionContext, THIRTY_SECONDS));
|
||||||
|
|
|
@ -16,6 +16,7 @@ using SafeExamBrowser.Contracts.Configuration.Settings;
|
||||||
using SafeExamBrowser.Contracts.Core.OperationModel;
|
using SafeExamBrowser.Contracts.Core.OperationModel;
|
||||||
using SafeExamBrowser.Contracts.Core.OperationModel.Events;
|
using SafeExamBrowser.Contracts.Core.OperationModel.Events;
|
||||||
using SafeExamBrowser.Contracts.I18n;
|
using SafeExamBrowser.Contracts.I18n;
|
||||||
|
using SafeExamBrowser.Contracts.Lockdown;
|
||||||
using SafeExamBrowser.Contracts.Logging;
|
using SafeExamBrowser.Contracts.Logging;
|
||||||
using SafeExamBrowser.Contracts.SystemComponents;
|
using SafeExamBrowser.Contracts.SystemComponents;
|
||||||
using SafeExamBrowser.Contracts.UserInterface.MessageBox;
|
using SafeExamBrowser.Contracts.UserInterface.MessageBox;
|
||||||
|
@ -28,6 +29,7 @@ namespace SafeExamBrowser.Runtime.Operations
|
||||||
private ILogger logger;
|
private ILogger logger;
|
||||||
private IRuntimeHost runtimeHost;
|
private IRuntimeHost runtimeHost;
|
||||||
private IServiceProxy service;
|
private IServiceProxy service;
|
||||||
|
private ISystemConfigurationUpdate systemConfigurationUpdate;
|
||||||
private int timeout_ms;
|
private int timeout_ms;
|
||||||
private IUserInfo userInfo;
|
private IUserInfo userInfo;
|
||||||
|
|
||||||
|
@ -39,12 +41,14 @@ namespace SafeExamBrowser.Runtime.Operations
|
||||||
IRuntimeHost runtimeHost,
|
IRuntimeHost runtimeHost,
|
||||||
IServiceProxy service,
|
IServiceProxy service,
|
||||||
SessionContext sessionContext,
|
SessionContext sessionContext,
|
||||||
|
ISystemConfigurationUpdate systemConfigurationUpdate,
|
||||||
int timeout_ms,
|
int timeout_ms,
|
||||||
IUserInfo userInfo) : base(sessionContext)
|
IUserInfo userInfo) : base(sessionContext)
|
||||||
{
|
{
|
||||||
this.logger = logger;
|
this.logger = logger;
|
||||||
this.runtimeHost = runtimeHost;
|
this.runtimeHost = runtimeHost;
|
||||||
this.service = service;
|
this.service = service;
|
||||||
|
this.systemConfigurationUpdate = systemConfigurationUpdate;
|
||||||
this.timeout_ms = timeout_ms;
|
this.timeout_ms = timeout_ms;
|
||||||
this.userInfo = userInfo;
|
this.userInfo = userInfo;
|
||||||
}
|
}
|
||||||
|
@ -188,6 +192,9 @@ namespace SafeExamBrowser.Runtime.Operations
|
||||||
{
|
{
|
||||||
logger.Error($"Failed to start new service session within {timeout_ms / 1000} seconds!");
|
logger.Error($"Failed to start new service session within {timeout_ms / 1000} seconds!");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
StatusChanged?.Invoke(TextKey.OperationStatus_UpdateSystemConfiguration);
|
||||||
|
systemConfigurationUpdate.Execute();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -217,6 +224,9 @@ namespace SafeExamBrowser.Runtime.Operations
|
||||||
{
|
{
|
||||||
logger.Error($"Failed to stop service session within {timeout_ms / 1000} seconds!");
|
logger.Error($"Failed to stop service session within {timeout_ms / 1000} seconds!");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
StatusChanged?.Invoke(TextKey.OperationStatus_UpdateSystemConfiguration);
|
||||||
|
systemConfigurationUpdate.Execute();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
|
@ -155,6 +155,10 @@
|
||||||
<Project>{10c62628-8e6a-45aa-9d97-339b119ad21d}</Project>
|
<Project>{10c62628-8e6a-45aa-9d97-339b119ad21d}</Project>
|
||||||
<Name>SafeExamBrowser.I18n</Name>
|
<Name>SafeExamBrowser.I18n</Name>
|
||||||
</ProjectReference>
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\SafeExamBrowser.Lockdown\SafeExamBrowser.Lockdown.csproj">
|
||||||
|
<Project>{386b6042-3e12-4753-9fc6-c88ea4f97030}</Project>
|
||||||
|
<Name>SafeExamBrowser.Lockdown</Name>
|
||||||
|
</ProjectReference>
|
||||||
<ProjectReference Include="..\SafeExamBrowser.Logging\SafeExamBrowser.Logging.csproj">
|
<ProjectReference Include="..\SafeExamBrowser.Logging\SafeExamBrowser.Logging.csproj">
|
||||||
<Project>{e107026c-2011-4552-a7d8-3a0d37881df6}</Project>
|
<Project>{e107026c-2011-4552-a7d8-3a0d37881df6}</Project>
|
||||||
<Name>SafeExamBrowser.Logging</Name>
|
<Name>SafeExamBrowser.Logging</Name>
|
||||||
|
|
|
@ -46,11 +46,12 @@ namespace SafeExamBrowser.Service
|
||||||
var proxyFactory = new ProxyFactory(new ProxyObjectFactory(), new ModuleLogger(logger, nameof(ProxyFactory)));
|
var proxyFactory = new ProxyFactory(new ProxyObjectFactory(), new ModuleLogger(logger, nameof(ProxyFactory)));
|
||||||
var serviceHost = new ServiceHost(AppConfig.SERVICE_ADDRESS, new HostObjectFactory(), new ModuleLogger(logger, nameof(ServiceHost)), FIVE_SECONDS);
|
var serviceHost = new ServiceHost(AppConfig.SERVICE_ADDRESS, new HostObjectFactory(), new ModuleLogger(logger, nameof(ServiceHost)), FIVE_SECONDS);
|
||||||
var sessionContext = new SessionContext();
|
var sessionContext = new SessionContext();
|
||||||
|
var systemConfigurationUpdate = new SystemConfigurationUpdate(new ModuleLogger(logger, nameof(SystemConfigurationUpdate)));
|
||||||
|
|
||||||
var bootstrapOperations = new Queue<IOperation>();
|
var bootstrapOperations = new Queue<IOperation>();
|
||||||
var sessionOperations = new Queue<IOperation>();
|
var sessionOperations = new Queue<IOperation>();
|
||||||
|
|
||||||
sessionContext.AutoRestoreMechanism = new AutoRestoreMechanism(featureBackup, new ModuleLogger(logger, nameof(AutoRestoreMechanism)), FIVE_SECONDS);
|
sessionContext.AutoRestoreMechanism = new AutoRestoreMechanism(featureBackup, new ModuleLogger(logger, nameof(AutoRestoreMechanism)), systemConfigurationUpdate, FIVE_SECONDS);
|
||||||
|
|
||||||
bootstrapOperations.Enqueue(new RestoreOperation(featureBackup, logger, sessionContext));
|
bootstrapOperations.Enqueue(new RestoreOperation(featureBackup, logger, sessionContext));
|
||||||
bootstrapOperations.Enqueue(new CommunicationHostOperation(serviceHost, logger));
|
bootstrapOperations.Enqueue(new CommunicationHostOperation(serviceHost, logger));
|
||||||
|
|
Loading…
Reference in a new issue