Compare commits

..

No commits in common. "master" and "3.7.0" have entirely different histories.

183 changed files with 2998 additions and 4243 deletions

View file

@ -21,5 +21,5 @@ jobs:
close-issue-message: "This issue was closed because it has been inactive for 14 days since being marked as stale." close-issue-message: "This issue was closed because it has been inactive for 14 days since being marked as stale."
days-before-pr-stale: -1 days-before-pr-stale: -1
days-before-pr-close: -1 days-before-pr-close: -1
exempt-issue-labels: "bug,enhancement,feature request,known issue" exempt-issue-labels: "bug,enhancement,feature request"
repo-token: ${{ secrets.GITHUB_TOKEN }} repo-token: ${{ secrets.GITHUB_TOKEN }}

BIN
Libraries/SimpleWifi.dll Normal file

Binary file not shown.

View file

@ -7,6 +7,7 @@ Refactored version of Safe Exam Browser for Windows with Chromium as integrated
SEB 3.x requires the prerequisites listed below in order to work correctly. These are automatically installed with the setup bundle and need only be manually installed when using the MSI packages. SEB 3.x requires the prerequisites listed below in order to work correctly. These are automatically installed with the setup bundle and need only be manually installed when using the MSI packages.
* .NET Framework 4.8 Runtime: https://dotnet.microsoft.com/download/dotnet-framework/net48 * .NET Framework 4.8 Runtime: https://dotnet.microsoft.com/download/dotnet-framework/net48
* Microsoft Edge WebView2 Runtime: https://go.microsoft.com/fwlink/p/?LinkId=2124703
* Visual C++ 2015-2019 Redistributable: https://support.microsoft.com/en-us/help/2977003/the-latest-supported-visual-c-downloads * Visual C++ 2015-2019 Redistributable: https://support.microsoft.com/en-us/help/2977003/the-latest-supported-visual-c-downloads
## Project Status ## Project Status
@ -16,7 +17,7 @@ SEB 3.x requires the prerequisites listed below in order to work correctly. Thes
| Aspect | Status | Details | | Aspect | Status | Details |
| ----------------- | --------------------------------------------------------------------------------------------------------------------- | --------------------------------------------------------------- | | ----------------- | --------------------------------------------------------------------------------------------------------------------- | --------------------------------------------------------------- |
| Development Build | ![Development Build Status](https://sebdev.ethz.ch/api/projects/status/kq78qrjtnpk82ti0?svg=true) | https://sebdev.ethz.ch/project/appveyor/seb-win-refactoring | | Development Build | ![Development Build Status](https://sebdev-let.ethz.ch/api/projects/status/kq78qrjtnpk82ti0?svg=true) | https://sebdev-let.ethz.ch/project/appveyor/seb-win-refactoring |
| Test Build | ![Test Build Status](https://ci.appveyor.com/api/projects/status/a56akt9r174570m7?svg=true) | https://ci.appveyor.com/project/dbuechel/seb-win-refactoring | | Test Build | ![Test Build Status](https://ci.appveyor.com/api/projects/status/a56akt9r174570m7?svg=true) | https://ci.appveyor.com/project/dbuechel/seb-win-refactoring |
| Test Run | ![AppVeyor Tests](https://img.shields.io/appveyor/tests/dbuechel/seb-win-refactoring?logo=appveyor&logoColor=%23ccc) | https://ci.appveyor.com/project/dbuechel/seb-win-refactoring | | Test Run | ![AppVeyor Tests](https://img.shields.io/appveyor/tests/dbuechel/seb-win-refactoring?logo=appveyor&logoColor=%23ccc) | https://ci.appveyor.com/project/dbuechel/seb-win-refactoring |
| Code Coverage | ![Code Coverage](https://codecov.io/gh/SafeExamBrowser/seb-win-refactoring/branch/master/graph/badge.svg) | https://codecov.io/gh/SafeExamBrowser/seb-win-refactoring | | Code Coverage | ![Code Coverage](https://codecov.io/gh/SafeExamBrowser/seb-win-refactoring/branch/master/graph/badge.svg) | https://codecov.io/gh/SafeExamBrowser/seb-win-refactoring |

View file

@ -1,35 +0,0 @@
# Security Policy
We only support the latest official relese version with respect to security vulnerabilities. Thus, only the latest or then the upcoming next release version
will receive vulnerability fixes and security updates. A vulnerability may however be reported for any version, unless it already has been fixed with a later
release version.
## Reporting a Vulnerability
> [!IMPORTANT]
> - Please _always_ verify that no later release version exists which fixes the vulnerability.
> - Please _always_ consult the documentation first before creating a vulnerability report: https://safeexambrowser.org/windows/win_usermanual_en.html.
> - Please _always_ attach the log file(s) of the affected session(s)! They can be found under `%LocalAppData%\SafeExamBrowser\Logs`.
**Describe the Vulnerability**
A clear and concise description of what the vulnerability is.
**Steps to Reproduce**
Steps to reproduce the behavior:
1. Go to '...'
2. Click on '....'
3. Scroll down to '....'
4. See ...
**Expected Behavior**
A clear and concise description of what you expected to happen.
**Screenshots**
If applicable, add screenshots to help explain your problem.
**Version Information**
- OS: [e.g. Windows 10 Professional, Version 1803]
- SEB-Version [e.g. SEB 3.0.1]
**Additional Context**
Add any other context about the vulnerability here.

View file

@ -6,6 +6,8 @@
<Import Project="..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props" Condition="Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props')" /> <Import Project="..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props" Condition="Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props')" />
<Import Project="..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props" Condition="Exists('..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props')" /> <Import Project="..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props" Condition="Exists('..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props')" />
<Import Project="..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props" Condition="Exists('..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props')" /> <Import Project="..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props" Condition="Exists('..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props')" />
<Import Project="..\packages\cef.redist.x86.120.2.7\build\cef.redist.x86.props" Condition="Exists('..\packages\cef.redist.x86.120.2.7\build\cef.redist.x86.props')" />
<Import Project="..\packages\cef.redist.x64.120.2.7\build\cef.redist.x64.props" Condition="Exists('..\packages\cef.redist.x64.120.2.7\build\cef.redist.x64.props')" />
<Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" /> <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
<PropertyGroup> <PropertyGroup>
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration> <Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
@ -204,6 +206,8 @@
<PropertyGroup> <PropertyGroup>
<ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText> <ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText>
</PropertyGroup> </PropertyGroup>
<Error Condition="!Exists('..\packages\cef.redist.x64.120.2.7\build\cef.redist.x64.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\cef.redist.x64.120.2.7\build\cef.redist.x64.props'))" />
<Error Condition="!Exists('..\packages\cef.redist.x86.120.2.7\build\cef.redist.x86.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\cef.redist.x86.120.2.7\build\cef.redist.x86.props'))" />
<Error Condition="!Exists('..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props'))" /> <Error Condition="!Exists('..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\chromiumembeddedframework.runtime.win-x64.121.3.13\build\chromiumembeddedframework.runtime.win-x64.props'))" />
<Error Condition="!Exists('..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props'))" /> <Error Condition="!Exists('..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\chromiumembeddedframework.runtime.win-x86.121.3.13\build\chromiumembeddedframework.runtime.win-x86.props'))" />
<Error Condition="!Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props'))" /> <Error Condition="!Exists('..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\CefSharp.Common.121.3.130\build\CefSharp.Common.props'))" />

View file

@ -1,6 +1,8 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<packages> <packages>
<package id="Castle.Core" version="5.1.1" targetFramework="net48" /> <package id="Castle.Core" version="5.1.1" targetFramework="net48" />
<package id="cef.redist.x64" version="120.2.7" targetFramework="net48" />
<package id="cef.redist.x86" version="120.2.7" targetFramework="net48" />
<package id="CefSharp.Common" version="121.3.130" targetFramework="net48" /> <package id="CefSharp.Common" version="121.3.130" targetFramework="net48" />
<package id="chromiumembeddedframework.runtime.win-x64" version="121.3.13" targetFramework="net48" /> <package id="chromiumembeddedframework.runtime.win-x64" version="121.3.13" targetFramework="net48" />
<package id="chromiumembeddedframework.runtime.win-x86" version="121.3.13" targetFramework="net48" /> <package id="chromiumembeddedframework.runtime.win-x86" version="121.3.13" targetFramework="net48" />

View file

@ -18,7 +18,6 @@ namespace SafeExamBrowser.Browser.Content
private string api; private string api;
private string clipboard; private string clipboard;
private string pageZoom;
internal ContentLoader(IText text) internal ContentLoader(IText text)
{ {
@ -98,22 +97,5 @@ namespace SafeExamBrowser.Browser.Content
return clipboard; return clipboard;
} }
internal string LoadPageZoom()
{
if (pageZoom == default)
{
var assembly = Assembly.GetAssembly(typeof(ContentLoader));
var path = $"{typeof(ContentLoader).Namespace}.PageZoom.js";
using (var stream = assembly.GetManifestResourceStream(path))
using (var reader = new StreamReader(stream))
{
pageZoom = reader.ReadToEnd();
}
}
return pageZoom;
}
} }
} }

View file

@ -1,16 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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/.
*/
function disableMouseWheelZoom(e) {
if (e.ctrlKey) {
e.preventDefault();
e.stopPropagation();
}
}
document.addEventListener('wheel', disableMouseWheelZoom, { passive: false });

View file

@ -138,11 +138,6 @@ namespace SafeExamBrowser.Browser.Handlers
filePath = Path.Combine(KnownFolders.Downloads.ExpandedPath, downloadItem.SuggestedFileName); filePath = Path.Combine(KnownFolders.Downloads.ExpandedPath, downloadItem.SuggestedFileName);
} }
if (File.Exists(filePath))
{
filePath = AppendIndexSuffixTo(filePath);
}
if (showDialog) if (showDialog)
{ {
logger.Debug($"Allowing user to select custom download location, with '{filePath}' as suggestion."); logger.Debug($"Allowing user to select custom download location, with '{filePath}' as suggestion.");
@ -160,26 +155,6 @@ namespace SafeExamBrowser.Browser.Handlers
} }
} }
private string AppendIndexSuffixTo(string filePath)
{
var directory = Path.GetDirectoryName(filePath);
var extension = Path.GetExtension(filePath);
var name = Path.GetFileNameWithoutExtension(filePath);
var path = default(string);
for (var suffix = 1; suffix < int.MaxValue; suffix++)
{
path = Path.Combine(directory, $"{name}({suffix}){extension}");
if (!File.Exists(path))
{
break;
}
}
return path;
}
private void RequestConfigurationFileDownload(DownloadItem downloadItem, IBeforeDownloadCallback callback) private void RequestConfigurationFileDownload(DownloadItem downloadItem, IBeforeDownloadCallback callback)
{ {
var args = new DownloadEventArgs { Url = downloadItem.Url }; var args = new DownloadEventArgs { Url = downloadItem.Url };

View file

@ -40,15 +40,9 @@ namespace SafeExamBrowser.Browser.Handlers
var configurationKey = keyGenerator.CalculateConfigurationKeyHash(settings.ConfigurationKey, frame.Url); var configurationKey = keyGenerator.CalculateConfigurationKeyHash(settings.ConfigurationKey, frame.Url);
var api = contentLoader.LoadApi(browserExamKey, configurationKey, appConfig.ProgramBuildVersion); var api = contentLoader.LoadApi(browserExamKey, configurationKey, appConfig.ProgramBuildVersion);
var clipboardScript = contentLoader.LoadClipboard(); var clipboardScript = contentLoader.LoadClipboard();
var pageZoomScript = contentLoader.LoadPageZoom();
frame.ExecuteJavaScriptAsync(api); frame.ExecuteJavaScriptAsync(api);
if (!settings.AllowPageZoom)
{
frame.ExecuteJavaScriptAsync(pageZoomScript);
}
if (!settings.AllowPrint) if (!settings.AllowPrint)
{ {
frame.ExecuteJavaScriptAsync($"window.print = function() {{ alert('{text.Get(TextKey.Browser_PrintNotAllowed)}') }}"); frame.ExecuteJavaScriptAsync($"window.print = function() {{ alert('{text.Get(TextKey.Browser_PrintNotAllowed)}') }}");

View file

@ -396,7 +396,6 @@ namespace SafeExamBrowser.Browser.Handlers
var endpointUrl = default(string); var endpointUrl = default(string);
var start = session.IndexOf("=") + 1; var start = session.IndexOf("=") + 1;
var end = session.IndexOf(";"); var end = session.IndexOf(";");
var name = session.Substring(0, start - 1);
var value = session.Substring(start, end - start); var value = session.Substring(start, end - start);
var uri = new Uri(requestUrl); var uri = new Uri(requestUrl);
@ -414,7 +413,7 @@ namespace SafeExamBrowser.Browser.Handlers
using (var handler = new HttpClientHandler { UseCookies = false }) using (var handler = new HttpClientHandler { UseCookies = false })
using (var client = new HttpClient(handler)) using (var client = new HttpClient(handler))
{ {
message.Headers.Add("Cookie", $"{name}={value}"); message.Headers.Add("Cookie", $"MoodleSession={value}");
var result = await client.SendAsync(message); var result = await client.SendAsync(message);

View file

@ -194,9 +194,7 @@
<ItemGroup> <ItemGroup>
<EmbeddedResource Include="Content\Clipboard.js" /> <EmbeddedResource Include="Content\Clipboard.js" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup />
<EmbeddedResource Include="Content\PageZoom.js" />
</ItemGroup>
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" /> <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
<PropertyGroup> <PropertyGroup>
<PostBuildEvent> <PostBuildEvent>

View file

@ -14,7 +14,6 @@ using Moq;
using SafeExamBrowser.Applications.Contracts; using SafeExamBrowser.Applications.Contracts;
using SafeExamBrowser.Browser.Contracts; using SafeExamBrowser.Browser.Contracts;
using SafeExamBrowser.Browser.Contracts.Events; using SafeExamBrowser.Browser.Contracts.Events;
using SafeExamBrowser.Client.Contracts;
using SafeExamBrowser.Client.Operations.Events; using SafeExamBrowser.Client.Operations.Events;
using SafeExamBrowser.Communication.Contracts.Data; using SafeExamBrowser.Communication.Contracts.Data;
using SafeExamBrowser.Communication.Contracts.Events; using SafeExamBrowser.Communication.Contracts.Events;
@ -34,8 +33,7 @@ using SafeExamBrowser.Server.Contracts;
using SafeExamBrowser.Server.Contracts.Data; using SafeExamBrowser.Server.Contracts.Data;
using SafeExamBrowser.Settings; using SafeExamBrowser.Settings;
using SafeExamBrowser.Settings.Monitoring; using SafeExamBrowser.Settings.Monitoring;
using SafeExamBrowser.Settings.UserInterface; using SafeExamBrowser.SystemComponents.Contracts.Registry;
using SafeExamBrowser.SystemComponents.Contracts.Network;
using SafeExamBrowser.UserInterface.Contracts; using SafeExamBrowser.UserInterface.Contracts;
using SafeExamBrowser.UserInterface.Contracts.FileSystemDialog; using SafeExamBrowser.UserInterface.Contracts.FileSystemDialog;
using SafeExamBrowser.UserInterface.Contracts.MessageBox; using SafeExamBrowser.UserInterface.Contracts.MessageBox;
@ -56,7 +54,6 @@ namespace SafeExamBrowser.Client.UnitTests
private Mock<IBrowserApplication> browser; private Mock<IBrowserApplication> browser;
private Mock<IClientHost> clientHost; private Mock<IClientHost> clientHost;
private ClientContext context; private ClientContext context;
private Mock<ICoordinator> coordinator;
private Mock<IDisplayMonitor> displayMonitor; private Mock<IDisplayMonitor> displayMonitor;
private Mock<IExplorerShell> explorerShell; private Mock<IExplorerShell> explorerShell;
private Mock<IFileSystemDialog> fileSystemDialog; private Mock<IFileSystemDialog> fileSystemDialog;
@ -64,15 +61,15 @@ namespace SafeExamBrowser.Client.UnitTests
private Mock<IIntegrityModule> integrityModule; private Mock<IIntegrityModule> integrityModule;
private Mock<ILogger> logger; private Mock<ILogger> logger;
private Mock<IMessageBox> messageBox; private Mock<IMessageBox> messageBox;
private Mock<INetworkAdapter> networkAdapter;
private Mock<IOperationSequence> operationSequence; private Mock<IOperationSequence> operationSequence;
private Mock<IRegistry> registry;
private Mock<IRuntimeProxy> runtimeProxy; private Mock<IRuntimeProxy> runtimeProxy;
private Mock<ISystemSentinel> sentinel;
private Mock<IServerProxy> server; private Mock<IServerProxy> server;
private Guid sessionId; private Guid sessionId;
private AppSettings settings; private AppSettings settings;
private Mock<Action> shutdown; private Mock<Action> shutdown;
private Mock<ISplashScreen> splashScreen; private Mock<ISplashScreen> splashScreen;
private Mock<ISystemMonitor> systemMonitor;
private Mock<ITaskbar> taskbar; private Mock<ITaskbar> taskbar;
private Mock<IText> text; private Mock<IText> text;
private Mock<IUserInterfaceFactory> uiFactory; private Mock<IUserInterfaceFactory> uiFactory;
@ -90,7 +87,6 @@ namespace SafeExamBrowser.Client.UnitTests
browser = new Mock<IBrowserApplication>(); browser = new Mock<IBrowserApplication>();
clientHost = new Mock<IClientHost>(); clientHost = new Mock<IClientHost>();
context = new ClientContext(); context = new ClientContext();
coordinator = new Mock<ICoordinator>();
displayMonitor = new Mock<IDisplayMonitor>(); displayMonitor = new Mock<IDisplayMonitor>();
explorerShell = new Mock<IExplorerShell>(); explorerShell = new Mock<IExplorerShell>();
fileSystemDialog = new Mock<IFileSystemDialog>(); fileSystemDialog = new Mock<IFileSystemDialog>();
@ -98,15 +94,15 @@ namespace SafeExamBrowser.Client.UnitTests
integrityModule = new Mock<IIntegrityModule>(); integrityModule = new Mock<IIntegrityModule>();
logger = new Mock<ILogger>(); logger = new Mock<ILogger>();
messageBox = new Mock<IMessageBox>(); messageBox = new Mock<IMessageBox>();
networkAdapter = new Mock<INetworkAdapter>();
operationSequence = new Mock<IOperationSequence>(); operationSequence = new Mock<IOperationSequence>();
registry = new Mock<IRegistry>();
runtimeProxy = new Mock<IRuntimeProxy>(); runtimeProxy = new Mock<IRuntimeProxy>();
sentinel = new Mock<ISystemSentinel>();
server = new Mock<IServerProxy>(); server = new Mock<IServerProxy>();
sessionId = Guid.NewGuid(); sessionId = Guid.NewGuid();
settings = new AppSettings(); settings = new AppSettings();
shutdown = new Mock<Action>(); shutdown = new Mock<Action>();
splashScreen = new Mock<ISplashScreen>(); splashScreen = new Mock<ISplashScreen>();
systemMonitor = new Mock<ISystemMonitor>();
taskbar = new Mock<ITaskbar>(); taskbar = new Mock<ITaskbar>();
text = new Mock<IText>(); text = new Mock<IText>();
uiFactory = new Mock<IUserInterfaceFactory>(); uiFactory = new Mock<IUserInterfaceFactory>();
@ -120,19 +116,18 @@ namespace SafeExamBrowser.Client.UnitTests
actionCenter.Object, actionCenter.Object,
applicationMonitor.Object, applicationMonitor.Object,
context, context,
coordinator.Object,
displayMonitor.Object, displayMonitor.Object,
explorerShell.Object, explorerShell.Object,
fileSystemDialog.Object, fileSystemDialog.Object,
hashAlgorithm.Object, hashAlgorithm.Object,
logger.Object, logger.Object,
messageBox.Object, messageBox.Object,
networkAdapter.Object,
operationSequence.Object, operationSequence.Object,
registry.Object,
runtimeProxy.Object, runtimeProxy.Object,
shutdown.Object, shutdown.Object,
splashScreen.Object, splashScreen.Object,
sentinel.Object, systemMonitor.Object,
taskbar.Object, taskbar.Object,
text.Object, text.Object,
uiFactory.Object); uiFactory.Object);
@ -156,7 +151,7 @@ namespace SafeExamBrowser.Client.UnitTests
var shell = 0; var shell = 0;
var workingArea = 0; var workingArea = 0;
settings.UserInterface.Taskbar.EnableTaskbar = true; settings.Taskbar.EnableTaskbar = true;
actionCenter.Setup(a => a.InitializeBounds()).Callback(() => boundsActionCenter = ++order); actionCenter.Setup(a => a.InitializeBounds()).Callback(() => boundsActionCenter = ++order);
explorerShell.Setup(e => e.Terminate()).Callback(() => shell = ++order); explorerShell.Setup(e => e.Terminate()).Callback(() => shell = ++order);
@ -190,7 +185,7 @@ namespace SafeExamBrowser.Client.UnitTests
var shell = 0; var shell = 0;
var workingArea = 0; var workingArea = 0;
settings.UserInterface.Taskbar.EnableTaskbar = false; settings.Taskbar.EnableTaskbar = false;
actionCenter.Setup(a => a.InitializeBounds()).Callback(() => boundsActionCenter = ++order); actionCenter.Setup(a => a.InitializeBounds()).Callback(() => boundsActionCenter = ++order);
explorerShell.Setup(e => e.Terminate()).Callback(() => shell = ++order); explorerShell.Setup(e => e.Terminate()).Callback(() => shell = ++order);
@ -223,9 +218,9 @@ namespace SafeExamBrowser.Client.UnitTests
lockScreen.Setup(l => l.WaitForResult()).Returns(result); lockScreen.Setup(l => l.WaitForResult()).Returns(result);
runtimeProxy.Setup(p => p.RequestShutdown()).Returns(new CommunicationResult(true)); runtimeProxy.Setup(p => p.RequestShutdown()).Returns(new CommunicationResult(true));
uiFactory uiFactory
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>(), It.IsAny<LockScreenSettings>())) .Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
.Returns(lockScreen.Object) .Returns(lockScreen.Object)
.Callback<string, string, IEnumerable<LockScreenOption>, LockScreenSettings>((m, t, o, s) => result.OptionId = o.First().Id); .Callback<string, string, IEnumerable<LockScreenOption>>((m, t, o) => result.OptionId = o.First().Id);
sut.TryStart(); sut.TryStart();
applicationMonitor.Raise(m => m.TerminationFailed += null, new List<RunningApplication>()); applicationMonitor.Raise(m => m.TerminationFailed += null, new List<RunningApplication>());
@ -242,9 +237,9 @@ namespace SafeExamBrowser.Client.UnitTests
lockScreen.Setup(l => l.WaitForResult()).Returns(result); lockScreen.Setup(l => l.WaitForResult()).Returns(result);
runtimeProxy.Setup(p => p.RequestShutdown()).Returns(new CommunicationResult(true)); runtimeProxy.Setup(p => p.RequestShutdown()).Returns(new CommunicationResult(true));
uiFactory uiFactory
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>(), It.IsAny<LockScreenSettings>())) .Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
.Returns(lockScreen.Object) .Returns(lockScreen.Object)
.Callback<string, string, IEnumerable<LockScreenOption>, LockScreenSettings>((m, t, o, s) => result.OptionId = o.Last().Id); .Callback<string, string, IEnumerable<LockScreenOption>>((m, t, o) => result.OptionId = o.Last().Id);
sut.TryStart(); sut.TryStart();
applicationMonitor.Raise(m => m.TerminationFailed += null, new List<RunningApplication>()); applicationMonitor.Raise(m => m.TerminationFailed += null, new List<RunningApplication>());
@ -276,7 +271,7 @@ namespace SafeExamBrowser.Client.UnitTests
lockScreen.Setup(l => l.WaitForResult()).Callback(() => wait = ++order).Returns(result); lockScreen.Setup(l => l.WaitForResult()).Callback(() => wait = ++order).Returns(result);
lockScreen.Setup(l => l.Close()).Callback(() => close = ++order); lockScreen.Setup(l => l.Close()).Callback(() => close = ++order);
uiFactory uiFactory
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>(), It.IsAny<LockScreenSettings>())) .Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
.Returns(lockScreen.Object); .Returns(lockScreen.Object);
sut.TryStart(); sut.TryStart();
@ -313,7 +308,7 @@ namespace SafeExamBrowser.Client.UnitTests
hashAlgorithm.Setup(a => a.GenerateHashFor(It.Is<string>(p => p == result.Password))).Returns(hash); hashAlgorithm.Setup(a => a.GenerateHashFor(It.Is<string>(p => p == result.Password))).Returns(hash);
lockScreen.Setup(l => l.WaitForResult()).Returns(lockScreenResult); lockScreen.Setup(l => l.WaitForResult()).Returns(lockScreenResult);
uiFactory uiFactory
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>(), It.IsAny<LockScreenSettings>())) .Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
.Returns(lockScreen.Object); .Returns(lockScreen.Object);
sut.TryStart(); sut.TryStart();
@ -502,7 +497,7 @@ namespace SafeExamBrowser.Client.UnitTests
var order = 0; var order = 0;
var workingArea = 0; var workingArea = 0;
settings.UserInterface.Taskbar.EnableTaskbar = true; settings.Taskbar.EnableTaskbar = true;
actionCenter.Setup(t => t.InitializeBounds()).Callback(() => boundsActionCenter = ++order); actionCenter.Setup(t => t.InitializeBounds()).Callback(() => boundsActionCenter = ++order);
displayMonitor.Setup(m => m.InitializePrimaryDisplay(It.Is<int>(h => h == height))).Callback(() => workingArea = ++order); displayMonitor.Setup(m => m.InitializePrimaryDisplay(It.Is<int>(h => h == height))).Callback(() => workingArea = ++order);
@ -533,7 +528,7 @@ namespace SafeExamBrowser.Client.UnitTests
var order = 0; var order = 0;
var workingArea = 0; var workingArea = 0;
settings.UserInterface.Taskbar.EnableTaskbar = false; settings.Taskbar.EnableTaskbar = false;
actionCenter.Setup(t => t.InitializeBounds()).Callback(() => boundsActionCenter = ++order); actionCenter.Setup(t => t.InitializeBounds()).Callback(() => boundsActionCenter = ++order);
displayMonitor.Setup(w => w.InitializePrimaryDisplay(It.Is<int>(h => h == 0))).Callback(() => workingArea = ++order); displayMonitor.Setup(w => w.InitializePrimaryDisplay(It.Is<int>(h => h == 0))).Callback(() => workingArea = ++order);
@ -562,9 +557,7 @@ namespace SafeExamBrowser.Client.UnitTests
displayMonitor.Setup(m => m.ValidateConfiguration(It.IsAny<DisplaySettings>())).Returns(new ValidationResult { IsAllowed = false }); displayMonitor.Setup(m => m.ValidateConfiguration(It.IsAny<DisplaySettings>())).Returns(new ValidationResult { IsAllowed = false });
lockScreen.Setup(l => l.WaitForResult()).Returns(new LockScreenResult()); lockScreen.Setup(l => l.WaitForResult()).Returns(new LockScreenResult());
uiFactory uiFactory.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>())).Returns(lockScreen.Object);
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>(), It.IsAny<LockScreenSettings>()))
.Returns(lockScreen.Object);
sut.TryStart(); sut.TryStart();
displayMonitor.Raise(d => d.DisplayChanged += null); displayMonitor.Raise(d => d.DisplayChanged += null);
@ -729,17 +722,13 @@ namespace SafeExamBrowser.Client.UnitTests
var args = new DownloadEventArgs(); var args = new DownloadEventArgs();
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist"; appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
coordinator.Setup(c => c.RequestReconfigurationLock()).Returns(true);
runtimeProxy.Setup(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>())).Returns(new CommunicationResult(true)); runtimeProxy.Setup(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>())).Returns(new CommunicationResult(true));
sut.TryStart(); sut.TryStart();
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args); browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
args.Callback(true, string.Empty); args.Callback(true, string.Empty);
coordinator.Verify(c => c.RequestReconfigurationLock(), Times.Once);
coordinator.Verify(c => c.ReleaseReconfigurationLock(), Times.Never);
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Once); runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Once);
Assert.IsTrue(args.AllowDownload); Assert.IsTrue(args.AllowDownload);
} }
@ -756,30 +745,7 @@ namespace SafeExamBrowser.Client.UnitTests
sut.TryStart(); sut.TryStart();
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args); browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
coordinator.Verify(c => c.RequestReconfigurationLock(), Times.Never);
coordinator.Verify(c => c.ReleaseReconfigurationLock(), Times.Never);
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never); runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
Assert.IsFalse(args.AllowDownload);
}
[TestMethod]
public void Reconfiguration_MustNotAllowConcurrentExecution()
{
var args = new DownloadEventArgs();
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
coordinator.Setup(c => c.RequestReconfigurationLock()).Returns(false);
runtimeProxy.Setup(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>())).Returns(new CommunicationResult(true));
sut.TryStart();
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
args.Callback?.Invoke(true, string.Empty);
coordinator.Verify(c => c.RequestReconfigurationLock(), Times.Once);
coordinator.Verify(c => c.ReleaseReconfigurationLock(), Times.Never);
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
Assert.IsFalse(args.AllowDownload); Assert.IsFalse(args.AllowDownload);
} }
@ -789,7 +755,6 @@ namespace SafeExamBrowser.Client.UnitTests
var args = new DownloadEventArgs { Url = "sebs://www.somehost.org/some/path/some_configuration.seb?query=123" }; var args = new DownloadEventArgs { Url = "sebs://www.somehost.org/some/path/some_configuration.seb?query=123" };
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist"; appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
coordinator.Setup(c => c.RequestReconfigurationLock()).Returns(true);
settings.Security.AllowReconfiguration = true; settings.Security.AllowReconfiguration = true;
settings.Security.QuitPasswordHash = "abc123"; settings.Security.QuitPasswordHash = "abc123";
settings.Security.ReconfigurationUrl = "sebs://www.somehost.org/some/path/*.seb?query=123"; settings.Security.ReconfigurationUrl = "sebs://www.somehost.org/some/path/*.seb?query=123";
@ -799,10 +764,7 @@ namespace SafeExamBrowser.Client.UnitTests
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args); browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
args.Callback(true, string.Empty); args.Callback(true, string.Empty);
coordinator.Verify(c => c.RequestReconfigurationLock(), Times.Once);
coordinator.Verify(c => c.ReleaseReconfigurationLock(), Times.Never);
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Once); runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Once);
Assert.IsTrue(args.AllowDownload); Assert.IsTrue(args.AllowDownload);
} }
@ -817,10 +779,7 @@ namespace SafeExamBrowser.Client.UnitTests
sut.TryStart(); sut.TryStart();
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args); browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
coordinator.Verify(c => c.RequestReconfigurationLock(), Times.Never);
coordinator.Verify(c => c.ReleaseReconfigurationLock(), Times.Never);
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never); runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
Assert.IsFalse(args.AllowDownload); Assert.IsFalse(args.AllowDownload);
} }
@ -836,10 +795,7 @@ namespace SafeExamBrowser.Client.UnitTests
sut.TryStart(); sut.TryStart();
browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args); browser.Raise(b => b.ConfigurationDownloadRequested += null, "filepath.seb", args);
coordinator.Verify(c => c.RequestReconfigurationLock(), Times.Never);
coordinator.Verify(c => c.ReleaseReconfigurationLock(), Times.Never);
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never); runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
Assert.IsFalse(args.AllowDownload); Assert.IsFalse(args.AllowDownload);
} }
@ -852,7 +808,7 @@ namespace SafeExamBrowser.Client.UnitTests
var args = new DownloadEventArgs(); var args = new DownloadEventArgs();
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist"; appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
coordinator.Setup(c => c.RequestReconfigurationLock()).Returns(true); settings.Security.AllowReconfiguration = true;
messageBox.Setup(m => m.Show( messageBox.Setup(m => m.Show(
It.IsAny<TextKey>(), It.IsAny<TextKey>(),
It.IsAny<TextKey>(), It.IsAny<TextKey>(),
@ -862,14 +818,11 @@ namespace SafeExamBrowser.Client.UnitTests
runtimeProxy.Setup(r => r.RequestReconfiguration( runtimeProxy.Setup(r => r.RequestReconfiguration(
It.Is<string>(p => p == downloadPath), It.Is<string>(p => p == downloadPath),
It.Is<string>(u => u == downloadUrl))).Returns(new CommunicationResult(true)); It.Is<string>(u => u == downloadUrl))).Returns(new CommunicationResult(true));
settings.Security.AllowReconfiguration = true;
sut.TryStart(); sut.TryStart();
browser.Raise(b => b.ConfigurationDownloadRequested += null, filename, args); browser.Raise(b => b.ConfigurationDownloadRequested += null, filename, args);
args.Callback(true, downloadUrl, downloadPath); args.Callback(true, downloadUrl, downloadPath);
coordinator.Verify(c => c.RequestReconfigurationLock(), Times.Once);
coordinator.Verify(c => c.ReleaseReconfigurationLock(), Times.Never);
runtimeProxy.Verify(r => r.RequestReconfiguration(It.Is<string>(p => p == downloadPath), It.Is<string>(u => u == downloadUrl)), Times.Once); runtimeProxy.Verify(r => r.RequestReconfiguration(It.Is<string>(p => p == downloadPath), It.Is<string>(u => u == downloadUrl)), Times.Once);
Assert.AreEqual(downloadPath, args.DownloadPath); Assert.AreEqual(downloadPath, args.DownloadPath);
@ -885,7 +838,7 @@ namespace SafeExamBrowser.Client.UnitTests
var args = new DownloadEventArgs(); var args = new DownloadEventArgs();
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist"; appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
coordinator.Setup(c => c.RequestReconfigurationLock()).Returns(true); settings.Security.AllowReconfiguration = true;
messageBox.Setup(m => m.Show( messageBox.Setup(m => m.Show(
It.IsAny<TextKey>(), It.IsAny<TextKey>(),
It.IsAny<TextKey>(), It.IsAny<TextKey>(),
@ -895,14 +848,11 @@ namespace SafeExamBrowser.Client.UnitTests
runtimeProxy.Setup(r => r.RequestReconfiguration( runtimeProxy.Setup(r => r.RequestReconfiguration(
It.Is<string>(p => p == downloadPath), It.Is<string>(p => p == downloadPath),
It.Is<string>(u => u == downloadUrl))).Returns(new CommunicationResult(true)); It.Is<string>(u => u == downloadUrl))).Returns(new CommunicationResult(true));
settings.Security.AllowReconfiguration = true;
sut.TryStart(); sut.TryStart();
browser.Raise(b => b.ConfigurationDownloadRequested += null, filename, args); browser.Raise(b => b.ConfigurationDownloadRequested += null, filename, args);
args.Callback(false, downloadPath); args.Callback(false, downloadPath);
coordinator.Verify(c => c.RequestReconfigurationLock(), Times.Once);
coordinator.Verify(c => c.ReleaseReconfigurationLock(), Times.Once);
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never); runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Never);
} }
@ -915,7 +865,7 @@ namespace SafeExamBrowser.Client.UnitTests
var args = new DownloadEventArgs(); var args = new DownloadEventArgs();
appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist"; appConfig.TemporaryDirectory = @"C:\Folder\Does\Not\Exist";
coordinator.Setup(c => c.RequestReconfigurationLock()).Returns(true); settings.Security.AllowReconfiguration = true;
messageBox.Setup(m => m.Show( messageBox.Setup(m => m.Show(
It.IsAny<TextKey>(), It.IsAny<TextKey>(),
It.IsAny<TextKey>(), It.IsAny<TextKey>(),
@ -925,21 +875,18 @@ namespace SafeExamBrowser.Client.UnitTests
runtimeProxy.Setup(r => r.RequestReconfiguration( runtimeProxy.Setup(r => r.RequestReconfiguration(
It.Is<string>(p => p == downloadPath), It.Is<string>(p => p == downloadPath),
It.Is<string>(u => u == downloadUrl))).Returns(new CommunicationResult(false)); It.Is<string>(u => u == downloadUrl))).Returns(new CommunicationResult(false));
settings.Security.AllowReconfiguration = true;
sut.TryStart(); sut.TryStart();
browser.Raise(b => b.ConfigurationDownloadRequested += null, filename, args); browser.Raise(b => b.ConfigurationDownloadRequested += null, filename, args);
args.Callback(true, downloadUrl, downloadPath); args.Callback(true, downloadUrl, downloadPath);
coordinator.Verify(c => c.RequestReconfigurationLock(), Times.Once); runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Once);
coordinator.Verify(c => c.ReleaseReconfigurationLock(), Times.Once);
messageBox.Verify(m => m.Show( messageBox.Verify(m => m.Show(
It.IsAny<TextKey>(), It.IsAny<TextKey>(),
It.IsAny<TextKey>(), It.IsAny<TextKey>(),
It.IsAny<MessageBoxAction>(), It.IsAny<MessageBoxAction>(),
It.Is<MessageBoxIcon>(i => i == MessageBoxIcon.Error), It.Is<MessageBoxIcon>(i => i == MessageBoxIcon.Error),
It.IsAny<IWindow>()), Times.Once); It.IsAny<IWindow>()), Times.Once);
runtimeProxy.Verify(r => r.RequestReconfiguration(It.IsAny<string>(), It.IsAny<string>()), Times.Once);
} }
[TestMethod] [TestMethod]
@ -995,8 +942,8 @@ namespace SafeExamBrowser.Client.UnitTests
[TestMethod] [TestMethod]
public void Shutdown_MustCloseActionCenterAndTaskbarIfEnabled() public void Shutdown_MustCloseActionCenterAndTaskbarIfEnabled()
{ {
settings.UserInterface.ActionCenter.EnableActionCenter = true; settings.ActionCenter.EnableActionCenter = true;
settings.UserInterface.Taskbar.EnableTaskbar = true; settings.Taskbar.EnableTaskbar = true;
sut.Terminate(); sut.Terminate();
@ -1007,8 +954,8 @@ namespace SafeExamBrowser.Client.UnitTests
[TestMethod] [TestMethod]
public void Shutdown_MustNotCloseActionCenterAndTaskbarIfNotEnabled() public void Shutdown_MustNotCloseActionCenterAndTaskbarIfNotEnabled()
{ {
settings.UserInterface.ActionCenter.EnableActionCenter = false; settings.ActionCenter.EnableActionCenter = false;
settings.UserInterface.Taskbar.EnableTaskbar = false; settings.Taskbar.EnableTaskbar = false;
sut.Terminate(); sut.Terminate();
@ -1179,7 +1126,7 @@ namespace SafeExamBrowser.Client.UnitTests
[TestMethod] [TestMethod]
public void Startup_MustCorrectlyShowTaskbar() public void Startup_MustCorrectlyShowTaskbar()
{ {
settings.UserInterface.Taskbar.EnableTaskbar = true; settings.Taskbar.EnableTaskbar = true;
sut.TryStart(); sut.TryStart();
taskbar.Verify(t => t.Show(), Times.Once); taskbar.Verify(t => t.Show(), Times.Once);
@ -1191,7 +1138,7 @@ namespace SafeExamBrowser.Client.UnitTests
taskbar.Verify(t => t.Show(), Times.Never); taskbar.Verify(t => t.Show(), Times.Never);
taskbar.Reset(); taskbar.Reset();
settings.UserInterface.Taskbar.EnableTaskbar = false; settings.Taskbar.EnableTaskbar = false;
operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Success); operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Success);
sut.TryStart(); sut.TryStart();
@ -1201,7 +1148,7 @@ namespace SafeExamBrowser.Client.UnitTests
[TestMethod] [TestMethod]
public void Startup_MustCorrectlyShowActionCenter() public void Startup_MustCorrectlyShowActionCenter()
{ {
settings.UserInterface.ActionCenter.EnableActionCenter = true; settings.ActionCenter.EnableActionCenter = true;
sut.TryStart(); sut.TryStart();
actionCenter.Verify(t => t.Promote(), Times.Once); actionCenter.Verify(t => t.Promote(), Times.Once);
@ -1215,7 +1162,7 @@ namespace SafeExamBrowser.Client.UnitTests
actionCenter.Verify(t => t.Show(), Times.Never); actionCenter.Verify(t => t.Show(), Times.Never);
actionCenter.Reset(); actionCenter.Reset();
settings.UserInterface.ActionCenter.EnableActionCenter = false; settings.ActionCenter.EnableActionCenter = false;
operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Success); operationSequence.Setup(o => o.TryPerform()).Returns(OperationResult.Success);
sut.TryStart(); sut.TryStart();
@ -1277,18 +1224,13 @@ namespace SafeExamBrowser.Client.UnitTests
{ {
var lockScreen = new Mock<ILockScreen>(); var lockScreen = new Mock<ILockScreen>();
coordinator.Setup(c => c.RequestSessionLock()).Returns(true);
lockScreen.Setup(l => l.WaitForResult()).Returns(new LockScreenResult());
settings.Service.IgnoreService = true; settings.Service.IgnoreService = true;
uiFactory lockScreen.Setup(l => l.WaitForResult()).Returns(new LockScreenResult());
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>(), It.IsAny<LockScreenSettings>())) uiFactory.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>())).Returns(lockScreen.Object);
.Returns(lockScreen.Object);
sut.TryStart(); sut.TryStart();
sentinel.Raise(s => s.SessionChanged += null); systemMonitor.Raise(m => m.SessionChanged += null);
coordinator.Verify(c => c.RequestSessionLock(), Times.Once);
coordinator.Verify(c => c.ReleaseSessionLock(), Times.Once);
lockScreen.Verify(l => l.Show(), Times.Once); lockScreen.Verify(l => l.Show(), Times.Once);
} }
@ -1298,20 +1240,17 @@ namespace SafeExamBrowser.Client.UnitTests
var lockScreen = new Mock<ILockScreen>(); var lockScreen = new Mock<ILockScreen>();
var result = new LockScreenResult(); var result = new LockScreenResult();
coordinator.Setup(c => c.RequestSessionLock()).Returns(true); settings.Service.IgnoreService = true;
lockScreen.Setup(l => l.WaitForResult()).Returns(result); lockScreen.Setup(l => l.WaitForResult()).Returns(result);
runtimeProxy.Setup(r => r.RequestShutdown()).Returns(new CommunicationResult(true)); runtimeProxy.Setup(r => r.RequestShutdown()).Returns(new CommunicationResult(true));
settings.Service.IgnoreService = true;
uiFactory uiFactory
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>(), It.IsAny<LockScreenSettings>())) .Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>()))
.Callback(new Action<string, string, IEnumerable<LockScreenOption>, LockScreenSettings>((message, title, options, settings) => result.OptionId = options.Last().Id)) .Callback(new Action<string, string, IEnumerable<LockScreenOption>>((message, title, options) => result.OptionId = options.Last().Id))
.Returns(lockScreen.Object); .Returns(lockScreen.Object);
sut.TryStart(); sut.TryStart();
sentinel.Raise(s => s.SessionChanged += null); systemMonitor.Raise(m => m.SessionChanged += null);
coordinator.Verify(c => c.RequestSessionLock(), Times.Once);
coordinator.Verify(c => c.ReleaseSessionLock(), Times.Once);
lockScreen.Verify(l => l.Show(), Times.Once); lockScreen.Verify(l => l.Show(), Times.Once);
runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once); runtimeProxy.Verify(p => p.RequestShutdown(), Times.Once);
} }
@ -1325,12 +1264,10 @@ namespace SafeExamBrowser.Client.UnitTests
settings.Service.DisableUserLock = false; settings.Service.DisableUserLock = false;
settings.Service.DisableUserSwitch = false; settings.Service.DisableUserSwitch = false;
lockScreen.Setup(l => l.WaitForResult()).Returns(new LockScreenResult()); lockScreen.Setup(l => l.WaitForResult()).Returns(new LockScreenResult());
uiFactory uiFactory.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>())).Returns(lockScreen.Object);
.Setup(f => f.CreateLockScreen(It.IsAny<string>(), It.IsAny<string>(), It.IsAny<IEnumerable<LockScreenOption>>(), It.IsAny<LockScreenSettings>()))
.Returns(lockScreen.Object);
sut.TryStart(); sut.TryStart();
sentinel.Raise(s => s.SessionChanged += null); systemMonitor.Raise(m => m.SessionChanged += null);
lockScreen.Verify(l => l.Show(), Times.Never); lockScreen.Verify(l => l.Show(), Times.Never);
} }

View file

@ -1,118 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.Threading;
using System.Threading.Tasks;
using Microsoft.VisualStudio.TestTools.UnitTesting;
namespace SafeExamBrowser.Client.UnitTests
{
[TestClass]
public class CoordinatorTests
{
private Coordinator sut;
[TestInitialize]
public void Initialize()
{
sut = new Coordinator();
}
[TestMethod]
public void ReconfigurationLock_MustWorkCorrectly()
{
Assert.IsFalse(sut.IsReconfigurationLocked());
sut.RequestReconfigurationLock();
var result = Parallel.For(1, 1000, (_) =>
{
Assert.IsTrue(sut.IsReconfigurationLocked());
Assert.IsFalse(sut.RequestReconfigurationLock());
});
Assert.IsTrue(result.IsCompleted);
result = Parallel.For(1, 1000, (_) =>
{
sut.ReleaseReconfigurationLock();
});
Assert.IsFalse(sut.IsReconfigurationLocked());
Assert.IsTrue(result.IsCompleted);
}
[TestMethod]
public void RequestReconfigurationLock_MustOnlyAllowLockingOnce()
{
var count = 0;
Assert.IsFalse(sut.IsReconfigurationLocked());
var result = Parallel.For(1, 1000, (_) =>
{
var acquired = sut.RequestReconfigurationLock();
if (acquired)
{
Interlocked.Increment(ref count);
}
});
Assert.AreEqual(1, count);
Assert.IsTrue(sut.IsReconfigurationLocked());
Assert.IsTrue(result.IsCompleted);
}
[TestMethod]
public void RequestSessionLock_MustOnlyAllowLockingOnce()
{
var count = 0;
Assert.IsFalse(sut.IsSessionLocked());
var result = Parallel.For(1, 1000, (_) =>
{
var acquired = sut.RequestSessionLock();
if (acquired)
{
Interlocked.Increment(ref count);
}
});
Assert.AreEqual(1, count);
Assert.IsTrue(sut.IsSessionLocked());
Assert.IsTrue(result.IsCompleted);
}
[TestMethod]
public void SessionLock_MustWorkCorrectly()
{
Assert.IsFalse(sut.IsSessionLocked());
sut.RequestSessionLock();
var result = Parallel.For(1, 1000, (_) =>
{
Assert.IsTrue(sut.IsSessionLocked());
Assert.IsFalse(sut.RequestSessionLock());
});
Assert.IsTrue(result.IsCompleted);
result = Parallel.For(1, 1000, (_) =>
{
sut.ReleaseSessionLock();
});
Assert.IsFalse(sut.IsSessionLocked());
Assert.IsTrue(result.IsCompleted);
}
}
}

View file

@ -64,9 +64,9 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustNotInitializeBrowserIfNotEnabled() public void Perform_MustNotInitializeBrowserIfNotEnabled()
{ {
settings.ActionCenter.EnableActionCenter = true;
settings.Browser.EnableBrowser = false; settings.Browser.EnableBrowser = false;
settings.UserInterface.ActionCenter.EnableActionCenter = true; settings.Taskbar.EnableTaskbar = true;
settings.UserInterface.Taskbar.EnableTaskbar = true;
sut.Perform(); sut.Perform();
@ -79,17 +79,17 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustCorrectlyInitializeControls() public void Perform_MustCorrectlyInitializeControls()
{ {
settings.ActionCenter.EnableActionCenter = false;
settings.Browser.EnableBrowser = true; settings.Browser.EnableBrowser = true;
settings.UserInterface.ActionCenter.EnableActionCenter = false; settings.Taskbar.EnableTaskbar = false;
settings.UserInterface.Taskbar.EnableTaskbar = false;
sut.Perform(); sut.Perform();
actionCenter.Verify(a => a.AddApplicationControl(It.IsAny<IApplicationControl>(), true), Times.Never); actionCenter.Verify(a => a.AddApplicationControl(It.IsAny<IApplicationControl>(), true), Times.Never);
taskbar.Verify(t => t.AddApplicationControl(It.IsAny<IApplicationControl>(), true), Times.Never); taskbar.Verify(t => t.AddApplicationControl(It.IsAny<IApplicationControl>(), true), Times.Never);
settings.UserInterface.ActionCenter.EnableActionCenter = true; settings.ActionCenter.EnableActionCenter = true;
settings.UserInterface.Taskbar.EnableTaskbar = true; settings.Taskbar.EnableTaskbar = true;
sut.Perform(); sut.Perform();

View file

@ -43,7 +43,7 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
var order = 0; var order = 0;
context.Settings = new AppSettings(); context.Settings = new AppSettings();
context.Settings.UserInterface.Taskbar.EnableTaskbar = true; context.Settings.Taskbar.EnableTaskbar = true;
displayMonitor.Setup(d => d.InitializePrimaryDisplay(It.IsAny<int>())).Callback(() => Assert.AreEqual(++order, 1)); displayMonitor.Setup(d => d.InitializePrimaryDisplay(It.IsAny<int>())).Callback(() => Assert.AreEqual(++order, 1));
displayMonitor.Setup(d => d.StartMonitoringDisplayChanges()).Callback(() => Assert.AreEqual(++order, 2)); displayMonitor.Setup(d => d.StartMonitoringDisplayChanges()).Callback(() => Assert.AreEqual(++order, 2));
@ -60,7 +60,7 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
var height = 25; var height = 25;
context.Settings = new AppSettings(); context.Settings = new AppSettings();
context.Settings.UserInterface.Taskbar.EnableTaskbar = true; context.Settings.Taskbar.EnableTaskbar = true;
taskbar.Setup(t => t.GetAbsoluteHeight()).Returns(height); taskbar.Setup(t => t.GetAbsoluteHeight()).Returns(height);
sut.Perform(); sut.Perform();
@ -75,7 +75,7 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
var height = 25; var height = 25;
context.Settings = new AppSettings(); context.Settings = new AppSettings();
context.Settings.UserInterface.Taskbar.EnableTaskbar = false; context.Settings.Taskbar.EnableTaskbar = false;
taskbar.Setup(t => t.GetAbsoluteHeight()).Returns(height); taskbar.Setup(t => t.GetAbsoluteHeight()).Returns(height);
sut.Perform(); sut.Perform();

View file

@ -100,9 +100,9 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
context.Activators.Add(actionCenterActivator.Object); context.Activators.Add(actionCenterActivator.Object);
context.Activators.Add(taskviewActivator.Object); context.Activators.Add(taskviewActivator.Object);
context.Activators.Add(terminationActivator.Object); context.Activators.Add(terminationActivator.Object);
context.Settings.ActionCenter.EnableActionCenter = true;
context.Settings.Keyboard.AllowAltTab = true; context.Settings.Keyboard.AllowAltTab = true;
context.Settings.Security.AllowTermination = true; context.Settings.Security.AllowTermination = true;
context.Settings.UserInterface.ActionCenter.EnableActionCenter = true;
sut.Perform(); sut.Perform();
@ -130,11 +130,11 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
context.Applications.Add(application1.Object); context.Applications.Add(application1.Object);
context.Applications.Add(application2.Object); context.Applications.Add(application2.Object);
context.Applications.Add(application3.Object); context.Applications.Add(application3.Object);
context.Settings.ActionCenter.EnableActionCenter = true;
context.Settings.Taskbar.EnableTaskbar = true;
context.Settings.Applications.Whitelist.Add(application1Settings); context.Settings.Applications.Whitelist.Add(application1Settings);
context.Settings.Applications.Whitelist.Add(application2Settings); context.Settings.Applications.Whitelist.Add(application2Settings);
context.Settings.Applications.Whitelist.Add(application3Settings); context.Settings.Applications.Whitelist.Add(application3Settings);
context.Settings.UserInterface.ActionCenter.EnableActionCenter = true;
context.Settings.UserInterface.Taskbar.EnableTaskbar = true;
sut.Perform(); sut.Perform();
@ -168,11 +168,11 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
context.Applications.Add(application1.Object); context.Applications.Add(application1.Object);
context.Applications.Add(application2.Object); context.Applications.Add(application2.Object);
context.Applications.Add(application3.Object); context.Applications.Add(application3.Object);
context.Settings.ActionCenter.EnableActionCenter = false;
context.Settings.Taskbar.EnableTaskbar = false;
context.Settings.Applications.Whitelist.Add(application1Settings); context.Settings.Applications.Whitelist.Add(application1Settings);
context.Settings.Applications.Whitelist.Add(application2Settings); context.Settings.Applications.Whitelist.Add(application2Settings);
context.Settings.Applications.Whitelist.Add(application3Settings); context.Settings.Applications.Whitelist.Add(application3Settings);
context.Settings.UserInterface.ActionCenter.EnableActionCenter = false;
context.Settings.UserInterface.Taskbar.EnableTaskbar = false;
sut.Perform(); sut.Perform();
@ -211,10 +211,10 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustInitializeClock() public void Perform_MustInitializeClock()
{ {
context.Settings.UserInterface.ActionCenter.EnableActionCenter = true; context.Settings.ActionCenter.EnableActionCenter = true;
context.Settings.UserInterface.ActionCenter.ShowClock = true; context.Settings.ActionCenter.ShowClock = true;
context.Settings.UserInterface.Taskbar.EnableTaskbar = true; context.Settings.Taskbar.EnableTaskbar = true;
context.Settings.UserInterface.Taskbar.ShowClock = true; context.Settings.Taskbar.ShowClock = true;
sut.Perform(); sut.Perform();
@ -225,10 +225,10 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustNotInitializeClock() public void Perform_MustNotInitializeClock()
{ {
context.Settings.UserInterface.ActionCenter.EnableActionCenter = true; context.Settings.ActionCenter.EnableActionCenter = true;
context.Settings.UserInterface.ActionCenter.ShowClock = false; context.Settings.ActionCenter.ShowClock = false;
context.Settings.UserInterface.Taskbar.EnableTaskbar = true; context.Settings.Taskbar.EnableTaskbar = true;
context.Settings.UserInterface.Taskbar.ShowClock = false; context.Settings.Taskbar.ShowClock = false;
sut.Perform(); sut.Perform();
@ -239,9 +239,9 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustInitializeQuitButton() public void Perform_MustInitializeQuitButton()
{ {
context.Settings.ActionCenter.EnableActionCenter = true;
context.Settings.Taskbar.EnableTaskbar = true;
context.Settings.Security.AllowTermination = false; context.Settings.Security.AllowTermination = false;
context.Settings.UserInterface.ActionCenter.EnableActionCenter = true;
context.Settings.UserInterface.Taskbar.EnableTaskbar = true;
sut.Perform(); sut.Perform();
@ -265,12 +265,12 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustInitializeNotifications() public void Perform_MustInitializeNotifications()
{ {
context.Settings.UserInterface.ActionCenter.EnableActionCenter = true; context.Settings.ActionCenter.EnableActionCenter = true;
context.Settings.UserInterface.ActionCenter.ShowApplicationInfo = true; context.Settings.ActionCenter.ShowApplicationInfo = true;
context.Settings.UserInterface.ActionCenter.ShowApplicationLog = true; context.Settings.ActionCenter.ShowApplicationLog = true;
context.Settings.UserInterface.Taskbar.EnableTaskbar = true; context.Settings.Taskbar.EnableTaskbar = true;
context.Settings.UserInterface.Taskbar.ShowApplicationInfo = true; context.Settings.Taskbar.ShowApplicationInfo = true;
context.Settings.UserInterface.Taskbar.ShowApplicationLog = true; context.Settings.Taskbar.ShowApplicationLog = true;
sut.Perform(); sut.Perform();
@ -283,10 +283,10 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
{ {
var logControl = new Mock<INotificationControl>(); var logControl = new Mock<INotificationControl>();
context.Settings.UserInterface.ActionCenter.EnableActionCenter = true; context.Settings.ActionCenter.EnableActionCenter = true;
context.Settings.UserInterface.ActionCenter.ShowApplicationLog = false; context.Settings.ActionCenter.ShowApplicationLog = false;
context.Settings.UserInterface.Taskbar.EnableTaskbar = true; context.Settings.Taskbar.EnableTaskbar = true;
context.Settings.UserInterface.Taskbar.ShowApplicationLog = false; context.Settings.Taskbar.ShowApplicationLog = false;
uiFactory uiFactory
.Setup(f => f.CreateNotificationControl(It.IsAny<INotification>(), It.IsAny<Location>())) .Setup(f => f.CreateNotificationControl(It.IsAny<INotification>(), It.IsAny<Location>()))
@ -301,14 +301,14 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustInitializeSystemComponents() public void Perform_MustInitializeSystemComponents()
{ {
context.Settings.UserInterface.ActionCenter.EnableActionCenter = true; context.Settings.ActionCenter.EnableActionCenter = true;
context.Settings.UserInterface.ActionCenter.ShowAudio = true; context.Settings.ActionCenter.ShowAudio = true;
context.Settings.UserInterface.ActionCenter.ShowKeyboardLayout = true; context.Settings.ActionCenter.ShowKeyboardLayout = true;
context.Settings.UserInterface.ActionCenter.ShowNetwork = true; context.Settings.ActionCenter.ShowNetwork = true;
context.Settings.UserInterface.Taskbar.EnableTaskbar = true; context.Settings.Taskbar.EnableTaskbar = true;
context.Settings.UserInterface.Taskbar.ShowAudio = true; context.Settings.Taskbar.ShowAudio = true;
context.Settings.UserInterface.Taskbar.ShowKeyboardLayout = true; context.Settings.Taskbar.ShowKeyboardLayout = true;
context.Settings.UserInterface.Taskbar.ShowNetwork = true; context.Settings.Taskbar.ShowNetwork = true;
systemInfo.SetupGet(s => s.HasBattery).Returns(true); systemInfo.SetupGet(s => s.HasBattery).Returns(true);
uiFactory.Setup(f => f.CreateAudioControl(It.IsAny<IAudio>(), It.IsAny<Location>())).Returns(new Mock<ISystemControl>().Object); uiFactory.Setup(f => f.CreateAudioControl(It.IsAny<IAudio>(), It.IsAny<Location>())).Returns(new Mock<ISystemControl>().Object);
@ -329,14 +329,14 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustNotInitializeSystemComponents() public void Perform_MustNotInitializeSystemComponents()
{ {
context.Settings.UserInterface.ActionCenter.EnableActionCenter = true; context.Settings.ActionCenter.EnableActionCenter = true;
context.Settings.UserInterface.ActionCenter.ShowAudio = false; context.Settings.ActionCenter.ShowAudio = false;
context.Settings.UserInterface.ActionCenter.ShowKeyboardLayout = false; context.Settings.ActionCenter.ShowKeyboardLayout = false;
context.Settings.UserInterface.ActionCenter.ShowNetwork = false; context.Settings.ActionCenter.ShowNetwork = false;
context.Settings.UserInterface.Taskbar.EnableTaskbar = true; context.Settings.Taskbar.EnableTaskbar = true;
context.Settings.UserInterface.Taskbar.ShowAudio = false; context.Settings.Taskbar.ShowAudio = false;
context.Settings.UserInterface.Taskbar.ShowKeyboardLayout = false; context.Settings.Taskbar.ShowKeyboardLayout = false;
context.Settings.UserInterface.Taskbar.ShowNetwork = false; context.Settings.Taskbar.ShowNetwork = false;
systemInfo.SetupGet(s => s.HasBattery).Returns(false); systemInfo.SetupGet(s => s.HasBattery).Returns(false);
uiFactory.Setup(f => f.CreateAudioControl(It.IsAny<IAudio>(), It.IsAny<Location>())).Returns(new Mock<ISystemControl>().Object); uiFactory.Setup(f => f.CreateAudioControl(It.IsAny<IAudio>(), It.IsAny<Location>())).Returns(new Mock<ISystemControl>().Object);
@ -360,7 +360,7 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
var actionCenterActivator = new Mock<IActionCenterActivator>(); var actionCenterActivator = new Mock<IActionCenterActivator>();
context.Activators.Add(actionCenterActivator.Object); context.Activators.Add(actionCenterActivator.Object);
context.Settings.UserInterface.ActionCenter.EnableActionCenter = false; context.Settings.ActionCenter.EnableActionCenter = false;
sut.Perform(); sut.Perform();
@ -385,7 +385,7 @@ namespace SafeExamBrowser.Client.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustNotInitializeTaskbarIfNotEnabled() public void Perform_MustNotInitializeTaskbarIfNotEnabled()
{ {
context.Settings.UserInterface.Taskbar.EnableTaskbar = false; context.Settings.Taskbar.EnableTaskbar = false;
sut.Perform(); sut.Perform();
taskbar.VerifyNoOtherCalls(); taskbar.VerifyNoOtherCalls();
} }

View file

@ -0,0 +1,53 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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 Microsoft.VisualStudio.TestTools.UnitTesting;
using Moq;
using SafeExamBrowser.Client.Operations;
using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.System;
namespace SafeExamBrowser.Client.UnitTests.Operations
{
[TestClass]
public class SystemMonitorOperationTests
{
private ClientContext context;
private Mock<ISystemMonitor> systemMonitor;
private Mock<ILogger> logger;
private SystemMonitorOperation sut;
[TestInitialize]
public void Initialize()
{
context = new ClientContext();
systemMonitor = new Mock<ISystemMonitor>();
logger = new Mock<ILogger>();
sut = new SystemMonitorOperation(context, systemMonitor.Object, logger.Object);
}
[TestMethod]
public void Perform_MustStartMonitor()
{
sut.Perform();
systemMonitor.Verify(s => s.Start(), Times.Once);
systemMonitor.VerifyNoOtherCalls();
}
[TestMethod]
public void Revert_MustStopMonitor()
{
sut.Revert();
systemMonitor.Verify(s => s.Stop(), Times.Once);
systemMonitor.VerifyNoOtherCalls();
}
}
}

View file

@ -163,9 +163,9 @@
<Compile Include="Communication\ClientHostTests.cs" /> <Compile Include="Communication\ClientHostTests.cs" />
<Compile Include="Notifications\AboutNotificationControllerTests.cs" /> <Compile Include="Notifications\AboutNotificationControllerTests.cs" />
<Compile Include="Notifications\LogNotificationControllerTests.cs" /> <Compile Include="Notifications\LogNotificationControllerTests.cs" />
<Compile Include="Operations\SystemMonitorOperationTests.cs" />
<Compile Include="Properties\AssemblyInfo.cs" /> <Compile Include="Properties\AssemblyInfo.cs" />
<Compile Include="ClientControllerTests.cs" /> <Compile Include="ClientControllerTests.cs" />
<Compile Include="CoordinatorTests.cs" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<None Include="app.config"> <None Include="app.config">

View file

@ -20,7 +20,7 @@ namespace SafeExamBrowser.Client
private const int ILMCM_LANGUAGEBAROFF = 0x00002; private const int ILMCM_LANGUAGEBAROFF = 0x00002;
private static readonly Mutex Mutex = new Mutex(true, AppConfig.CLIENT_MUTEX_NAME); private static readonly Mutex Mutex = new Mutex(true, AppConfig.CLIENT_MUTEX_NAME);
private readonly CompositionRoot instances = new CompositionRoot(); private CompositionRoot instances = new CompositionRoot();
[STAThread] [STAThread]
public static void Main() public static void Main()

View file

@ -16,7 +16,6 @@ using System.Threading.Tasks;
using SafeExamBrowser.Applications.Contracts; using SafeExamBrowser.Applications.Contracts;
using SafeExamBrowser.Browser.Contracts; using SafeExamBrowser.Browser.Contracts;
using SafeExamBrowser.Browser.Contracts.Events; using SafeExamBrowser.Browser.Contracts.Events;
using SafeExamBrowser.Client.Contracts;
using SafeExamBrowser.Client.Operations.Events; using SafeExamBrowser.Client.Operations.Events;
using SafeExamBrowser.Communication.Contracts.Data; using SafeExamBrowser.Communication.Contracts.Data;
using SafeExamBrowser.Communication.Contracts.Events; using SafeExamBrowser.Communication.Contracts.Events;
@ -31,14 +30,12 @@ using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.Applications; using SafeExamBrowser.Monitoring.Contracts.Applications;
using SafeExamBrowser.Monitoring.Contracts.Display; using SafeExamBrowser.Monitoring.Contracts.Display;
using SafeExamBrowser.Monitoring.Contracts.System; using SafeExamBrowser.Monitoring.Contracts.System;
using SafeExamBrowser.Monitoring.Contracts.System.Events;
using SafeExamBrowser.Proctoring.Contracts; using SafeExamBrowser.Proctoring.Contracts;
using SafeExamBrowser.Proctoring.Contracts.Events; using SafeExamBrowser.Proctoring.Contracts.Events;
using SafeExamBrowser.Server.Contracts; using SafeExamBrowser.Server.Contracts;
using SafeExamBrowser.Server.Contracts.Data; using SafeExamBrowser.Server.Contracts.Data;
using SafeExamBrowser.Settings; using SafeExamBrowser.Settings;
using SafeExamBrowser.SystemComponents.Contracts.Network; using SafeExamBrowser.SystemComponents.Contracts.Registry;
using SafeExamBrowser.SystemComponents.Contracts.Network.Events;
using SafeExamBrowser.UserInterface.Contracts; using SafeExamBrowser.UserInterface.Contracts;
using SafeExamBrowser.UserInterface.Contracts.FileSystemDialog; using SafeExamBrowser.UserInterface.Contracts.FileSystemDialog;
using SafeExamBrowser.UserInterface.Contracts.MessageBox; using SafeExamBrowser.UserInterface.Contracts.MessageBox;
@ -54,19 +51,18 @@ namespace SafeExamBrowser.Client
private readonly IActionCenter actionCenter; private readonly IActionCenter actionCenter;
private readonly IApplicationMonitor applicationMonitor; private readonly IApplicationMonitor applicationMonitor;
private readonly ClientContext context; private readonly ClientContext context;
private readonly ICoordinator coordinator;
private readonly IDisplayMonitor displayMonitor; private readonly IDisplayMonitor displayMonitor;
private readonly IExplorerShell explorerShell; private readonly IExplorerShell explorerShell;
private readonly IFileSystemDialog fileSystemDialog; private readonly IFileSystemDialog fileSystemDialog;
private readonly IHashAlgorithm hashAlgorithm; private readonly IHashAlgorithm hashAlgorithm;
private readonly ILogger logger; private readonly ILogger logger;
private readonly IMessageBox messageBox; private readonly IMessageBox messageBox;
private readonly INetworkAdapter networkAdapter;
private readonly IOperationSequence operations; private readonly IOperationSequence operations;
private readonly IRegistry registry;
private readonly IRuntimeProxy runtime; private readonly IRuntimeProxy runtime;
private readonly Action shutdown; private readonly Action shutdown;
private readonly ISplashScreen splashScreen; private readonly ISplashScreen splashScreen;
private readonly ISystemSentinel sentinel; private readonly ISystemMonitor systemMonitor;
private readonly ITaskbar taskbar; private readonly ITaskbar taskbar;
private readonly IText text; private readonly IText text;
private readonly IUserInterfaceFactory uiFactory; private readonly IUserInterfaceFactory uiFactory;
@ -79,24 +75,24 @@ namespace SafeExamBrowser.Client
private AppSettings Settings => context.Settings; private AppSettings Settings => context.Settings;
private ILockScreen lockScreen; private ILockScreen lockScreen;
private bool sessionLocked;
internal ClientController( internal ClientController(
IActionCenter actionCenter, IActionCenter actionCenter,
IApplicationMonitor applicationMonitor, IApplicationMonitor applicationMonitor,
ClientContext context, ClientContext context,
ICoordinator coordinator,
IDisplayMonitor displayMonitor, IDisplayMonitor displayMonitor,
IExplorerShell explorerShell, IExplorerShell explorerShell,
IFileSystemDialog fileSystemDialog, IFileSystemDialog fileSystemDialog,
IHashAlgorithm hashAlgorithm, IHashAlgorithm hashAlgorithm,
ILogger logger, ILogger logger,
IMessageBox messageBox, IMessageBox messageBox,
INetworkAdapter networkAdapter,
IOperationSequence operations, IOperationSequence operations,
IRegistry registry,
IRuntimeProxy runtime, IRuntimeProxy runtime,
Action shutdown, Action shutdown,
ISplashScreen splashScreen, ISplashScreen splashScreen,
ISystemSentinel sentinel, ISystemMonitor systemMonitor,
ITaskbar taskbar, ITaskbar taskbar,
IText text, IText text,
IUserInterfaceFactory uiFactory) IUserInterfaceFactory uiFactory)
@ -104,19 +100,18 @@ namespace SafeExamBrowser.Client
this.actionCenter = actionCenter; this.actionCenter = actionCenter;
this.applicationMonitor = applicationMonitor; this.applicationMonitor = applicationMonitor;
this.context = context; this.context = context;
this.coordinator = coordinator;
this.displayMonitor = displayMonitor; this.displayMonitor = displayMonitor;
this.explorerShell = explorerShell; this.explorerShell = explorerShell;
this.fileSystemDialog = fileSystemDialog; this.fileSystemDialog = fileSystemDialog;
this.hashAlgorithm = hashAlgorithm; this.hashAlgorithm = hashAlgorithm;
this.logger = logger; this.logger = logger;
this.messageBox = messageBox; this.messageBox = messageBox;
this.networkAdapter = networkAdapter;
this.operations = operations; this.operations = operations;
this.registry = registry;
this.runtime = runtime; this.runtime = runtime;
this.shutdown = shutdown; this.shutdown = shutdown;
this.splashScreen = splashScreen; this.splashScreen = splashScreen;
this.sentinel = sentinel; this.systemMonitor = systemMonitor;
this.taskbar = taskbar; this.taskbar = taskbar;
this.text = text; this.text = text;
this.uiFactory = uiFactory; this.uiFactory = uiFactory;
@ -141,7 +136,6 @@ namespace SafeExamBrowser.Client
ShowShell(); ShowShell();
AutoStartApplications(); AutoStartApplications();
ScheduleIntegrityVerification(); ScheduleIntegrityVerification();
StartMonitoring();
var communication = runtime.InformClientReady(); var communication = runtime.InformClientReady();
@ -180,6 +174,7 @@ namespace SafeExamBrowser.Client
CloseShell(); CloseShell();
DeregisterEvents(); DeregisterEvents();
UpdateSessionIntegrity(); UpdateSessionIntegrity();
TerminateIntegrityVerification();
var success = operations.TryRevert() == OperationResult.Success; var success = operations.TryRevert() == OperationResult.Success;
@ -219,12 +214,9 @@ namespace SafeExamBrowser.Client
ClientHost.ServerFailureActionRequested += ClientHost_ServerFailureActionRequested; ClientHost.ServerFailureActionRequested += ClientHost_ServerFailureActionRequested;
ClientHost.Shutdown += ClientHost_Shutdown; ClientHost.Shutdown += ClientHost_Shutdown;
displayMonitor.DisplayChanged += DisplayMonitor_DisplaySettingsChanged; displayMonitor.DisplayChanged += DisplayMonitor_DisplaySettingsChanged;
networkAdapter.CredentialsRequired += NetworkAdapter_CredentialsRequired; registry.ValueChanged += Registry_ValueChanged;
runtime.ConnectionLost += Runtime_ConnectionLost; runtime.ConnectionLost += Runtime_ConnectionLost;
sentinel.CursorChanged += Sentinel_CursorChanged; systemMonitor.SessionChanged += SystemMonitor_SessionChanged;
sentinel.EaseOfAccessChanged += Sentinel_EaseOfAccessChanged;
sentinel.SessionChanged += Sentinel_SessionChanged;
sentinel.StickyKeysChanged += Sentinel_StickyKeysChanged;
taskbar.LoseFocusRequested += Taskbar_LoseFocusRequested; taskbar.LoseFocusRequested += Taskbar_LoseFocusRequested;
taskbar.QuitButtonClicked += Shell_QuitButtonClicked; taskbar.QuitButtonClicked += Shell_QuitButtonClicked;
@ -247,11 +239,9 @@ namespace SafeExamBrowser.Client
applicationMonitor.ExplorerStarted -= ApplicationMonitor_ExplorerStarted; applicationMonitor.ExplorerStarted -= ApplicationMonitor_ExplorerStarted;
applicationMonitor.TerminationFailed -= ApplicationMonitor_TerminationFailed; applicationMonitor.TerminationFailed -= ApplicationMonitor_TerminationFailed;
displayMonitor.DisplayChanged -= DisplayMonitor_DisplaySettingsChanged; displayMonitor.DisplayChanged -= DisplayMonitor_DisplaySettingsChanged;
registry.ValueChanged -= Registry_ValueChanged;
runtime.ConnectionLost -= Runtime_ConnectionLost; runtime.ConnectionLost -= Runtime_ConnectionLost;
sentinel.CursorChanged -= Sentinel_CursorChanged; systemMonitor.SessionChanged -= SystemMonitor_SessionChanged;
sentinel.EaseOfAccessChanged -= Sentinel_EaseOfAccessChanged;
sentinel.SessionChanged -= Sentinel_SessionChanged;
sentinel.StickyKeysChanged -= Sentinel_StickyKeysChanged;
taskbar.LoseFocusRequested -= Taskbar_LoseFocusRequested; taskbar.LoseFocusRequested -= Taskbar_LoseFocusRequested;
taskbar.QuitButtonClicked -= Shell_QuitButtonClicked; taskbar.QuitButtonClicked -= Shell_QuitButtonClicked;
@ -289,12 +279,12 @@ namespace SafeExamBrowser.Client
private void CloseShell() private void CloseShell()
{ {
if (Settings?.UserInterface.ActionCenter.EnableActionCenter == true) if (Settings?.ActionCenter.EnableActionCenter == true)
{ {
actionCenter.Close(); actionCenter.Close();
} }
if (Settings?.UserInterface.Taskbar.EnableTaskbar == true) if (Settings?.Taskbar.EnableTaskbar == true)
{ {
taskbar.Close(); taskbar.Close();
} }
@ -302,12 +292,12 @@ namespace SafeExamBrowser.Client
private void ShowShell() private void ShowShell()
{ {
if (Settings.UserInterface.ActionCenter.EnableActionCenter) if (Settings.ActionCenter.EnableActionCenter)
{ {
actionCenter.Promote(); actionCenter.Promote();
} }
if (Settings.UserInterface.Taskbar.EnableTaskbar) if (Settings.Taskbar.EnableTaskbar)
{ {
taskbar.Show(); taskbar.Show();
} }
@ -334,7 +324,6 @@ namespace SafeExamBrowser.Client
private void PrepareShutdown() private void PrepareShutdown()
{ {
FinalizeProctoring(); FinalizeProctoring();
StopMonitoring();
} }
private void FinalizeProctoring() private void FinalizeProctoring()
@ -366,33 +355,28 @@ namespace SafeExamBrowser.Client
timer.Elapsed += (o, args) => VerifyApplicationIntegrity(); timer.Elapsed += (o, args) => VerifyApplicationIntegrity();
timer.Interval = TEN_MINUTES + (new Random().NextDouble() * FIVE_MINUTES); timer.Interval = TEN_MINUTES + (new Random().NextDouble() * FIVE_MINUTES);
timer.Start(); timer.Start();
}
private void StartMonitoring()
{
sentinel.StartMonitoringSystemEvents();
if (!Settings.Security.AllowStickyKeys)
{
sentinel.StartMonitoringStickyKeys();
}
if (Settings.Security.VerifyCursorConfiguration) if (Settings.Security.VerifyCursorConfiguration)
{ {
sentinel.StartMonitoringCursors(); if (registry.TryGetNames(RegistryValue.UserHive.Cursors_Key, out var names))
{
foreach (var name in names)
{
registry.StartMonitoring(RegistryValue.UserHive.Cursors_Key, name);
}
}
else
{
logger.Warn("Failed to start monitoring cursor registry values!");
}
} }
if (Settings.Service.IgnoreService) if (Settings.Service.IgnoreService)
{ {
sentinel.StartMonitoringEaseOfAccess(); registry.StartMonitoring(RegistryValue.MachineHive.EaseOfAccess_Key, RegistryValue.MachineHive.EaseOfAccess_Name);
} }
} }
private void StopMonitoring()
{
sentinel.StopMonitoring();
}
private void VerifyApplicationIntegrity() private void VerifyApplicationIntegrity()
{ {
logger.Info($"Attempting to verify application integrity..."); logger.Info($"Attempting to verify application integrity...");
@ -419,7 +403,7 @@ namespace SafeExamBrowser.Client
{ {
var hasQuitPassword = !string.IsNullOrEmpty(Settings.Security.QuitPasswordHash); var hasQuitPassword = !string.IsNullOrEmpty(Settings.Security.QuitPasswordHash);
if (hasQuitPassword && Settings.Security.VerifySessionIntegrity) if (hasQuitPassword)
{ {
logger.Info($"Attempting to verify session integrity..."); logger.Info($"Attempting to verify session integrity...");
@ -456,12 +440,17 @@ namespace SafeExamBrowser.Client
} }
} }
private void TerminateIntegrityVerification()
{
registry.StopMonitoring();
}
private void ApplicationMonitor_ExplorerStarted() private void ApplicationMonitor_ExplorerStarted()
{ {
logger.Info("Trying to terminate Windows explorer..."); logger.Info("Trying to terminate Windows explorer...");
explorerShell.Terminate(); explorerShell.Terminate();
logger.Info("Re-initializing working area..."); logger.Info("Re-initializing working area...");
displayMonitor.InitializePrimaryDisplay(Settings.UserInterface.Taskbar.EnableTaskbar ? taskbar.GetAbsoluteHeight() : 0); displayMonitor.InitializePrimaryDisplay(Settings.Taskbar.EnableTaskbar ? taskbar.GetAbsoluteHeight() : 0);
logger.Info("Re-initializing shell..."); logger.Info("Re-initializing shell...");
actionCenter.InitializeBounds(); actionCenter.InitializeBounds();
taskbar.InitializeBounds(); taskbar.InitializeBounds();
@ -492,36 +481,6 @@ namespace SafeExamBrowser.Client
} }
private void Browser_ConfigurationDownloadRequested(string fileName, DownloadEventArgs args) private void Browser_ConfigurationDownloadRequested(string fileName, DownloadEventArgs args)
{
args.AllowDownload = false;
if (IsAllowedToReconfigure(args.Url))
{
if (coordinator.RequestReconfigurationLock())
{
args.AllowDownload = true;
args.Callback = Browser_ConfigurationDownloadFinished;
args.DownloadPath = Path.Combine(context.AppConfig.TemporaryDirectory, fileName);
splashScreen.Show();
splashScreen.BringToForeground();
splashScreen.SetIndeterminate();
splashScreen.UpdateStatus(TextKey.OperationStatus_InitializeSession, true);
logger.Info($"Allowed download request for configuration file '{fileName}'.");
}
else
{
logger.Warn($"A reconfiguration is already in progress, denied download request for configuration file '{fileName}'!");
}
}
else
{
logger.Info($"Reconfiguration is not allowed, denied download request for configuration file '{fileName}'.");
}
}
private bool IsAllowedToReconfigure(string url)
{ {
var allow = false; var allow = false;
var hasQuitPassword = !string.IsNullOrWhiteSpace(Settings.Security.QuitPasswordHash); var hasQuitPassword = !string.IsNullOrWhiteSpace(Settings.Security.QuitPasswordHash);
@ -533,9 +492,9 @@ namespace SafeExamBrowser.Client
{ {
var expression = Regex.Escape(Settings.Security.ReconfigurationUrl).Replace(@"\*", ".*"); var expression = Regex.Escape(Settings.Security.ReconfigurationUrl).Replace(@"\*", ".*");
var regex = new Regex($"^{expression}$", RegexOptions.IgnoreCase); var regex = new Regex($"^{expression}$", RegexOptions.IgnoreCase);
var sebUrl = url.Replace(Uri.UriSchemeHttps, context.AppConfig.SebUriSchemeSecure).Replace(Uri.UriSchemeHttp, context.AppConfig.SebUriScheme); var sebUrl = args.Url.Replace(Uri.UriSchemeHttps, context.AppConfig.SebUriSchemeSecure).Replace(Uri.UriSchemeHttp, context.AppConfig.SebUriScheme);
allow = Settings.Security.AllowReconfiguration && (regex.IsMatch(url) || regex.IsMatch(sebUrl)); allow = Settings.Security.AllowReconfiguration && (regex.IsMatch(args.Url) || regex.IsMatch(sebUrl));
} }
else else
{ {
@ -547,7 +506,24 @@ namespace SafeExamBrowser.Client
allow = Settings.ConfigurationMode == ConfigurationMode.ConfigureClient || Settings.Security.AllowReconfiguration; allow = Settings.ConfigurationMode == ConfigurationMode.ConfigureClient || Settings.Security.AllowReconfiguration;
} }
return allow; if (allow)
{
args.AllowDownload = true;
args.Callback = Browser_ConfigurationDownloadFinished;
args.DownloadPath = Path.Combine(context.AppConfig.TemporaryDirectory, fileName);
splashScreen.Show();
splashScreen.BringToForeground();
splashScreen.SetIndeterminate();
splashScreen.UpdateStatus(TextKey.OperationStatus_InitializeSession, true);
logger.Info($"Allowed download request for configuration file '{fileName}'.");
}
else
{
args.AllowDownload = false;
logger.Info($"Denied download request for configuration file '{fileName}'.");
}
} }
private void Browser_ConfigurationDownloadFinished(bool success, string url, string filePath = null) private void Browser_ConfigurationDownloadFinished(bool success, string url, string filePath = null)
@ -565,19 +541,15 @@ namespace SafeExamBrowser.Client
else else
{ {
logger.Error($"Failed to communicate reconfiguration request for '{filePath}'!"); logger.Error($"Failed to communicate reconfiguration request for '{filePath}'!");
messageBox.Show(TextKey.MessageBox_ReconfigurationError, TextKey.MessageBox_ReconfigurationErrorTitle, icon: MessageBoxIcon.Error, parent: splashScreen); messageBox.Show(TextKey.MessageBox_ReconfigurationError, TextKey.MessageBox_ReconfigurationErrorTitle, icon: MessageBoxIcon.Error, parent: splashScreen);
splashScreen.Hide(); splashScreen.Hide();
coordinator.ReleaseReconfigurationLock();
} }
} }
else else
{ {
logger.Error($"Failed to download configuration file '{filePath}'!"); logger.Error($"Failed to download configuration file '{filePath}'!");
messageBox.Show(TextKey.MessageBox_ConfigurationDownloadError, TextKey.MessageBox_ConfigurationDownloadErrorTitle, icon: MessageBoxIcon.Error, parent: splashScreen); messageBox.Show(TextKey.MessageBox_ConfigurationDownloadError, TextKey.MessageBox_ConfigurationDownloadErrorTitle, icon: MessageBoxIcon.Error, parent: splashScreen);
splashScreen.Hide(); splashScreen.Hide();
coordinator.ReleaseReconfigurationLock();
} }
} }
@ -665,7 +637,6 @@ namespace SafeExamBrowser.Client
{ {
logger.Info("The reconfiguration was aborted by the runtime."); logger.Info("The reconfiguration was aborted by the runtime.");
splashScreen.Hide(); splashScreen.Hide();
coordinator.ReleaseReconfigurationLock();
} }
private void ClientHost_ReconfigurationDenied(ReconfigurationEventArgs args) private void ClientHost_ReconfigurationDenied(ReconfigurationEventArgs args)
@ -673,7 +644,6 @@ namespace SafeExamBrowser.Client
logger.Info($"The reconfiguration request for '{args.ConfigurationPath}' was denied by the runtime!"); logger.Info($"The reconfiguration request for '{args.ConfigurationPath}' was denied by the runtime!");
messageBox.Show(TextKey.MessageBox_ReconfigurationDenied, TextKey.MessageBox_ReconfigurationDeniedTitle, parent: splashScreen); messageBox.Show(TextKey.MessageBox_ReconfigurationDenied, TextKey.MessageBox_ReconfigurationDeniedTitle, parent: splashScreen);
splashScreen.Hide(); splashScreen.Hide();
coordinator.ReleaseReconfigurationLock();
} }
private void ClientHost_ServerFailureActionRequested(ServerFailureActionRequestEventArgs args) private void ClientHost_ServerFailureActionRequested(ServerFailureActionRequestEventArgs args)
@ -695,7 +665,7 @@ namespace SafeExamBrowser.Client
private void DisplayMonitor_DisplaySettingsChanged() private void DisplayMonitor_DisplaySettingsChanged()
{ {
logger.Info("Re-initializing working area..."); logger.Info("Re-initializing working area...");
displayMonitor.InitializePrimaryDisplay(Settings.UserInterface.Taskbar.EnableTaskbar ? taskbar.GetAbsoluteHeight() : 0); displayMonitor.InitializePrimaryDisplay(Settings.Taskbar.EnableTaskbar ? taskbar.GetAbsoluteHeight() : 0);
logger.Info("Re-initializing shell..."); logger.Info("Re-initializing shell...");
actionCenter.InitializeBounds(); actionCenter.InitializeBounds();
@ -720,18 +690,6 @@ namespace SafeExamBrowser.Client
} }
} }
private void NetworkAdapter_CredentialsRequired(CredentialsRequiredEventArgs args)
{
var message = text.Get(TextKey.CredentialsDialog_WirelessNetworkMessage).Replace("%%_NAME_%%", args.NetworkName);
var title = text.Get(TextKey.CredentialsDialog_WirelessNetworkTitle);
var dialog = uiFactory.CreateCredentialsDialog(CredentialsDialogPurpose.WirelessNetwork, message, title);
var result = dialog.Show();
args.Password = result.Password;
args.Success = result.Success;
args.Username = result.Username;
}
private void Operations_ActionRequired(ActionRequiredEventArgs args) private void Operations_ActionRequired(ActionRequiredEventArgs args)
{ {
switch (args) switch (args)
@ -784,25 +742,31 @@ namespace SafeExamBrowser.Client
splashScreen.UpdateStatus(status, true); splashScreen.UpdateStatus(status, true);
} }
private void Runtime_ConnectionLost() private void Registry_ValueChanged(string key, string name, object oldValue, object newValue)
{ {
logger.Error("Lost connection to the runtime!"); if (key == RegistryValue.UserHive.Cursors_Key)
{
messageBox.Show(TextKey.MessageBox_ApplicationError, TextKey.MessageBox_ApplicationErrorTitle, icon: MessageBoxIcon.Error); HandleCursorRegistryChange(key, name, oldValue, newValue);
shutdown.Invoke(); }
else if (key == RegistryValue.MachineHive.EaseOfAccess_Key)
{
HandleEaseOfAccessRegistryChange(key, name, oldValue, newValue);
}
} }
private void Sentinel_CursorChanged(SentinelEventArgs args) private void HandleCursorRegistryChange(string key, string name, object oldValue, object newValue)
{ {
if (coordinator.RequestSessionLock()) logger.Warn($@"The cursor registry value '{key}\{name}' has changed from '{oldValue}' to '{newValue}'! Attempting to show lock screen...");
if (!sessionLocked)
{ {
var message = text.Get(TextKey.LockScreen_CursorMessage); var message = text.Get(TextKey.LockScreen_CursorMessage);
var title = text.Get(TextKey.LockScreen_Title); var title = text.Get(TextKey.LockScreen_Title);
var continueOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_CursorContinueOption) }; var continueOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_CursorContinueOption) };
var terminateOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_CursorTerminateOption) }; var terminateOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_CursorTerminateOption) };
args.Allow = true; sessionLocked = true;
logger.Info("Cursor changed! Attempting to show lock screen..."); registry.StopMonitoring(key, name);
var result = ShowLockScreen(message, title, new[] { continueOption, terminateOption }); var result = ShowLockScreen(message, title, new[] { continueOption, terminateOption });
@ -816,25 +780,27 @@ namespace SafeExamBrowser.Client
TryRequestShutdown(); TryRequestShutdown();
} }
coordinator.ReleaseSessionLock(); sessionLocked = false;
} }
else else
{ {
logger.Info("Cursor changed but lock screen is already active."); logger.Info("Lock screen is already active.");
} }
} }
private void Sentinel_EaseOfAccessChanged(SentinelEventArgs args) private void HandleEaseOfAccessRegistryChange(string key, string name, object oldValue, object newValue)
{ {
if (coordinator.RequestSessionLock()) logger.Warn($@"The ease of access registry value '{key}\{name}' has changed from '{oldValue}' to '{newValue}'! Attempting to show lock screen...");
if (!sessionLocked)
{ {
var message = text.Get(TextKey.LockScreen_EaseOfAccessMessage); var message = text.Get(TextKey.LockScreen_EaseOfAccessMessage);
var title = text.Get(TextKey.LockScreen_Title); var title = text.Get(TextKey.LockScreen_Title);
var continueOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_EaseOfAccessContinueOption) }; var continueOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_EaseOfAccessContinueOption) };
var terminateOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_EaseOfAccessTerminateOption) }; var terminateOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_EaseOfAccessTerminateOption) };
args.Allow = true; sessionLocked = true;
logger.Info("Ease of access changed! Attempting to show lock screen..."); registry.StopMonitoring(key, name);
var result = ShowLockScreen(message, title, new[] { continueOption, terminateOption }); var result = ShowLockScreen(message, title, new[] { continueOption, terminateOption });
@ -848,85 +814,20 @@ namespace SafeExamBrowser.Client
TryRequestShutdown(); TryRequestShutdown();
} }
coordinator.ReleaseSessionLock(); sessionLocked = false;
} }
else else
{ {
logger.Info("Ease of access changed but lock screen is already active."); logger.Info("Lock screen is already active.");
} }
} }
private void Sentinel_SessionChanged() private void Runtime_ConnectionLost()
{ {
var allow = !Settings.Service.IgnoreService && (!Settings.Service.DisableUserLock || !Settings.Service.DisableUserSwitch); logger.Error("Lost connection to the runtime!");
var disable = Settings.Security.DisableSessionChangeLockScreen; messageBox.Show(TextKey.MessageBox_ApplicationError, TextKey.MessageBox_ApplicationErrorTitle, icon: MessageBoxIcon.Error);
if (allow || disable) shutdown.Invoke();
{
logger.Info($"Detected user session change, but {(allow ? "session locking and/or switching is allowed" : "lock screen is deactivated")}.");
}
else
{
var message = text.Get(TextKey.LockScreen_UserSessionMessage);
var title = text.Get(TextKey.LockScreen_Title);
var continueOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_UserSessionContinueOption) };
var terminateOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_UserSessionTerminateOption) };
logger.Warn("User session changed! Attempting to show lock screen...");
if (coordinator.RequestSessionLock())
{
var result = ShowLockScreen(message, title, new[] { continueOption, terminateOption });
if (result.OptionId == continueOption.Id)
{
logger.Info("The session will be allowed to resume as requested by the user...");
}
else if (result.OptionId == terminateOption.Id)
{
logger.Info("Attempting to shutdown as requested by the user...");
TryRequestShutdown();
}
coordinator.ReleaseSessionLock();
}
else
{
logger.Warn("User session changed but lock screen is already active.");
}
}
}
private void Sentinel_StickyKeysChanged(SentinelEventArgs args)
{
if (coordinator.RequestSessionLock())
{
var message = text.Get(TextKey.LockScreen_StickyKeysMessage);
var title = text.Get(TextKey.LockScreen_Title);
var continueOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_StickyKeysContinueOption) };
var terminateOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_StickyKeysTerminateOption) };
args.Allow = true;
logger.Info("Sticky keys changed! Attempting to show lock screen...");
var result = ShowLockScreen(message, title, new[] { continueOption, terminateOption });
if (result.OptionId == continueOption.Id)
{
logger.Info("The session will be allowed to resume as requested by the user...");
}
else if (result.OptionId == terminateOption.Id)
{
logger.Info("Attempting to shutdown as requested by the user...");
TryRequestShutdown();
}
coordinator.ReleaseSessionLock();
}
else
{
logger.Info("Sticky keys changed but lock screen is already active.");
}
} }
private void Server_LockScreenConfirmed() private void Server_LockScreenConfirmed()
@ -939,10 +840,11 @@ namespace SafeExamBrowser.Client
{ {
logger.Info("Attempting to show lock screen as requested by the server..."); logger.Info("Attempting to show lock screen as requested by the server...");
if (coordinator.RequestSessionLock()) if (!sessionLocked)
{ {
sessionLocked = true;
ShowLockScreen(message, text.Get(TextKey.LockScreen_Title), Enumerable.Empty<LockScreenOption>()); ShowLockScreen(message, text.Get(TextKey.LockScreen_Title), Enumerable.Empty<LockScreenOption>());
coordinator.ReleaseSessionLock(); sessionLocked = false;
} }
else else
{ {
@ -963,6 +865,44 @@ namespace SafeExamBrowser.Client
ResumeActivators(); ResumeActivators();
} }
private void SystemMonitor_SessionChanged()
{
var allow = !Settings.Service.IgnoreService && (!Settings.Service.DisableUserLock || !Settings.Service.DisableUserSwitch);
var disable = Settings.Security.DisableSessionChangeLockScreen;
var message = text.Get(TextKey.LockScreen_UserSessionMessage);
var title = text.Get(TextKey.LockScreen_Title);
var continueOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_UserSessionContinueOption) };
var terminateOption = new LockScreenOption { Text = text.Get(TextKey.LockScreen_UserSessionTerminateOption) };
if (allow || disable)
{
logger.Info($"Detected user session change, but {(allow ? "session locking and/or switching is allowed" : "lock screen is deactivated")}.");
}
else
{
logger.Warn("Detected user session change!");
if (!sessionLocked)
{
sessionLocked = true;
var result = ShowLockScreen(message, title, new[] { continueOption, terminateOption });
if (result.OptionId == terminateOption.Id)
{
logger.Info("Attempting to shutdown as requested by the user...");
TryRequestShutdown();
}
sessionLocked = false;
}
else
{
logger.Info("Lock screen is already active.");
}
}
}
private void Taskbar_LoseFocusRequested(bool forward) private void Taskbar_LoseFocusRequested(bool forward)
{ {
Browser.Focus(forward); Browser.Focus(forward);
@ -1050,7 +990,7 @@ namespace SafeExamBrowser.Client
logger.Info("Showing lock screen..."); logger.Info("Showing lock screen...");
PauseActivators(); PauseActivators();
lockScreen = uiFactory.CreateLockScreen(message, title, options, Settings.UserInterface.LockScreen); lockScreen = uiFactory.CreateLockScreen(message, title, options);
lockScreen.Show(); lockScreen.Show();
if (Settings.SessionMode == SessionMode.Server) if (Settings.SessionMode == SessionMode.Server)

View file

@ -114,13 +114,12 @@ namespace SafeExamBrowser.Client
var applicationFactory = new ApplicationFactory(applicationMonitor, ModuleLogger(nameof(ApplicationFactory)), nativeMethods, processFactory, new Registry(ModuleLogger(nameof(Registry)))); var applicationFactory = new ApplicationFactory(applicationMonitor, ModuleLogger(nameof(ApplicationFactory)), nativeMethods, processFactory, new Registry(ModuleLogger(nameof(Registry))));
var clipboard = new Clipboard(ModuleLogger(nameof(Clipboard)), nativeMethods); var clipboard = new Clipboard(ModuleLogger(nameof(Clipboard)), nativeMethods);
var coordinator = new Coordinator();
var displayMonitor = new DisplayMonitor(ModuleLogger(nameof(DisplayMonitor)), nativeMethods, systemInfo); var displayMonitor = new DisplayMonitor(ModuleLogger(nameof(DisplayMonitor)), nativeMethods, systemInfo);
var explorerShell = new ExplorerShell(ModuleLogger(nameof(ExplorerShell)), nativeMethods); var explorerShell = new ExplorerShell(ModuleLogger(nameof(ExplorerShell)), nativeMethods);
var fileSystemDialog = BuildFileSystemDialog(); var fileSystemDialog = BuildFileSystemDialog();
var hashAlgorithm = new HashAlgorithm(); var hashAlgorithm = new HashAlgorithm();
var sentinel = new SystemSentinel(ModuleLogger(nameof(SystemSentinel)), nativeMethods, registry);
var splashScreen = uiFactory.CreateSplashScreen(); var splashScreen = uiFactory.CreateSplashScreen();
var systemMonitor = new SystemMonitor(ModuleLogger(nameof(SystemMonitor)));
var operations = new Queue<IOperation>(); var operations = new Queue<IOperation>();
@ -136,6 +135,7 @@ namespace SafeExamBrowser.Client
operations.Enqueue(new LazyInitializationOperation(BuildMouseInterceptorOperation)); operations.Enqueue(new LazyInitializationOperation(BuildMouseInterceptorOperation));
operations.Enqueue(new ApplicationOperation(context, applicationFactory, applicationMonitor, logger, text)); operations.Enqueue(new ApplicationOperation(context, applicationFactory, applicationMonitor, logger, text));
operations.Enqueue(new DisplayMonitorOperation(context, displayMonitor, logger, taskbar)); operations.Enqueue(new DisplayMonitorOperation(context, displayMonitor, logger, taskbar));
operations.Enqueue(new SystemMonitorOperation(context, systemMonitor, logger));
operations.Enqueue(new LazyInitializationOperation(BuildShellOperation)); operations.Enqueue(new LazyInitializationOperation(BuildShellOperation));
operations.Enqueue(new LazyInitializationOperation(BuildBrowserOperation)); operations.Enqueue(new LazyInitializationOperation(BuildBrowserOperation));
operations.Enqueue(new LazyInitializationOperation(BuildServerOperation)); operations.Enqueue(new LazyInitializationOperation(BuildServerOperation));
@ -148,19 +148,18 @@ namespace SafeExamBrowser.Client
actionCenter, actionCenter,
applicationMonitor, applicationMonitor,
context, context,
coordinator,
displayMonitor, displayMonitor,
explorerShell, explorerShell,
fileSystemDialog, fileSystemDialog,
hashAlgorithm, hashAlgorithm,
logger, logger,
messageBox, messageBox,
networkAdapter,
sequence, sequence,
registry,
runtimeProxy, runtimeProxy,
shutdown, shutdown,
splashScreen, splashScreen,
sentinel, systemMonitor,
taskbar, taskbar,
text, text,
uiFactory); uiFactory);

View file

@ -1,46 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.Client.Contracts
{
/// <summary>
/// Coordinates concurrent operations of the client application.
/// </summary>
internal interface ICoordinator
{
/// <summary>
/// Indicates whether the reconfiguration lock is currently occupied.
/// </summary>
bool IsReconfigurationLocked();
/// <summary>
/// Indicates whether the session lock is currently occupied.
/// </summary>
bool IsSessionLocked();
/// <summary>
/// Releases the reconfiguration lock.
/// </summary>
void ReleaseReconfigurationLock();
/// <summary>
/// Releases the session lock.
/// </summary>
void ReleaseSessionLock();
/// <summary>
/// Attempts to acquire the unique reconfiguration lock. Returns <c>true</c> if successful, otherwise <c>false</c>.
/// </summary>
bool RequestReconfigurationLock();
/// <summary>
/// Attempts to acquire the unique session lock. Returns <c>true</c> if successful, otherwise <c>false</c>.
/// </summary>
bool RequestSessionLock();
}
}

View file

@ -1,78 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.Collections.Concurrent;
using SafeExamBrowser.Client.Contracts;
namespace SafeExamBrowser.Client
{
internal class Coordinator : ICoordinator
{
private readonly ConcurrentBag<Guid> reconfiguration;
private readonly ConcurrentBag<Guid> session;
internal Coordinator()
{
reconfiguration = new ConcurrentBag<Guid>();
session = new ConcurrentBag<Guid>();
}
public bool IsReconfigurationLocked()
{
return !reconfiguration.IsEmpty;
}
public bool IsSessionLocked()
{
return !session.IsEmpty;
}
public void ReleaseReconfigurationLock()
{
reconfiguration.TryTake(out _);
}
public void ReleaseSessionLock()
{
session.TryTake(out _);
}
public bool RequestReconfigurationLock()
{
var acquired = false;
lock (reconfiguration)
{
if (reconfiguration.IsEmpty)
{
reconfiguration.Add(Guid.NewGuid());
acquired = true;
}
}
return acquired;
}
public bool RequestSessionLock()
{
var acquired = false;
lock (session)
{
if (session.IsEmpty)
{
session.Add(Guid.NewGuid());
acquired = true;
}
}
return acquired;
}
}
}

View file

@ -50,12 +50,12 @@ namespace SafeExamBrowser.Client.Operations
{ {
Context.Browser.Initialize(); Context.Browser.Initialize();
if (Context.Settings.UserInterface.ActionCenter.EnableActionCenter) if (Context.Settings.ActionCenter.EnableActionCenter)
{ {
actionCenter.AddApplicationControl(uiFactory.CreateApplicationControl(Context.Browser, Location.ActionCenter), true); actionCenter.AddApplicationControl(uiFactory.CreateApplicationControl(Context.Browser, Location.ActionCenter), true);
} }
if (Context.Settings.UserInterface.Taskbar.EnableTaskbar) if (Context.Settings.Taskbar.EnableTaskbar)
{ {
taskbar.AddApplicationControl(uiFactory.CreateApplicationControl(Context.Browser, Location.Taskbar), true); taskbar.AddApplicationControl(uiFactory.CreateApplicationControl(Context.Browser, Location.Taskbar), true);
} }

View file

@ -36,7 +36,7 @@ namespace SafeExamBrowser.Client.Operations
logger.Info("Initializing working area..."); logger.Info("Initializing working area...");
StatusChanged?.Invoke(TextKey.OperationStatus_InitializeWorkingArea); StatusChanged?.Invoke(TextKey.OperationStatus_InitializeWorkingArea);
displayMonitor.InitializePrimaryDisplay(Context.Settings.UserInterface.Taskbar.EnableTaskbar ? taskbar.GetAbsoluteHeight() : 0); displayMonitor.InitializePrimaryDisplay(Context.Settings.Taskbar.EnableTaskbar ? taskbar.GetAbsoluteHeight() : 0);
displayMonitor.StartMonitoringDisplayChanges(); displayMonitor.StartMonitoringDisplayChanges();
return OperationResult.Success; return OperationResult.Success;

View file

@ -107,7 +107,7 @@ namespace SafeExamBrowser.Client.Operations
{ {
foreach (var activator in Context.Activators) foreach (var activator in Context.Activators)
{ {
if (Context.Settings.UserInterface.ActionCenter.EnableActionCenter && activator is IActionCenterActivator actionCenterActivator) if (Context.Settings.ActionCenter.EnableActionCenter && activator is IActionCenterActivator actionCenterActivator)
{ {
actionCenter.Register(actionCenterActivator); actionCenter.Register(actionCenterActivator);
actionCenterActivator.Start(); actionCenterActivator.Start();
@ -124,7 +124,7 @@ namespace SafeExamBrowser.Client.Operations
terminationActivator.Start(); terminationActivator.Start();
} }
if (Context.Settings.UserInterface.Taskbar.EnableTaskbar && activator is ITaskbarActivator taskbarActivator) if (Context.Settings.Taskbar.EnableTaskbar && activator is ITaskbarActivator taskbarActivator)
{ {
taskbar.Register(taskbarActivator); taskbar.Register(taskbarActivator);
taskbarActivator.Start(); taskbarActivator.Start();
@ -134,7 +134,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeActionCenter() private void InitializeActionCenter()
{ {
if (Context.Settings.UserInterface.ActionCenter.EnableActionCenter) if (Context.Settings.ActionCenter.EnableActionCenter)
{ {
logger.Info("Initializing action center..."); logger.Info("Initializing action center...");
actionCenter.InitializeText(text); actionCenter.InitializeText(text);
@ -168,7 +168,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeTaskbar() private void InitializeTaskbar()
{ {
if (Context.Settings.UserInterface.Taskbar.EnableTaskbar) if (Context.Settings.Taskbar.EnableTaskbar)
{ {
logger.Info("Initializing taskbar..."); logger.Info("Initializing taskbar...");
taskbar.InitializeText(text); taskbar.InitializeText(text);
@ -232,7 +232,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeAboutNotificationForActionCenter() private void InitializeAboutNotificationForActionCenter()
{ {
if (Context.Settings.UserInterface.ActionCenter.ShowApplicationInfo) if (Context.Settings.ActionCenter.ShowApplicationInfo)
{ {
actionCenter.AddNotificationControl(uiFactory.CreateNotificationControl(aboutNotification, Location.ActionCenter)); actionCenter.AddNotificationControl(uiFactory.CreateNotificationControl(aboutNotification, Location.ActionCenter));
} }
@ -240,7 +240,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeAboutNotificationForTaskbar() private void InitializeAboutNotificationForTaskbar()
{ {
if (Context.Settings.UserInterface.Taskbar.ShowApplicationInfo) if (Context.Settings.Taskbar.ShowApplicationInfo)
{ {
taskbar.AddNotificationControl(uiFactory.CreateNotificationControl(aboutNotification, Location.Taskbar)); taskbar.AddNotificationControl(uiFactory.CreateNotificationControl(aboutNotification, Location.Taskbar));
} }
@ -248,7 +248,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeAudioForActionCenter() private void InitializeAudioForActionCenter()
{ {
if (Context.Settings.UserInterface.ActionCenter.ShowAudio) if (Context.Settings.ActionCenter.ShowAudio)
{ {
actionCenter.AddSystemControl(uiFactory.CreateAudioControl(audio, Location.ActionCenter)); actionCenter.AddSystemControl(uiFactory.CreateAudioControl(audio, Location.ActionCenter));
} }
@ -256,7 +256,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeAudioForTaskbar() private void InitializeAudioForTaskbar()
{ {
if (Context.Settings.UserInterface.Taskbar.ShowAudio) if (Context.Settings.Taskbar.ShowAudio)
{ {
taskbar.AddSystemControl(uiFactory.CreateAudioControl(audio, Location.Taskbar)); taskbar.AddSystemControl(uiFactory.CreateAudioControl(audio, Location.Taskbar));
} }
@ -264,17 +264,17 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeClockForActionCenter() private void InitializeClockForActionCenter()
{ {
actionCenter.ShowClock = Context.Settings.UserInterface.ActionCenter.ShowClock; actionCenter.ShowClock = Context.Settings.ActionCenter.ShowClock;
} }
private void InitializeClockForTaskbar() private void InitializeClockForTaskbar()
{ {
taskbar.ShowClock = Context.Settings.UserInterface.Taskbar.ShowClock; taskbar.ShowClock = Context.Settings.Taskbar.ShowClock;
} }
private void InitializeLogNotificationForActionCenter() private void InitializeLogNotificationForActionCenter()
{ {
if (Context.Settings.UserInterface.ActionCenter.ShowApplicationLog) if (Context.Settings.ActionCenter.ShowApplicationLog)
{ {
actionCenter.AddNotificationControl(uiFactory.CreateNotificationControl(logNotification, Location.ActionCenter)); actionCenter.AddNotificationControl(uiFactory.CreateNotificationControl(logNotification, Location.ActionCenter));
} }
@ -282,7 +282,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeLogNotificationForTaskbar() private void InitializeLogNotificationForTaskbar()
{ {
if (Context.Settings.UserInterface.Taskbar.ShowApplicationLog) if (Context.Settings.Taskbar.ShowApplicationLog)
{ {
taskbar.AddNotificationControl(uiFactory.CreateNotificationControl(logNotification, Location.Taskbar)); taskbar.AddNotificationControl(uiFactory.CreateNotificationControl(logNotification, Location.Taskbar));
} }
@ -290,7 +290,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeKeyboardLayoutForActionCenter() private void InitializeKeyboardLayoutForActionCenter()
{ {
if (Context.Settings.UserInterface.ActionCenter.ShowKeyboardLayout) if (Context.Settings.ActionCenter.ShowKeyboardLayout)
{ {
actionCenter.AddSystemControl(uiFactory.CreateKeyboardLayoutControl(keyboard, Location.ActionCenter)); actionCenter.AddSystemControl(uiFactory.CreateKeyboardLayoutControl(keyboard, Location.ActionCenter));
} }
@ -298,7 +298,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeKeyboardLayoutForTaskbar() private void InitializeKeyboardLayoutForTaskbar()
{ {
if (Context.Settings.UserInterface.Taskbar.ShowKeyboardLayout) if (Context.Settings.Taskbar.ShowKeyboardLayout)
{ {
taskbar.AddSystemControl(uiFactory.CreateKeyboardLayoutControl(keyboard, Location.Taskbar)); taskbar.AddSystemControl(uiFactory.CreateKeyboardLayoutControl(keyboard, Location.Taskbar));
} }
@ -332,7 +332,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeNetworkForActionCenter() private void InitializeNetworkForActionCenter()
{ {
if (Context.Settings.UserInterface.ActionCenter.ShowNetwork) if (Context.Settings.ActionCenter.ShowNetwork)
{ {
actionCenter.AddSystemControl(uiFactory.CreateNetworkControl(networkAdapter, Location.ActionCenter)); actionCenter.AddSystemControl(uiFactory.CreateNetworkControl(networkAdapter, Location.ActionCenter));
} }
@ -340,7 +340,7 @@ namespace SafeExamBrowser.Client.Operations
private void InitializeNetworkForTaskbar() private void InitializeNetworkForTaskbar()
{ {
if (Context.Settings.UserInterface.Taskbar.ShowNetwork) if (Context.Settings.Taskbar.ShowNetwork)
{ {
taskbar.AddSystemControl(uiFactory.CreateNetworkControl(networkAdapter, Location.Taskbar)); taskbar.AddSystemControl(uiFactory.CreateNetworkControl(networkAdapter, Location.Taskbar));
} }

View file

@ -0,0 +1,51 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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 SafeExamBrowser.Core.Contracts.OperationModel;
using SafeExamBrowser.Core.Contracts.OperationModel.Events;
using SafeExamBrowser.I18n.Contracts;
using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.System;
namespace SafeExamBrowser.Client.Operations
{
internal class SystemMonitorOperation : ClientOperation
{
private readonly ILogger logger;
private readonly ISystemMonitor systemMonitor;
public override event ActionRequiredEventHandler ActionRequired { add { } remove { } }
public override event StatusChangedEventHandler StatusChanged;
public SystemMonitorOperation(ClientContext context, ISystemMonitor systemMonitor, ILogger logger) : base(context)
{
this.logger = logger;
this.systemMonitor = systemMonitor;
}
public override OperationResult Perform()
{
logger.Info("Initializing system events...");
StatusChanged?.Invoke(TextKey.OperationStatus_InitializeSystemEvents);
systemMonitor.Start();
return OperationResult.Success;
}
public override OperationResult Revert()
{
logger.Info("Finalizing system events...");
StatusChanged?.Invoke(TextKey.OperationStatus_FinalizeSystemEvents);
systemMonitor.Stop();
return OperationResult.Success;
}
}
}

View file

@ -16,9 +16,6 @@ using System.Windows;
// to COM components. If you need to access a type in this assembly from // to COM components. If you need to access a type in this assembly from
// COM, set the ComVisible attribute to true on that type. // COM, set the ComVisible attribute to true on that type.
[assembly: ComVisible(false)] [assembly: ComVisible(false)]
// Required for mocking internal contracts with Moq
[assembly: InternalsVisibleTo("DynamicProxyGenAssembly2")]
[assembly: InternalsVisibleTo("SafeExamBrowser.Client.UnitTests")] [assembly: InternalsVisibleTo("SafeExamBrowser.Client.UnitTests")]
//In order to begin building localizable applications, set //In order to begin building localizable applications, set

View file

@ -74,8 +74,6 @@
<Compile Include="App.cs" /> <Compile Include="App.cs" />
<Compile Include="ClientContext.cs" /> <Compile Include="ClientContext.cs" />
<Compile Include="ClientController.cs" /> <Compile Include="ClientController.cs" />
<Compile Include="Contracts\ICoordinator.cs" />
<Compile Include="Coordinator.cs" />
<Compile Include="Operations\ClientHostDisconnectionOperation.cs" /> <Compile Include="Operations\ClientHostDisconnectionOperation.cs" />
<Compile Include="Operations\ClientOperation.cs" /> <Compile Include="Operations\ClientOperation.cs" />
<Compile Include="Operations\ConfigurationOperation.cs" /> <Compile Include="Operations\ConfigurationOperation.cs" />
@ -97,6 +95,7 @@
<Compile Include="Operations\ApplicationOperation.cs" /> <Compile Include="Operations\ApplicationOperation.cs" />
<Compile Include="Operations\ServerOperation.cs" /> <Compile Include="Operations\ServerOperation.cs" />
<Compile Include="Operations\ShellOperation.cs" /> <Compile Include="Operations\ShellOperation.cs" />
<Compile Include="Operations\SystemMonitorOperation.cs" />
<Compile Include="Properties\AssemblyInfo.cs"> <Compile Include="Properties\AssemblyInfo.cs">
<SubType>Code</SubType> <SubType>Code</SubType>
</Compile> </Compile>

View file

@ -26,11 +26,6 @@ namespace SafeExamBrowser.Configuration.Contracts
/// </summary> /// </summary>
public Guid ClientAuthenticationToken { get; set; } public Guid ClientAuthenticationToken { get; set; }
/// <summary>
/// Indicates whether a configuration resource needs to be loaded in the browser because it requires authentication or is a webpage.
/// </summary>
public bool IsBrowserResource { get; set; }
/// <summary> /// <summary>
/// The unique session identifier. /// The unique session identifier.
/// </summary> /// </summary>

View file

@ -1,62 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.Threading.Tasks;
using Microsoft.VisualStudio.TestTools.UnitTesting;
using Moq;
using SafeExamBrowser.Configuration.Contracts;
using SafeExamBrowser.Configuration.Contracts.Integrity;
using SafeExamBrowser.Configuration.Cryptography;
using SafeExamBrowser.Logging.Contracts;
namespace SafeExamBrowser.Configuration.UnitTests.Cryptography
{
[TestClass]
public class KeyGeneratorTests
{
private AppConfig appConfig;
private Mock<IIntegrityModule> integrityModule;
private Mock<ILogger> logger;
private KeyGenerator sut;
[TestInitialize]
public void Initialize()
{
appConfig = new AppConfig();
integrityModule = new Mock<IIntegrityModule>();
logger = new Mock<ILogger>();
sut = new KeyGenerator(appConfig, integrityModule.Object, logger.Object);
}
[TestMethod]
[ExpectedException(typeof(Exception), AllowDerivedTypes = true)]
public void CalculateBrowserExamKeyHash_MustFailWithoutUrl()
{
sut.CalculateBrowserExamKeyHash(default, default, default);
}
[TestMethod]
[ExpectedException(typeof(Exception), AllowDerivedTypes = true)]
public void CalculateConfigurationKeyHash_MustFailWithoutUrl()
{
sut.CalculateConfigurationKeyHash(default, default);
}
[TestMethod]
public void MustAllowForConcurrentKeyHashCalculation()
{
Parallel.For(0, 1000, (_) =>
{
sut.CalculateBrowserExamKeyHash(default, default, "https://www.safeexambrowser.org");
sut.CalculateConfigurationKeyHash(default, "https://www.safeexambrowser.org");
});
}
}
}

View file

@ -148,7 +148,6 @@
<Compile Include="Cryptography\PasswordEncryptionTests.cs" /> <Compile Include="Cryptography\PasswordEncryptionTests.cs" />
<Compile Include="Cryptography\PublicKeyEncryptionTests.cs" /> <Compile Include="Cryptography\PublicKeyEncryptionTests.cs" />
<Compile Include="Cryptography\PublicKeySymmetricEncryptionTests.cs" /> <Compile Include="Cryptography\PublicKeySymmetricEncryptionTests.cs" />
<Compile Include="Cryptography\KeyGeneratorTests.cs" />
<Compile Include="DataCompression\GZipCompressorTests.cs" /> <Compile Include="DataCompression\GZipCompressorTests.cs" />
<Compile Include="DataFormats\BinaryParserTests.cs" /> <Compile Include="DataFormats\BinaryParserTests.cs" />
<Compile Include="DataFormats\BinarySerializerTests.cs" /> <Compile Include="DataFormats\BinarySerializerTests.cs" />

View file

@ -31,7 +31,7 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
new UserInterfaceDataMapper() new UserInterfaceDataMapper()
}; };
internal void Map(IDictionary<string, object> rawData, AppSettings settings) internal void MapRawDataToSettings(IDictionary<string, object> rawData, AppSettings settings)
{ {
foreach (var item in rawData) foreach (var item in rawData)
{ {

View file

@ -575,12 +575,12 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
var useCustomForDesktop = rawData.TryGetValue(Keys.Browser.UserAgentModeDesktop, out var value) && value as int? != DEFAULT; var useCustomForDesktop = rawData.TryGetValue(Keys.Browser.UserAgentModeDesktop, out var value) && value as int? != DEFAULT;
var useCustomForMobile = rawData.TryGetValue(Keys.Browser.UserAgentModeMobile, out value) && value as int? != DEFAULT; var useCustomForMobile = rawData.TryGetValue(Keys.Browser.UserAgentModeMobile, out value) && value as int? != DEFAULT;
if (settings.UserInterface.Mode == UserInterfaceMode.Desktop && useCustomForDesktop) if (settings.UserInterfaceMode == UserInterfaceMode.Desktop && useCustomForDesktop)
{ {
settings.Browser.UseCustomUserAgent = true; settings.Browser.UseCustomUserAgent = true;
settings.Browser.CustomUserAgent = rawData[Keys.Browser.CustomUserAgentDesktop] as string; settings.Browser.CustomUserAgent = rawData[Keys.Browser.CustomUserAgentDesktop] as string;
} }
else if (settings.UserInterface.Mode == UserInterfaceMode.Mobile && useCustomForMobile) else if (settings.UserInterfaceMode == UserInterfaceMode.Mobile && useCustomForMobile)
{ {
settings.Browser.UseCustomUserAgent = true; settings.Browser.UseCustomUserAgent = true;
settings.Browser.CustomUserAgent = rawData[Keys.Browser.CustomUserAgentMobile] as string; settings.Browser.CustomUserAgent = rawData[Keys.Browser.CustomUserAgentMobile] as string;

View file

@ -21,6 +21,60 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
case Keys.Proctoring.ForceRaiseHandMessage: case Keys.Proctoring.ForceRaiseHandMessage:
MapForceRaiseHandMessage(settings, value); MapForceRaiseHandMessage(settings, value);
break; break;
case Keys.Proctoring.JitsiMeet.AllowChat:
MapJitsiMeetAllowChat(settings, value);
break;
case Keys.Proctoring.JitsiMeet.AllowClosedCaptions:
MapJitsiMeetAllowClosedCaptions(settings, value);
break;
case Keys.Proctoring.JitsiMeet.AllowRaiseHand:
MapJitsiMeetAllowRaiseHands(settings, value);
break;
case Keys.Proctoring.JitsiMeet.AllowRecording:
MapJitsiMeetAllowRecording(settings, value);
break;
case Keys.Proctoring.JitsiMeet.AllowTileView:
MapJitsiMeetAllowTileView(settings, value);
break;
case Keys.Proctoring.JitsiMeet.AudioMuted:
MapJitsiMeetAudioMuted(settings, value);
break;
case Keys.Proctoring.JitsiMeet.AudioOnly:
MapJitsiMeetAudioOnly(settings, value);
break;
case Keys.Proctoring.JitsiMeet.Enabled:
MapJitsiMeetEnabled(settings, value);
break;
case Keys.Proctoring.JitsiMeet.ReceiveAudio:
MapJitsiMeetReceiveAudio(settings, value);
break;
case Keys.Proctoring.JitsiMeet.ReceiveVideo:
MapJitsiMeetReceiveVideo(settings, value);
break;
case Keys.Proctoring.JitsiMeet.RoomName:
MapJitsiMeetRoomName(settings, value);
break;
case Keys.Proctoring.JitsiMeet.SendAudio:
MapJitsiMeetSendAudio(settings, value);
break;
case Keys.Proctoring.JitsiMeet.SendVideo:
MapJitsiMeetSendVideo(settings, value);
break;
case Keys.Proctoring.JitsiMeet.ServerUrl:
MapJitsiMeetServerUrl(settings, value);
break;
case Keys.Proctoring.JitsiMeet.ShowMeetingName:
MapJitsiMeetShowMeetingName(settings, value);
break;
case Keys.Proctoring.JitsiMeet.Subject:
MapJitsiMeetSubject(settings, value);
break;
case Keys.Proctoring.JitsiMeet.Token:
MapJitsiMeetToken(settings, value);
break;
case Keys.Proctoring.JitsiMeet.VideoMuted:
MapJitsiMeetVideoMuted(settings, value);
break;
case Keys.Proctoring.ScreenProctoring.ClientId: case Keys.Proctoring.ScreenProctoring.ClientId:
MapClientId(settings, value); MapClientId(settings, value);
break; break;
@ -66,6 +120,51 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
case Keys.Proctoring.ShowTaskbarNotification: case Keys.Proctoring.ShowTaskbarNotification:
MapShowTaskbarNotification(settings, value); MapShowTaskbarNotification(settings, value);
break; break;
case Keys.Proctoring.WindowVisibility:
MapWindowVisibility(settings, value);
break;
case Keys.Proctoring.Zoom.AllowChat:
MapZoomAllowChat(settings, value);
break;
case Keys.Proctoring.Zoom.AllowClosedCaptions:
MapZoomAllowClosedCaptions(settings, value);
break;
case Keys.Proctoring.Zoom.AllowRaiseHand:
MapZoomAllowRaiseHands(settings, value);
break;
case Keys.Proctoring.Zoom.AudioMuted:
MapZoomAudioMuted(settings, value);
break;
case Keys.Proctoring.Zoom.Enabled:
MapZoomEnabled(settings, value);
break;
case Keys.Proctoring.Zoom.MeetingNumber:
MapZoomMeetingNumber(settings, value);
break;
case Keys.Proctoring.Zoom.ReceiveAudio:
MapZoomReceiveAudio(settings, value);
break;
case Keys.Proctoring.Zoom.ReceiveVideo:
MapZoomReceiveVideo(settings, value);
break;
case Keys.Proctoring.Zoom.SendAudio:
MapZoomSendAudio(settings, value);
break;
case Keys.Proctoring.Zoom.SendVideo:
MapZoomSendVideo(settings, value);
break;
case Keys.Proctoring.Zoom.Signature:
MapZoomSignature(settings, value);
break;
case Keys.Proctoring.Zoom.Subject:
MapZoomSubject(settings, value);
break;
case Keys.Proctoring.Zoom.UserName:
MapZoomUserName(settings, value);
break;
case Keys.Proctoring.Zoom.VideoMuted:
MapZoomVideoMuted(settings, value);
break;
} }
} }
@ -77,6 +176,150 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
} }
} }
private void MapJitsiMeetAllowChat(AppSettings settings, object value)
{
if (value is bool allow)
{
settings.Proctoring.JitsiMeet.AllowChat = allow;
}
}
private void MapJitsiMeetAllowClosedCaptions(AppSettings settings, object value)
{
if (value is bool allow)
{
settings.Proctoring.JitsiMeet.AllowClosedCaptions = allow;
}
}
private void MapJitsiMeetAllowRaiseHands(AppSettings settings, object value)
{
if (value is bool allow)
{
settings.Proctoring.JitsiMeet.AllowRaiseHand = allow;
}
}
private void MapJitsiMeetAllowRecording(AppSettings settings, object value)
{
if (value is bool allow)
{
settings.Proctoring.JitsiMeet.AllowRecording = allow;
}
}
private void MapJitsiMeetAllowTileView(AppSettings settings, object value)
{
if (value is bool allow)
{
settings.Proctoring.JitsiMeet.AllowTileView = allow;
}
}
private void MapJitsiMeetAudioMuted(AppSettings settings, object value)
{
if (value is bool audioMuted)
{
settings.Proctoring.JitsiMeet.AudioMuted = audioMuted;
}
}
private void MapJitsiMeetAudioOnly(AppSettings settings, object value)
{
if (value is bool audioOnly)
{
settings.Proctoring.JitsiMeet.AudioOnly = audioOnly;
}
}
private void MapJitsiMeetEnabled(AppSettings settings, object value)
{
if (value is bool enabled)
{
settings.Proctoring.JitsiMeet.Enabled = enabled;
}
}
private void MapJitsiMeetReceiveAudio(AppSettings settings, object value)
{
if (value is bool receive)
{
settings.Proctoring.JitsiMeet.ReceiveAudio = receive;
}
}
private void MapJitsiMeetReceiveVideo(AppSettings settings, object value)
{
if (value is bool receive)
{
settings.Proctoring.JitsiMeet.ReceiveVideo = receive;
}
}
private void MapJitsiMeetRoomName(AppSettings settings, object value)
{
if (value is string name)
{
settings.Proctoring.JitsiMeet.RoomName = name;
}
}
private void MapJitsiMeetSendAudio(AppSettings settings, object value)
{
if (value is bool send)
{
settings.Proctoring.JitsiMeet.SendAudio = send;
}
}
private void MapJitsiMeetSendVideo(AppSettings settings, object value)
{
if (value is bool send)
{
settings.Proctoring.JitsiMeet.SendVideo = send;
}
}
private void MapJitsiMeetServerUrl(AppSettings settings, object value)
{
if (value is string url)
{
settings.Proctoring.JitsiMeet.ServerUrl = url;
}
}
private void MapJitsiMeetShowMeetingName(AppSettings settings, object value)
{
if (value is bool show)
{
settings.Proctoring.JitsiMeet.ShowMeetingName = show;
}
}
private void MapJitsiMeetSubject(AppSettings settings, object value)
{
if (value is string subject)
{
settings.Proctoring.JitsiMeet.Subject = subject;
}
}
private void MapJitsiMeetToken(AppSettings settings, object value)
{
if (value is string token)
{
settings.Proctoring.JitsiMeet.Token = token;
}
}
private void MapJitsiMeetVideoMuted(AppSettings settings, object value)
{
if (value is bool muted)
{
settings.Proctoring.JitsiMeet.VideoMuted = muted;
}
}
private void MapCaptureApplicationData(AppSettings settings, object value) private void MapCaptureApplicationData(AppSettings settings, object value)
{ {
if (value is bool capture) if (value is bool capture)
@ -220,5 +463,144 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
settings.Proctoring.ShowTaskbarNotification = show; settings.Proctoring.ShowTaskbarNotification = show;
} }
} }
private void MapWindowVisibility(AppSettings settings, object value)
{
const int HIDDEN = 0;
const int ALLOW_SHOW = 1;
const int ALLOW_HIDE = 2;
const int VISIBLE = 3;
if (value is int visibility)
{
switch (visibility)
{
case HIDDEN:
settings.Proctoring.WindowVisibility = WindowVisibility.Hidden;
break;
case ALLOW_SHOW:
settings.Proctoring.WindowVisibility = WindowVisibility.AllowToShow;
break;
case ALLOW_HIDE:
settings.Proctoring.WindowVisibility = WindowVisibility.AllowToHide;
break;
case VISIBLE:
settings.Proctoring.WindowVisibility = WindowVisibility.Visible;
break;
}
}
}
private void MapZoomAllowChat(AppSettings settings, object value)
{
if (value is bool allow)
{
settings.Proctoring.Zoom.AllowChat = allow;
}
}
private void MapZoomAllowClosedCaptions(AppSettings settings, object value)
{
if (value is bool allow)
{
settings.Proctoring.Zoom.AllowClosedCaptions = allow;
}
}
private void MapZoomAllowRaiseHands(AppSettings settings, object value)
{
if (value is bool allow)
{
settings.Proctoring.Zoom.AllowRaiseHand = allow;
}
}
private void MapZoomAudioMuted(AppSettings settings, object value)
{
if (value is bool muted)
{
settings.Proctoring.Zoom.AudioMuted = muted;
}
}
private void MapZoomEnabled(AppSettings settings, object value)
{
if (value is bool enabled)
{
settings.Proctoring.Zoom.Enabled = enabled;
}
}
private void MapZoomMeetingNumber(AppSettings settings, object value)
{
if (value is string number)
{
settings.Proctoring.Zoom.MeetingNumber = number;
}
}
private void MapZoomReceiveAudio(AppSettings settings, object value)
{
if (value is bool receive)
{
settings.Proctoring.Zoom.ReceiveAudio = receive;
}
}
private void MapZoomReceiveVideo(AppSettings settings, object value)
{
if (value is bool receive)
{
settings.Proctoring.Zoom.ReceiveVideo = receive;
}
}
private void MapZoomSendAudio(AppSettings settings, object value)
{
if (value is bool send)
{
settings.Proctoring.Zoom.SendAudio = send;
}
}
private void MapZoomSendVideo(AppSettings settings, object value)
{
if (value is bool send)
{
settings.Proctoring.Zoom.SendVideo = send;
}
}
private void MapZoomSignature(AppSettings settings, object value)
{
if (value is string signature)
{
settings.Proctoring.Zoom.Signature = signature;
}
}
private void MapZoomSubject(AppSettings settings, object value)
{
if (value is string subject)
{
settings.Proctoring.Zoom.Subject = subject;
}
}
private void MapZoomUserName(AppSettings settings, object value)
{
if (value is string name)
{
settings.Proctoring.Zoom.UserName = name;
}
}
private void MapZoomVideoMuted(AppSettings settings, object value)
{
if (value is bool muted)
{
settings.Proctoring.Zoom.VideoMuted = muted;
}
}
} }
} }

View file

@ -26,9 +26,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
case Keys.Security.AllowReconfiguration: case Keys.Security.AllowReconfiguration:
MapAllowReconfiguration(settings, value); MapAllowReconfiguration(settings, value);
break; break;
case Keys.Security.AllowStickyKeys:
MapAllowStickyKeys(settings, value);
break;
case Keys.Security.AllowTermination: case Keys.Security.AllowTermination:
MapAllowTermination(settings, value); MapAllowTermination(settings, value);
break; break;
@ -50,9 +47,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
case Keys.Security.VerifyCursorConfiguration: case Keys.Security.VerifyCursorConfiguration:
MapVerifyCursorConfiguration(settings, value); MapVerifyCursorConfiguration(settings, value);
break; break;
case Keys.Security.VerifySessionIntegrity:
MapVerifySessionIntegrity(settings, value);
break;
case Keys.Security.VersionRestrictions: case Keys.Security.VersionRestrictions:
MapVersionRestrictions(settings, value); MapVersionRestrictions(settings, value);
break; break;
@ -81,14 +75,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
} }
} }
private void MapAllowStickyKeys(AppSettings settings, object value)
{
if (value is bool allow)
{
settings.Security.AllowStickyKeys = allow;
}
}
private void MapAllowTermination(AppSettings settings, object value) private void MapAllowTermination(AppSettings settings, object value)
{ {
if (value is bool allow) if (value is bool allow)
@ -108,12 +94,12 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
if (settings.Security.AllowApplicationLogAccess) if (settings.Security.AllowApplicationLogAccess)
{ {
settings.UserInterface.ActionCenter.ShowApplicationLog = true; settings.ActionCenter.ShowApplicationLog = true;
} }
else else
{ {
settings.UserInterface.ActionCenter.ShowApplicationLog = false; settings.ActionCenter.ShowApplicationLog = false;
settings.UserInterface.Taskbar.ShowApplicationLog = false; settings.Taskbar.ShowApplicationLog = false;
} }
} }
@ -189,14 +175,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
} }
} }
private void MapVerifySessionIntegrity(AppSettings settings, object value)
{
if (value is bool verify)
{
settings.Security.VerifySessionIntegrity = verify;
}
}
private void MapVersionRestrictions(AppSettings settings, object value) private void MapVersionRestrictions(AppSettings settings, object value)
{ {
if (value is IList<object> restrictions) if (value is IList<object> restrictions)

View file

@ -20,9 +20,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
case Keys.UserInterface.ActionCenter.EnableActionCenter: case Keys.UserInterface.ActionCenter.EnableActionCenter:
MapEnableActionCenter(settings, value); MapEnableActionCenter(settings, value);
break; break;
case Keys.UserInterface.LockScreen.BackgroundColor:
MapLockScreenBackgroundColor(settings, value);
break;
case Keys.UserInterface.SystemControls.Audio.Show: case Keys.UserInterface.SystemControls.Audio.Show:
MapShowAudio(settings, value); MapShowAudio(settings, value);
break; break;
@ -57,15 +54,7 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
{ {
if (value is bool enable) if (value is bool enable)
{ {
settings.UserInterface.ActionCenter.EnableActionCenter = enable; settings.ActionCenter.EnableActionCenter = enable;
}
}
private void MapLockScreenBackgroundColor(AppSettings settings, object value)
{
if (value is string color)
{
settings.UserInterface.LockScreen.BackgroundColor = color;
} }
} }
@ -73,8 +62,8 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
{ {
if (value is bool show) if (value is bool show)
{ {
settings.UserInterface.ActionCenter.ShowAudio = show; settings.ActionCenter.ShowAudio = show;
settings.UserInterface.Taskbar.ShowAudio = show; settings.Taskbar.ShowAudio = show;
} }
} }
@ -82,8 +71,8 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
{ {
if (value is bool show) if (value is bool show)
{ {
settings.UserInterface.ActionCenter.ShowClock = show; settings.ActionCenter.ShowClock = show;
settings.UserInterface.Taskbar.ShowClock = show; settings.Taskbar.ShowClock = show;
} }
} }
@ -91,8 +80,8 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
{ {
if (value is bool show) if (value is bool show)
{ {
settings.UserInterface.ActionCenter.ShowKeyboardLayout = show; settings.ActionCenter.ShowKeyboardLayout = show;
settings.UserInterface.Taskbar.ShowKeyboardLayout = show; settings.Taskbar.ShowKeyboardLayout = show;
} }
} }
@ -100,8 +89,8 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
{ {
if (value is bool show) if (value is bool show)
{ {
settings.UserInterface.ActionCenter.ShowNetwork = show; settings.ActionCenter.ShowNetwork = show;
settings.UserInterface.Taskbar.ShowNetwork = show; settings.Taskbar.ShowNetwork = show;
} }
} }
@ -125,7 +114,7 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
{ {
if (value is bool enable) if (value is bool enable)
{ {
settings.UserInterface.Taskbar.EnableTaskbar = enable; settings.Taskbar.EnableTaskbar = enable;
} }
} }
@ -133,7 +122,7 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
{ {
if (value is bool show) if (value is bool show)
{ {
settings.UserInterface.Taskbar.ShowApplicationLog = show; settings.Taskbar.ShowApplicationLog = show;
} }
} }
@ -141,7 +130,7 @@ namespace SafeExamBrowser.Configuration.ConfigurationData.DataMapping
{ {
if (value is bool mobile) if (value is bool mobile)
{ {
settings.UserInterface.Mode = mobile ? UserInterfaceMode.Mobile : UserInterfaceMode.Desktop; settings.UserInterfaceMode = mobile ? UserInterfaceMode.Mobile : UserInterfaceMode.Desktop;
} }
} }
} }

View file

@ -12,6 +12,7 @@ using System.IO;
using System.Security.Cryptography; using System.Security.Cryptography;
using SafeExamBrowser.Settings; using SafeExamBrowser.Settings;
using SafeExamBrowser.Settings.Applications; using SafeExamBrowser.Settings.Applications;
using SafeExamBrowser.Settings.Proctoring;
using SafeExamBrowser.Settings.Security; using SafeExamBrowser.Settings.Security;
namespace SafeExamBrowser.Configuration.ConfigurationData namespace SafeExamBrowser.Configuration.ConfigurationData
@ -19,14 +20,9 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
internal class DataProcessor internal class DataProcessor
{ {
internal void Process(IDictionary<string, object> rawData, AppSettings settings) internal void Process(IDictionary<string, object> rawData, AppSettings settings)
{
ProcessDefault(settings);
CalculateConfigurationKey(rawData, settings);
}
internal void ProcessDefault(AppSettings settings)
{ {
AllowBrowserToolbarForReloading(settings); AllowBrowserToolbarForReloading(settings);
CalculateConfigurationKey(rawData, settings);
InitializeBrowserHomeFunctionality(settings); InitializeBrowserHomeFunctionality(settings);
InitializeClipboardSettings(settings); InitializeClipboardSettings(settings);
InitializeProctoringSettings(settings); InitializeProctoringSettings(settings);
@ -81,6 +77,15 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
private void InitializeProctoringSettings(AppSettings settings) private void InitializeProctoringSettings(AppSettings settings)
{ {
settings.Proctoring.Enabled = settings.Proctoring.ScreenProctoring.Enabled; settings.Proctoring.Enabled = settings.Proctoring.ScreenProctoring.Enabled;
// The video proctoring implementations are disabled for version 3.7.0 and will be completely removed for version 3.8.0.
settings.Proctoring.JitsiMeet.Enabled = false;
settings.Proctoring.Zoom.Enabled = false;
if (settings.Proctoring.JitsiMeet.Enabled && !settings.Proctoring.JitsiMeet.ReceiveVideo)
{
settings.Proctoring.WindowVisibility = WindowVisibility.Hidden;
}
} }
private void RemoveLegacyBrowsers(AppSettings settings) private void RemoveLegacyBrowsers(AppSettings settings)

View file

@ -100,6 +100,13 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
{ {
var settings = new AppSettings(); var settings = new AppSettings();
settings.ActionCenter.EnableActionCenter = true;
settings.ActionCenter.ShowApplicationInfo = true;
settings.ActionCenter.ShowApplicationLog = false;
settings.ActionCenter.ShowClock = true;
settings.ActionCenter.ShowKeyboardLayout = true;
settings.ActionCenter.ShowNetwork = false;
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "AA_v3.exe", OriginalName = "AA_v3.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "AA_v3.exe", OriginalName = "AA_v3.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "AeroAdmin.exe", OriginalName = "AeroAdmin.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "AeroAdmin.exe", OriginalName = "AeroAdmin.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "beamyourscreen-host.exe", OriginalName = "beamyourscreen-host.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "beamyourscreen-host.exe", OriginalName = "beamyourscreen-host.exe" });
@ -115,6 +122,7 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "Discord.exe", OriginalName = "Discord.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "Discord.exe", OriginalName = "Discord.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "Element.exe", OriginalName = "Element.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "Element.exe", OriginalName = "Element.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "g2mcomm.exe", OriginalName = "g2mcomm.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "g2mcomm.exe", OriginalName = "g2mcomm.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "g2mcomm.exe", OriginalName = "g2mcomm.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "g2mlauncher.exe", OriginalName = "g2mlauncher.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "g2mlauncher.exe", OriginalName = "g2mlauncher.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "g2mstart.exe", OriginalName = "g2mstart.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "g2mstart.exe", OriginalName = "g2mstart.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "GotoMeetingWinStore.exe", OriginalName = "GotoMeetingWinStore.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "GotoMeetingWinStore.exe", OriginalName = "GotoMeetingWinStore.exe" });
@ -132,7 +140,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "remoting_host.exe", OriginalName = "remoting_host.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "remoting_host.exe", OriginalName = "remoting_host.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "RPCService.exe", OriginalName = "RPCService.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "RPCService.exe", OriginalName = "RPCService.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "RPCSuite.exe", OriginalName = "RPCSuite.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "RPCSuite.exe", OriginalName = "RPCSuite.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "sethc.exe", OriginalName = "sethc.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "Skype.exe", OriginalName = "Skype.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "Skype.exe", OriginalName = "Skype.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "SkypeApp.exe", OriginalName = "SkypeApp.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "SkypeApp.exe", OriginalName = "SkypeApp.exe" });
settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "SkypeHost.exe", OriginalName = "SkypeHost.exe" }); settings.Applications.Blacklist.Add(new BlacklistApplication { ExecutableName = "SkypeHost.exe", OriginalName = "SkypeHost.exe" });
@ -165,13 +172,13 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
settings.Browser.AdditionalWindow.UrlPolicy = UrlPolicy.Never; settings.Browser.AdditionalWindow.UrlPolicy = UrlPolicy.Never;
settings.Browser.AllowConfigurationDownloads = true; settings.Browser.AllowConfigurationDownloads = true;
settings.Browser.AllowCustomDownAndUploadLocation = false; settings.Browser.AllowCustomDownAndUploadLocation = false;
settings.Browser.AllowDownloads = true; settings.Browser.AllowDownloads = false;
settings.Browser.AllowFind = true; settings.Browser.AllowFind = true;
settings.Browser.AllowPageZoom = true; settings.Browser.AllowPageZoom = true;
settings.Browser.AllowPdfReader = true; settings.Browser.AllowPdfReader = true;
settings.Browser.AllowPdfReaderToolbar = false; settings.Browser.AllowPdfReaderToolbar = false;
settings.Browser.AllowPrint = false; settings.Browser.AllowPrint = false;
settings.Browser.AllowUploads = true; settings.Browser.AllowUploads = false;
settings.Browser.DeleteCacheOnShutdown = true; settings.Browser.DeleteCacheOnShutdown = true;
settings.Browser.DeleteCookiesOnShutdown = true; settings.Browser.DeleteCookiesOnShutdown = true;
settings.Browser.DeleteCookiesOnStartup = true; settings.Browser.DeleteCookiesOnStartup = true;
@ -196,7 +203,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
settings.Browser.ShowFileSystemElementPath = true; settings.Browser.ShowFileSystemElementPath = true;
settings.Browser.StartUrl = "https://www.safeexambrowser.org/start"; settings.Browser.StartUrl = "https://www.safeexambrowser.org/start";
settings.Browser.UseCustomUserAgent = false; settings.Browser.UseCustomUserAgent = false;
settings.Browser.UseIsolatedClipboard = true;
settings.Browser.UseQueryParameter = false; settings.Browser.UseQueryParameter = false;
settings.Browser.UseTemporaryDownAndUploadDirectory = false; settings.Browser.UseTemporaryDownAndUploadDirectory = false;
@ -210,10 +216,7 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
settings.Keyboard.AllowAltEsc = false; settings.Keyboard.AllowAltEsc = false;
settings.Keyboard.AllowAltF4 = false; settings.Keyboard.AllowAltF4 = false;
settings.Keyboard.AllowAltTab = true; settings.Keyboard.AllowAltTab = true;
settings.Keyboard.AllowCtrlC = true;
settings.Keyboard.AllowCtrlEsc = false; settings.Keyboard.AllowCtrlEsc = false;
settings.Keyboard.AllowCtrlV = true;
settings.Keyboard.AllowCtrlX = true;
settings.Keyboard.AllowEsc = true; settings.Keyboard.AllowEsc = true;
settings.Keyboard.AllowF1 = true; settings.Keyboard.AllowF1 = true;
settings.Keyboard.AllowF2 = true; settings.Keyboard.AllowF2 = true;
@ -240,6 +243,20 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
settings.Proctoring.Enabled = false; settings.Proctoring.Enabled = false;
settings.Proctoring.ForceRaiseHandMessage = false; settings.Proctoring.ForceRaiseHandMessage = false;
settings.Proctoring.JitsiMeet.AllowChat = false;
settings.Proctoring.JitsiMeet.AllowClosedCaptions = false;
settings.Proctoring.JitsiMeet.AllowRaiseHand = false;
settings.Proctoring.JitsiMeet.AllowRecording = false;
settings.Proctoring.JitsiMeet.AllowTileView = false;
settings.Proctoring.JitsiMeet.AudioMuted = true;
settings.Proctoring.JitsiMeet.AudioOnly = false;
settings.Proctoring.JitsiMeet.Enabled = false;
settings.Proctoring.JitsiMeet.ReceiveAudio = false;
settings.Proctoring.JitsiMeet.ReceiveVideo = false;
settings.Proctoring.JitsiMeet.SendAudio = true;
settings.Proctoring.JitsiMeet.SendVideo = true;
settings.Proctoring.JitsiMeet.ShowMeetingName = false;
settings.Proctoring.JitsiMeet.VideoMuted = false;
settings.Proctoring.ScreenProctoring.Enabled = false; settings.Proctoring.ScreenProctoring.Enabled = false;
settings.Proctoring.ScreenProctoring.ImageDownscaling = 1.0; settings.Proctoring.ScreenProctoring.ImageDownscaling = 1.0;
settings.Proctoring.ScreenProctoring.ImageFormat = ImageFormat.Png; settings.Proctoring.ScreenProctoring.ImageFormat = ImageFormat.Png;
@ -251,16 +268,25 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
settings.Proctoring.ScreenProctoring.MinInterval = 1000; settings.Proctoring.ScreenProctoring.MinInterval = 1000;
settings.Proctoring.ShowRaiseHandNotification = true; settings.Proctoring.ShowRaiseHandNotification = true;
settings.Proctoring.ShowTaskbarNotification = true; settings.Proctoring.ShowTaskbarNotification = true;
settings.Proctoring.WindowVisibility = WindowVisibility.Hidden;
settings.Proctoring.Zoom.AllowChat = false;
settings.Proctoring.Zoom.AllowClosedCaptions = false;
settings.Proctoring.Zoom.AllowRaiseHand = false;
settings.Proctoring.Zoom.AudioMuted = true;
settings.Proctoring.Zoom.Enabled = false;
settings.Proctoring.Zoom.ReceiveAudio = false;
settings.Proctoring.Zoom.ReceiveVideo = false;
settings.Proctoring.Zoom.SendAudio = true;
settings.Proctoring.Zoom.SendVideo = true;
settings.Proctoring.Zoom.VideoMuted = false;
settings.Security.AllowApplicationLogAccess = false; settings.Security.AllowApplicationLogAccess = false;
settings.Security.AllowTermination = true; settings.Security.AllowTermination = true;
settings.Security.AllowReconfiguration = false; settings.Security.AllowReconfiguration = false;
settings.Security.AllowStickyKeys = false;
settings.Security.ClipboardPolicy = ClipboardPolicy.Isolated; settings.Security.ClipboardPolicy = ClipboardPolicy.Isolated;
settings.Security.DisableSessionChangeLockScreen = false; settings.Security.DisableSessionChangeLockScreen = false;
settings.Security.KioskMode = KioskMode.CreateNewDesktop; settings.Security.KioskMode = KioskMode.CreateNewDesktop;
settings.Security.VerifyCursorConfiguration = true; settings.Security.VerifyCursorConfiguration = true;
settings.Security.VerifySessionIntegrity = true;
settings.Security.VirtualMachinePolicy = VirtualMachinePolicy.Deny; settings.Security.VirtualMachinePolicy = VirtualMachinePolicy.Deny;
settings.Server.PingInterval = 1000; settings.Server.PingInterval = 1000;
@ -290,20 +316,14 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
settings.System.AlwaysOn = true; settings.System.AlwaysOn = true;
settings.UserInterface.ActionCenter.EnableActionCenter = true; settings.Taskbar.EnableTaskbar = true;
settings.UserInterface.ActionCenter.ShowApplicationInfo = true; settings.Taskbar.ShowApplicationInfo = false;
settings.UserInterface.ActionCenter.ShowApplicationLog = false; settings.Taskbar.ShowApplicationLog = false;
settings.UserInterface.ActionCenter.ShowClock = true; settings.Taskbar.ShowClock = true;
settings.UserInterface.ActionCenter.ShowKeyboardLayout = true; settings.Taskbar.ShowKeyboardLayout = true;
settings.UserInterface.ActionCenter.ShowNetwork = false; settings.Taskbar.ShowNetwork = false;
settings.UserInterface.LockScreen.BackgroundColor = "#ff0000";
settings.UserInterface.Mode = UserInterfaceMode.Desktop; settings.UserInterfaceMode = UserInterfaceMode.Desktop;
settings.UserInterface.Taskbar.EnableTaskbar = true;
settings.UserInterface.Taskbar.ShowApplicationInfo = false;
settings.UserInterface.Taskbar.ShowApplicationLog = false;
settings.UserInterface.Taskbar.ShowClock = true;
settings.UserInterface.Taskbar.ShowKeyboardLayout = true;
settings.UserInterface.Taskbar.ShowNetwork = false;
return settings; return settings;
} }

View file

@ -234,6 +234,29 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
internal const string ForceRaiseHandMessage = "raiseHandButtonAlwaysPromptMessage"; internal const string ForceRaiseHandMessage = "raiseHandButtonAlwaysPromptMessage";
internal const string ShowRaiseHand = "raiseHandButtonShow"; internal const string ShowRaiseHand = "raiseHandButtonShow";
internal const string ShowTaskbarNotification = "showProctoringViewButton"; internal const string ShowTaskbarNotification = "showProctoringViewButton";
internal const string WindowVisibility = "remoteProctoringViewShow";
internal static class JitsiMeet
{
internal const string AllowChat = "jitsiMeetFeatureFlagChat";
internal const string AllowClosedCaptions = "jitsiMeetFeatureFlagCloseCaptions";
internal const string AllowRaiseHand = "jitsiMeetFeatureFlagRaiseHand";
internal const string AllowRecording = "jitsiMeetFeatureFlagRecording";
internal const string AllowTileView = "jitsiMeetFeatureFlagTileView";
internal const string AudioMuted = "jitsiMeetAudioMuted";
internal const string AudioOnly = "jitsiMeetAudioOnly";
internal const string Enabled = "jitsiMeetEnable";
internal const string ReceiveAudio = "jitsiMeetReceiveAudio";
internal const string ReceiveVideo = "jitsiMeetReceiveVideo";
internal const string RoomName = "jitsiMeetRoom";
internal const string SendAudio = "jitsiMeetSendAudio";
internal const string SendVideo = "jitsiMeetSendVideo";
internal const string ServerUrl = "jitsiMeetServerURL";
internal const string ShowMeetingName = "jitsiMeetFeatureFlagDisplayMeetingName";
internal const string Subject = "jitsiMeetSubject";
internal const string Token = "jitsiMeetToken";
internal const string VideoMuted = "jitsiMeetVideoMuted";
}
internal static class ScreenProctoring internal static class ScreenProctoring
{ {
@ -255,6 +278,24 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
internal const string CaptureWindowTitle = "screenProctoringMetadataWindowTitleEnabled"; internal const string CaptureWindowTitle = "screenProctoringMetadataWindowTitleEnabled";
} }
} }
internal static class Zoom
{
internal const string AllowChat = "zoomFeatureFlagChat";
internal const string AllowClosedCaptions = "zoomFeatureFlagCloseCaptions";
internal const string AllowRaiseHand = "zoomFeatureFlagRaiseHand";
internal const string AudioMuted = "zoomAudioMuted";
internal const string Enabled = "zoomEnable";
internal const string MeetingNumber = "zoomRoom";
internal const string ReceiveAudio = "zoomReceiveAudio";
internal const string ReceiveVideo = "zoomReceiveVideo";
internal const string SendAudio = "zoomSendAudio";
internal const string SendVideo = "zoomSendVideo";
internal const string Signature = "zoomToken";
internal const string Subject = "zoomSubject";
internal const string UserName = "zoomUserInfoDisplayName";
internal const string VideoMuted = "zoomVideoMuted";
}
} }
internal static class Security internal static class Security
@ -262,7 +303,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
internal const string AdminPasswordHash = "hashedAdminPassword"; internal const string AdminPasswordHash = "hashedAdminPassword";
internal const string AllowApplicationLog = "allowApplicationLog"; internal const string AllowApplicationLog = "allowApplicationLog";
internal const string AllowReconfiguration = "examSessionReconfigureAllow"; internal const string AllowReconfiguration = "examSessionReconfigureAllow";
internal const string AllowStickyKeys = "allowStickyKeys";
internal const string AllowTermination = "allowQuit"; internal const string AllowTermination = "allowQuit";
internal const string AllowVirtualMachine = "allowVirtualMachine"; internal const string AllowVirtualMachine = "allowVirtualMachine";
internal const string ClipboardPolicy = "clipboardPolicy"; internal const string ClipboardPolicy = "clipboardPolicy";
@ -272,7 +312,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
internal const string QuitPasswordHash = "hashedQuitPassword"; internal const string QuitPasswordHash = "hashedQuitPassword";
internal const string ReconfigurationUrl = "examSessionReconfigureConfigURL"; internal const string ReconfigurationUrl = "examSessionReconfigureConfigURL";
internal const string VerifyCursorConfiguration = "enableCursorVerification"; internal const string VerifyCursorConfiguration = "enableCursorVerification";
internal const string VerifySessionIntegrity = "enableSessionVerification";
internal const string VersionRestrictions = "sebAllowedVersions"; internal const string VersionRestrictions = "sebAllowedVersions";
} }
@ -327,11 +366,6 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
internal const string EnableActionCenter = "showSideMenu"; internal const string EnableActionCenter = "showSideMenu";
} }
internal static class LockScreen
{
internal const string BackgroundColor = "lockScreenBackgroundColor";
}
internal static class SystemControls internal static class SystemControls
{ {
internal static class Audio internal static class Audio

View file

@ -122,8 +122,6 @@ namespace SafeExamBrowser.Configuration
var status = default(LoadStatus); var status = default(LoadStatus);
settings = LoadDefaultSettings(); settings = LoadDefaultSettings();
dataProcessor.ProcessDefault(settings);
logger.Info($"Initialized default settings, now attempting to load '{resource}'..."); logger.Info($"Initialized default settings, now attempting to load '{resource}'...");
try try
@ -138,7 +136,7 @@ namespace SafeExamBrowser.Configuration
if (status == LoadStatus.Success) if (status == LoadStatus.Success)
{ {
dataMapper.Map(data, settings); dataMapper.MapRawDataToSettings(data, settings);
dataProcessor.Process(data, settings); dataProcessor.Process(data, settings);
} }
} }

View file

@ -9,7 +9,6 @@
using System; using System;
using System.Security.Cryptography; using System.Security.Cryptography;
using System.Text; using System.Text;
using System.Threading;
using SafeExamBrowser.Configuration.Contracts; using SafeExamBrowser.Configuration.Contracts;
using SafeExamBrowser.Configuration.Contracts.Cryptography; using SafeExamBrowser.Configuration.Contracts.Cryptography;
using SafeExamBrowser.Configuration.Contracts.Integrity; using SafeExamBrowser.Configuration.Contracts.Integrity;
@ -21,7 +20,7 @@ namespace SafeExamBrowser.Configuration.Cryptography
{ {
private readonly object @lock = new object(); private readonly object @lock = new object();
private readonly ThreadLocal<SHA256Managed> algorithm; private readonly SHA256Managed algorithm;
private readonly AppConfig appConfig; private readonly AppConfig appConfig;
private readonly IIntegrityModule integrityModule; private readonly IIntegrityModule integrityModule;
private readonly ILogger logger; private readonly ILogger logger;
@ -30,7 +29,7 @@ namespace SafeExamBrowser.Configuration.Cryptography
public KeyGenerator(AppConfig appConfig, IIntegrityModule integrityModule, ILogger logger) public KeyGenerator(AppConfig appConfig, IIntegrityModule integrityModule, ILogger logger)
{ {
this.algorithm = new ThreadLocal<SHA256Managed>(() => new SHA256Managed()); this.algorithm = new SHA256Managed();
this.appConfig = appConfig; this.appConfig = appConfig;
this.integrityModule = integrityModule; this.integrityModule = integrityModule;
this.logger = logger; this.logger = logger;
@ -53,7 +52,7 @@ namespace SafeExamBrowser.Configuration.Cryptography
public string CalculateBrowserExamKeyHash(string configurationKey, byte[] salt, string url) public string CalculateBrowserExamKeyHash(string configurationKey, byte[] salt, string url)
{ {
var urlWithoutFragment = url.Split('#')[0]; var urlWithoutFragment = url.Split('#')[0];
var hash = algorithm.Value.ComputeHash(Encoding.UTF8.GetBytes(urlWithoutFragment + (browserExamKey ?? ComputeBrowserExamKey(configurationKey, salt)))); var hash = algorithm.ComputeHash(Encoding.UTF8.GetBytes(urlWithoutFragment + (browserExamKey ?? ComputeBrowserExamKey(configurationKey, salt))));
var key = ToString(hash); var key = ToString(hash);
return key; return key;
@ -62,7 +61,7 @@ namespace SafeExamBrowser.Configuration.Cryptography
public string CalculateConfigurationKeyHash(string configurationKey, string url) public string CalculateConfigurationKeyHash(string configurationKey, string url)
{ {
var urlWithoutFragment = url.Split('#')[0]; var urlWithoutFragment = url.Split('#')[0];
var hash = algorithm.Value.ComputeHash(Encoding.UTF8.GetBytes(urlWithoutFragment + configurationKey)); var hash = algorithm.ComputeHash(Encoding.UTF8.GetBytes(urlWithoutFragment + configurationKey));
var key = ToString(hash); var key = ToString(hash);
return key; return key;
@ -112,8 +111,6 @@ namespace SafeExamBrowser.Configuration.Cryptography
browserExamKey = key; browserExamKey = key;
} }
logger.Debug("Successfully calculated browser exam key using simplified calculation.");
} }
} }
} }

View file

@ -149,7 +149,7 @@ namespace SafeExamBrowser.Configuration.DataResources
{ {
data = default; data = default;
logger.Debug($"The {(IsUnauthorized(response) ? "resource needs authentication" : "response data is HTML")}."); logger.Debug($"The {(IsUnauthorized(response) ? "resource needs authentication" : " response data is HTML")}.");
return LoadStatus.LoadWithBrowser; return LoadStatus.LoadWithBrowser;
} }

View file

@ -47,11 +47,6 @@ namespace SafeExamBrowser.I18n.Contracts
BrowserWindow_ZoomMenuMinus, BrowserWindow_ZoomMenuMinus,
BrowserWindow_ZoomMenuPlus, BrowserWindow_ZoomMenuPlus,
Build, Build,
CredentialsDialog_PasswordLabel,
CredentialsDialog_UsernameLabel,
CredentialsDialog_UsernameOptionalLabel,
CredentialsDialog_WirelessNetworkMessage,
CredentialsDialog_WirelessNetworkTitle,
ExamSelectionDialog_Cancel, ExamSelectionDialog_Cancel,
ExamSelectionDialog_Message, ExamSelectionDialog_Message,
ExamSelectionDialog_Select, ExamSelectionDialog_Select,
@ -83,9 +78,6 @@ namespace SafeExamBrowser.I18n.Contracts
LockScreen_EaseOfAccessMessage, LockScreen_EaseOfAccessMessage,
LockScreen_EaseOfAccessTerminateOption, LockScreen_EaseOfAccessTerminateOption,
LockScreen_SessionIntegrityMessage, LockScreen_SessionIntegrityMessage,
LockScreen_StickyKeysMessage,
LockScreen_StickyKeysContinueOption,
LockScreen_StickyKeysTerminateOption,
LockScreen_Title, LockScreen_Title,
LockScreen_UnlockButton, LockScreen_UnlockButton,
LockScreen_UserSessionContinueOption, LockScreen_UserSessionContinueOption,

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
Build Build
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
Passwort:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
Benutzername:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
Benutzername (falls erforderlich):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
Bitte geben Sie die erforderlichen Anmeldeinformationen ein, um eine Verbindung mit dem drahtlosen Netzwerk "%%_NAME_%%" herzustellen.
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
Netzwerk-Authentifizierung erforderlich
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
Abbrechen Abbrechen
</Entry> </Entry>
@ -207,15 +192,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
Die letzte Sitzung mit der momentan aktiven Konfiguration oder Start-URL wurde nicht ordnungsgemäss beendet! Geben Sie bitte das korrekte Passwort ein, um SEB zu entsperren. Die letzte Sitzung mit der momentan aktiven Konfiguration oder Start-URL wurde nicht ordnungsgemäss beendet! Geben Sie bitte das korrekte Passwort ein, um SEB zu entsperren.
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
Zustand der Sticky-Keys temporär erlauben. Dies gilt nur für die momentan laufende Sitzung!
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
Ein verbotener Zustand der Sticky-Keys wurde detektiert. Bitte wählen Sie eine der verfügbaren Optionen aus und geben Sie das korrekte Passwort ein, um SEB zu entsperren.
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
Safe Exam Browser beenden. WARNUNG: Sie werden keine Möglichkeit haben, Daten zu speichern oder weitere Aktionen auszuführen, SEB wird sofort beendet!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
SEB GESPERRT SEB GESPERRT
</Entry> </Entry>

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
Build Build
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
Password:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
Username:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
Username (if required):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
Please enter the required credentials to establish a connection to the wireless network "%%_NAME_%%".
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
Network Authentication Required
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
Cancel Cancel
</Entry> </Entry>
@ -207,15 +192,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
The last session with the currently active configuration or start URL was not terminated properly! Please enter the correct password to unlock SEB. The last session with the currently active configuration or start URL was not terminated properly! Please enter the correct password to unlock SEB.
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
Temporarily allow the state of the sticky keys. This applies only to the currently running session!
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
A prohibited state of the sticky keys has been detected. In order to unlock SEB, please select one of the available options and enter the correct unlock password.
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
Terminate Safe Exam Browser. WARNING: There will be no possibility to save data or perform any further actions, the shutdown will be initiated immediately!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
SEB LOCKED SEB LOCKED
</Entry> </Entry>

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
Build Build
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
Contraseña:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
Nombre de usuario:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
Nombre de usuario (si se requiere):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
Por favor, introduzca las credenciales necesarias para establecer una conexión con la red inalámbrica "%%_NAME_%%".
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
Autenticación de red necesaria
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
Cancel Cancel
</Entry> </Entry>
@ -207,15 +192,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
La última sesión con la configuración activa o la URL de inicio no se ha cerrado correctamente. Por favor, introduzca la contraseña correcta para desbloquear SEB. La última sesión con la configuración activa o la URL de inicio no se ha cerrado correctamente. Por favor, introduzca la contraseña correcta para desbloquear SEB.
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
Permitir temporalmente el estado de las teclas adhesivas. Esto sólo se aplica a la sesión que se está ejecutando en ese momento.
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
Se ha detectado un estado prohibido de las teclas adhesivas. Para desbloquear SEB, seleccione una de las opciones disponibles e introduzca la contraseña de desbloqueo correcta.
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
Finalizar Safe Exam Browser. ADVERTENCIA: No habrá posibilidad de guardar datos o realizar ninguna otra acción, ¡el cierre se iniciará inmediatamente!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
SEB BLOQUEADO SEB BLOQUEADO
</Entry> </Entry>

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
Koosta Koosta
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
Parool:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
Kasutajanimi:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
Kasutajanimi (kui see on nõutav):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
Palun sisestage vajalikud volitused, et luua ühendus traadita võrguga "%%_NAME_%%".
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
Võrguautentimine nõutav
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
Tühista Tühista
</Entry> </Entry>
@ -207,15 +192,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
Viimast seanssi hetkel aktiivse seadistuse või algus-URL-iga ei suletud korrektselt! Palun sisestage SEB'i avamiseks õige parool. Viimast seanssi hetkel aktiivse seadistuse või algus-URL-iga ei suletud korrektselt! Palun sisestage SEB'i avamiseks õige parool.
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
Lubab ajutiselt kleepuvate võtmete oleku. See kehtib ainult hetkel käimasoleva seansi kohta!
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
Avastati kleepuvate võtmete keelatud olek. SEB avamiseks valige üks olemasolevatest võimalustest ja sisestage õige avamisparool.
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
Sulge Safe Exam Browser. HOIATUS: andmete salvestamise ega edasiste toimingute tegemise võimalust ei ole, sulgemine käivitatakse kohe!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
SEB ON LUKUSTATUD SEB ON LUKUSTATUD
</Entry> </Entry>

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
Build Build
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
Mot de passe:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
Nom d'utilisateur:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
Nom d'utilisateur (si requis):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
Veuillez saisir les informations d'identification requises pour établir une connexion au réseau sans fil "%%_NAME_%%".
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
Authentification réseau requise
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
Annuler Annuler
</Entry> </Entry>
@ -207,15 +192,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
La dernière session avec la configuration ou l'URL de démarrage actuellement active n'a pas été terminée correctement! Afin de déverrouiller SEB, veuillez entrer le mot de passe de déverrouillage correct. La dernière session avec la configuration ou l'URL de démarrage actuellement active n'a pas été terminée correctement! Afin de déverrouiller SEB, veuillez entrer le mot de passe de déverrouillage correct.
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
Autoriser temporairement l'état des clés autocollantes. Cela ne s'applique qu'à la session en cours!
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
Un état d'interdiction des clés autocollantes a été détecté. Pour déverrouiller SEB, veuillez sélectionner l'une des options disponibles et saisir le mot de passe de déverrouillage correct.
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
Terminate Safe Exam Browser. WARNING: There will be no possibility to save data or perform any further actions, the shutdown will be initiated immediately!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
SEB VEROUILLE SEB VEROUILLE
</Entry> </Entry>

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
Bangun Bangun
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
Kata sandi:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
Nama pengguna:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
Nama pengguna (jika diperlukan):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
Masukkan kredensial yang diperlukan untuk membuat sambungan ke jaringan nirkabel "%%_NAMA_%%".
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
Diperlukan Otentikasi Jaringan
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
Batal Batal
</Entry> </Entry>
@ -207,15 +192,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
Sesi terakhir dengan konfigurasi yang sedang aktif atau URL awal tidak diakhiri dengan benar! Silakan masukkan kata sandi yang benar untuk membuka kunci SEB. Sesi terakhir dengan konfigurasi yang sedang aktif atau URL awal tidak diakhiri dengan benar! Silakan masukkan kata sandi yang benar untuk membuka kunci SEB.
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
Mengizinkan status tombol tempel untuk sementara waktu. Ini hanya berlaku untuk sesi yang sedang berjalan!
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
Kondisi kunci tempel yang dilarang telah terdeteksi. Untuk membuka kunci SEB, pilih salah satu opsi yang tersedia dan masukkan kata sandi buka kunci yang benar.
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
Menghentikan Safe Exam Browser. PERINGATAN: Tidak akan ada kemungkinan untuk menyimpan data atau melakukan tindakan lebih lanjut, penghentian akan segera dilakukan!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
SEB TERKUNCI SEB TERKUNCI
</Entry> </Entry>

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
Build Build
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
Password:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
Nome utente:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
Nome utente (se richiesto):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
Inserire le credenziali richieste per stabilire una connessione alla rete wireless "%%_NAME_%%".
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
È richiesta l'autenticazione di rete
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
Annulla Annulla
</Entry> </Entry>
@ -207,15 +192,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
L'ultima sessione con la configurazione o l'URL di avvio attualmente attiva non è stata terminata correttamente! Per sbloccare SEB, inserisci la password di sblocco corretta. L'ultima sessione con la configurazione o l'URL di avvio attualmente attiva non è stata terminata correttamente! Per sbloccare SEB, inserisci la password di sblocco corretta.
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
Consente temporaneamente lo stato dei tasti adesivi. Questo vale solo per la sessione in corso!
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
È stato rilevato uno stato non consentito dei tasti adesivi. Per sbloccare SEB, selezionare una delle opzioni disponibili e inserire la password di sblocco corretta.
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
Terminare Safe Exam Browser. ATTENZIONE: Non sarà possibile salvare i dati o eseguire ulteriori azioni, la chiusura verrà avviata immediatamente!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
SEB BLOCCATO SEB BLOCCATO
</Entry> </Entry>

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
Build Build
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
Wachtwoord:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
Gebruikersnaam:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
Gebruikersnaam (indien vereist):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
Voer de vereiste gegevens in om verbinding te maken met het draadloze netwerk “%%_NAME_%%”.
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
Netwerkverificatie vereist
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
Annuleren Annuleren
</Entry> </Entry>
@ -207,15 +192,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
De laatste sessie met de huidige actieve configuratie of start URL is niet goed afgesloten! Voer het juiste wachtwoord in om SEB te ontgrendelen. De laatste sessie met de huidige actieve configuratie of start URL is niet goed afgesloten! Voer het juiste wachtwoord in om SEB te ontgrendelen.
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
Tijdelijk de toestand van de sticky keys toestaan. Dit geldt alleen voor de huidige sessie!
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
Er is een verboden toestand van de sticky keys gedetecteerd. Om SEB te ontgrendelen, selecteert u een van de beschikbare opties en voert u het juiste ontgrendelwachtwoord in.
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
Beëindig de Safe Exam Browser. WAARSCHUWING: er is geen mogelijkheid om gegevens op te slaan of verdere acties uit te voeren, de browser wordt onmiddellijk afgesloten!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
SEB VERGRENDELD SEB VERGRENDELD
</Entry> </Entry>

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
Строить Строить
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
Пароль:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
Имя пользователя:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
Имя пользователя (если требуется):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
Введите необходимые учетные данные для установления соединения с беспроводной сетью "%%_NAME_%%".
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
Требуется сетевая аутентификация
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
Отмена Отмена
</Entry> </Entry>
@ -206,15 +191,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
Последний сеанс с текущей активной конфигурацией или URL-адресом запуска не был завершен должным образом! Пожалуйста, введите правильный пароль, чтобы разблокировать SEB. Последний сеанс с текущей активной конфигурацией или URL-адресом запуска не был завершен должным образом! Пожалуйста, введите правильный пароль, чтобы разблокировать SEB.
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
Временно разрешить состояние липких ключей. Это относится только к текущему сеансу!
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
Обнаружено запрещенное состояние липких клавиш. Чтобы разблокировать SEB, пожалуйста, выберите один из доступных вариантов и введите правильный пароль разблокировки.
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
Завершить работу Safe Exam Browser. ВНИМАНИЕ: Сохранить данные или выполнить какие-либо дальнейшие действия будет невозможно, завершение работы будет инициировано немедленно!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
SEB ЗАБЛОКИРОВАН SEB ЗАБЛОКИРОВАН
</Entry> </Entry>

View file

@ -99,21 +99,6 @@
<Entry key="Build"> <Entry key="Build">
生成 生成
</Entry> </Entry>
<Entry key="CredentialsDialog_PasswordLabel">
密码:
</Entry>
<Entry key="CredentialsDialog_UsernameLabel">
用户名:
</Entry>
<Entry key="CredentialsDialog_UsernameOptionalLabel">
用户名(如需要):
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkMessage">
请输入与无线网络 "%%_NAME_%%" 建立连接所需的凭证。
</Entry>
<Entry key="CredentialsDialog_WirelessNetworkTitle">
需要网络验证
</Entry>
<Entry key="ExamSelectionDialog_Cancel"> <Entry key="ExamSelectionDialog_Cancel">
取消 取消
</Entry> </Entry>
@ -193,7 +178,7 @@
检测到禁止的显示配置。 要解锁 SEB请选择可用选项之一并输入正确的解锁密码。 检测到禁止的显示配置。 要解锁 SEB请选择可用选项之一并输入正确的解锁密码。
</Entry> </Entry>
<Entry key="LockScreen_DisplayConfigurationTerminateOption"> <Entry key="LockScreen_DisplayConfigurationTerminateOption">
终止安全考试浏览器。警告:将无法保存数据或执行任何进一步操作,将立即启动关机! 终止安全考试浏览器。 警告:将无法保存数据或执行任何进一步操作,将立即启动关机!
</Entry> </Entry>
<Entry key="LockScreen_EaseOfAccessContinueOption"> <Entry key="LockScreen_EaseOfAccessContinueOption">
暂时允许轻松访问的配置。这只适用于当前运行的会话! 暂时允许轻松访问的配置。这只适用于当前运行的会话!
@ -207,15 +192,6 @@
<Entry key="LockScreen_SessionIntegrityMessage"> <Entry key="LockScreen_SessionIntegrityMessage">
当前激活的配置或启动URL的最后一个会话没有被正确终止! 请输入正确的密码以解锁SEB。 当前激活的配置或启动URL的最后一个会话没有被正确终止! 请输入正确的密码以解锁SEB。
</Entry> </Entry>
<Entry key="LockScreen_StickyKeysContinueOption">
暂时允许粘性按键的状态。这仅适用于当前运行的会话!
</Entry>
<Entry key="LockScreen_StickyKeysMessage">
已检测到粘滞键处于禁用状态。要解锁 SEB请选择一个可用选项并输入正确的解锁密码。
</Entry>
<Entry key="LockScreen_StickyKeysTerminateOption">
终止安全考试浏览器。警告:将没有可能保存数据或执行任何进一步的行动,关闭将立即启动!
</Entry>
<Entry key="LockScreen_Title"> <Entry key="LockScreen_Title">
防作弊考试专用浏览器已锁定 防作弊考试专用浏览器已锁定
</Entry> </Entry>

View file

@ -65,17 +65,13 @@
<Compile Include="Display\IDisplayMonitor.cs" /> <Compile Include="Display\IDisplayMonitor.cs" />
<Compile Include="Display\ValidationResult.cs" /> <Compile Include="Display\ValidationResult.cs" />
<Compile Include="IClipboard.cs" /> <Compile Include="IClipboard.cs" />
<Compile Include="IRemoteSessionDetector.cs" />
<Compile Include="IVirtualMachineDetector.cs" />
<Compile Include="Keyboard\IKeyboardInterceptor.cs" /> <Compile Include="Keyboard\IKeyboardInterceptor.cs" />
<Compile Include="Mouse\IMouseInterceptor.cs" /> <Compile Include="Mouse\IMouseInterceptor.cs" />
<Compile Include="Applications\InitializationResult.cs" /> <Compile Include="Applications\InitializationResult.cs" />
<Compile Include="Applications\IApplicationMonitor.cs" /> <Compile Include="Applications\IApplicationMonitor.cs" />
<Compile Include="Properties\AssemblyInfo.cs" /> <Compile Include="Properties\AssemblyInfo.cs" />
<Compile Include="System\Events\SentinelEventArgs.cs" />
<Compile Include="System\Events\SentinelEventHandler.cs" />
<Compile Include="System\Events\SessionChangedEventHandler.cs" /> <Compile Include="System\Events\SessionChangedEventHandler.cs" />
<Compile Include="System\ISystemSentinel.cs" /> <Compile Include="System\ISystemMonitor.cs" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<ProjectReference Include="..\SafeExamBrowser.Settings\SafeExamBrowser.Settings.csproj"> <ProjectReference Include="..\SafeExamBrowser.Settings\SafeExamBrowser.Settings.csproj">

View file

@ -1,21 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.Monitoring.Contracts.System.Events
{
/// <summary>
/// The event arguments used for the <see cref="SentinelEventHandler"/> fired by the <see cref="ISystemSentinel"/>.
/// </summary>
public class SentinelEventArgs
{
/// <summary>
/// Indicates whether a configuration or state change shall be allowed.
/// </summary>
public bool Allow { get; set; }
}
}

View file

@ -1,15 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.Monitoring.Contracts.System.Events
{
/// <summary>
/// The default handler for events of the <see cref="ISystemSentinel"/>. Normally indicates that a configuration or state has changed.
/// </summary>
public delegate void SentinelEventHandler(SentinelEventArgs args);
}

View file

@ -0,0 +1,33 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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 SafeExamBrowser.Monitoring.Contracts.System.Events;
namespace SafeExamBrowser.Monitoring.Contracts.System
{
/// <summary>
/// Monitors the operating system, e.g. for user account related events.
/// </summary>
public interface ISystemMonitor
{
/// <summary>
/// Event fired when the active user session has changed.
/// </summary>
event SessionChangedEventHandler SessionChanged;
/// <summary>
/// Starts the monitoring.
/// </summary>
void Start();
/// <summary>
/// Stops the monitoring.
/// </summary>
void Stop();
}
}

View file

@ -1,88 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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 SafeExamBrowser.Monitoring.Contracts.System.Events;
namespace SafeExamBrowser.Monitoring.Contracts.System
{
/// <summary>
/// Provides the possibility to suppress and surveil various system components and functionalities.
/// </summary>
public interface ISystemSentinel
{
/// <summary>
/// Event fired when the cursor configuration has changed.
/// </summary>
event SentinelEventHandler CursorChanged;
/// <summary>
/// Event fired when the ease of access configuration has changed.
/// </summary>
event SentinelEventHandler EaseOfAccessChanged;
/// <summary>
/// Event fired when the active user session has changed.
/// </summary>
event SessionChangedEventHandler SessionChanged;
/// <summary>
/// Event fired when the sticky keys state has changed.
/// </summary>
event SentinelEventHandler StickyKeysChanged;
/// <summary>
/// Attempts to disable the sticky keys. Returns <c>true</c> if successful, otherwise <c>false</c>.
/// </summary>
bool DisableStickyKeys();
/// <summary>
/// Attempts to enable the sticky keys. Returns <c>true</c> if successful, otherwise <c>false</c>.
/// </summary>
bool EnableStickyKeys();
/// <summary>
/// Attempts to revert the sticky keys to their initial state. Returns <c>true</c> if successful, otherwise <c>false</c>.
/// </summary>
bool RevertStickyKeys();
/// <summary>
/// Starts monitoring the cursor configuration.
/// </summary>
void StartMonitoringCursors();
/// <summary>
/// Starts monitoring the ease of access configuration.
/// </summary>
void StartMonitoringEaseOfAccess();
/// <summary>
/// Starts monitoring the sticky keys state.
/// </summary>
void StartMonitoringStickyKeys();
/// <summary>
/// Starts monitoring the system events.
/// </summary>
void StartMonitoringSystemEvents();
/// <summary>
/// Stops all monitoring operations.
/// </summary>
void StopMonitoring();
/// <summary>
/// Verifies the cursor configuration. Returns <c>true</c> if permitted, otherwise <c>false</c>.
/// </summary>
bool VerifyCursors();
/// <summary>
/// Verifies the ease of access configuration. Returns <c>true</c> if permitted, otherwise <c>false</c>.
/// </summary>
bool VerifyEaseOfAccess();
}
}

View file

@ -98,13 +98,7 @@ namespace SafeExamBrowser.Monitoring.Applications
if (activeWindow != default && TryGetProcessFor(activeWindow, out var process)) if (activeWindow != default && TryGetProcessFor(activeWindow, out var process))
{ {
var window = new Window application = new ActiveApplication(process, new Window { Handle = activeWindow.Handle, Title = activeWindow.Title });
{
Handle = activeWindow.Handle,
Title = nativeMethods.GetWindowTitle(activeWindow.Handle)
};
application = new ActiveApplication(process, window);
} }
return application != default; return application != default;
@ -248,6 +242,7 @@ namespace SafeExamBrowser.Monitoring.Applications
{ {
var isClient = true; var isClient = true;
var isRuntime = true; var isRuntime = true;
var isWebView = true;
isClient &= process.Name == "SafeExamBrowser.Client.exe"; isClient &= process.Name == "SafeExamBrowser.Client.exe";
isClient &= process.OriginalName == "SafeExamBrowser.Client.exe"; isClient &= process.OriginalName == "SafeExamBrowser.Client.exe";
@ -255,12 +250,16 @@ namespace SafeExamBrowser.Monitoring.Applications
isRuntime &= process.Name == "SafeExamBrowser.exe"; isRuntime &= process.Name == "SafeExamBrowser.exe";
isRuntime &= process.OriginalName == "SafeExamBrowser.exe"; isRuntime &= process.OriginalName == "SafeExamBrowser.exe";
isWebView &= process.Name == "msedgewebview2.exe";
isWebView &= process.OriginalName == "msedgewebview2.exe";
#if !DEBUG #if !DEBUG
isClient &= process.Signature == "2bc82fe8e56a39f96bc6c4b91d6703a0379b76a2"; isClient &= process.Signature == "2bc82fe8e56a39f96bc6c4b91d6703a0379b76a2";
isRuntime &= process.Signature == "2bc82fe8e56a39f96bc6c4b91d6703a0379b76a2"; isRuntime &= process.Signature == "2bc82fe8e56a39f96bc6c4b91d6703a0379b76a2";
isWebView &= process.Signature == "a4baabd12432ab9c7c297385260e95c3dae83bf2";
#endif #endif
return isClient || isRuntime; return isClient || isRuntime || isWebView;
} }
private void Close(Window window) private void Close(Window window)

View file

@ -67,13 +67,7 @@
<Compile Include="Mouse\MouseInterceptor.cs" /> <Compile Include="Mouse\MouseInterceptor.cs" />
<Compile Include="Applications\ApplicationMonitor.cs" /> <Compile Include="Applications\ApplicationMonitor.cs" />
<Compile Include="Properties\AssemblyInfo.cs" /> <Compile Include="Properties\AssemblyInfo.cs" />
<Compile Include="RemoteSessionDetector.cs" /> <Compile Include="System\SystemMonitor.cs" />
<Compile Include="System\Components\Cursors.cs" />
<Compile Include="System\Components\EaseOfAccess.cs" />
<Compile Include="System\Components\StickyKeys.cs" />
<Compile Include="System\Components\SystemEvents.cs" />
<Compile Include="System\SystemSentinel.cs" />
<Compile Include="VirtualMachineDetector.cs" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<ProjectReference Include="..\SafeExamBrowser.Logging.Contracts\SafeExamBrowser.Logging.Contracts.csproj"> <ProjectReference Include="..\SafeExamBrowser.Logging.Contracts\SafeExamBrowser.Logging.Contracts.csproj">

View file

@ -1,152 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.Linq;
using System.Threading.Tasks;
using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.System.Events;
using SafeExamBrowser.SystemComponents.Contracts.Registry;
namespace SafeExamBrowser.Monitoring.System.Components
{
internal class Cursors
{
private static readonly string SYSTEM_PATH = $@"{Environment.ExpandEnvironmentVariables("%SystemRoot%")}\Cursors\";
private static readonly string USER_PATH = $@"{Environment.ExpandEnvironmentVariables("%LocalAppData%")}\Microsoft\Windows\Cursors\";
private readonly ILogger logger;
private readonly IRegistry registry;
internal event SentinelEventHandler CursorChanged;
internal Cursors(ILogger logger, IRegistry registry)
{
this.logger = logger;
this.registry = registry;
}
internal void StartMonitoring()
{
registry.ValueChanged += Registry_ValueChanged;
if (registry.TryGetNames(RegistryValue.UserHive.Cursors_Key, out var names))
{
foreach (var name in names)
{
registry.StartMonitoring(RegistryValue.UserHive.Cursors_Key, name);
}
logger.Info("Started monitoring cursors.");
}
else
{
logger.Warn("Failed to start monitoring cursor registry values!");
}
}
internal void StopMonitoring()
{
registry.ValueChanged -= Registry_ValueChanged;
if (registry.TryGetNames(RegistryValue.UserHive.Cursors_Key, out var names))
{
foreach (var name in names)
{
registry.StopMonitoring(RegistryValue.UserHive.Cursors_Key, name);
}
logger.Info("Stopped monitoring cursors.");
}
else
{
logger.Warn("Failed to stop monitoring cursor registry values!");
}
}
internal bool Verify()
{
logger.Info($"Starting cursor verification...");
var success = registry.TryGetNames(RegistryValue.UserHive.Cursors_Key, out var cursors);
if (success)
{
foreach (var cursor in cursors.Where(c => !string.IsNullOrWhiteSpace(c)))
{
success &= VerifyCursor(cursor);
}
if (success)
{
logger.Info("Cursor configuration successfully verified.");
}
else
{
logger.Warn("Cursor configuration is compromised!");
}
}
else
{
logger.Error("Failed to verify cursor configuration!");
}
return success;
}
private void Registry_ValueChanged(string key, string name, object oldValue, object newValue)
{
if (key == RegistryValue.UserHive.Cursors_Key)
{
HandleCursorChange(key, name, oldValue, newValue);
}
}
private void HandleCursorChange(string key, string name, object oldValue, object newValue)
{
var args = new SentinelEventArgs();
logger.Warn($@"The cursor registry value '{key}\{name}' has changed from '{oldValue}' to '{newValue}'!");
Task.Run(() => CursorChanged?.Invoke(args)).ContinueWith((_) =>
{
if (args.Allow)
{
registry.StopMonitoring(key, name);
}
});
}
private bool VerifyCursor(string cursor)
{
var success = true;
success &= registry.TryRead(RegistryValue.UserHive.Cursors_Key, cursor, out var value);
success &= !(value is string) || (value is string path && (string.IsNullOrWhiteSpace(path) || IsValidCursorPath(path)));
if (!success)
{
if (value != default)
{
logger.Warn($"Configuration of cursor '{cursor}' is compromised: '{value}'!");
}
else
{
logger.Warn($"Failed to verify configuration of cursor '{cursor}'!");
}
}
return success;
}
private bool IsValidCursorPath(string path)
{
return path.StartsWith(USER_PATH, StringComparison.OrdinalIgnoreCase) || path.StartsWith(SYSTEM_PATH, StringComparison.OrdinalIgnoreCase);
}
}
}

View file

@ -1,95 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.Threading.Tasks;
using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.System.Events;
using SafeExamBrowser.SystemComponents.Contracts.Registry;
namespace SafeExamBrowser.Monitoring.System.Components
{
internal class EaseOfAccess
{
private readonly ILogger logger;
private readonly IRegistry registry;
internal event SentinelEventHandler EaseOfAccessChanged;
internal EaseOfAccess(ILogger logger, IRegistry registry)
{
this.logger = logger;
this.registry = registry;
}
internal void StartMonitoring()
{
registry.ValueChanged += Registry_ValueChanged;
registry.StartMonitoring(RegistryValue.MachineHive.EaseOfAccess_Key, RegistryValue.MachineHive.EaseOfAccess_Name);
logger.Info("Started monitoring ease of access.");
}
internal void StopMonitoring()
{
registry.ValueChanged -= Registry_ValueChanged;
registry.StopMonitoring(RegistryValue.MachineHive.EaseOfAccess_Key, RegistryValue.MachineHive.EaseOfAccess_Name);
logger.Info("Stopped monitoring ease of access.");
}
internal bool Verify()
{
logger.Info($"Starting ease of access verification...");
var success = registry.TryRead(RegistryValue.MachineHive.EaseOfAccess_Key, RegistryValue.MachineHive.EaseOfAccess_Name, out var value);
if (success)
{
if (value is string s && string.IsNullOrWhiteSpace(s))
{
logger.Info("Ease of access configuration successfully verified.");
}
else
{
logger.Warn($"Ease of access configuration is compromised: '{value}'!");
success = false;
}
}
else
{
success = true;
logger.Info("Ease of access configuration successfully verified (value does not exist).");
}
return success;
}
private void Registry_ValueChanged(string key, string name, object oldValue, object newValue)
{
if (key == RegistryValue.MachineHive.EaseOfAccess_Key)
{
HandleEaseOfAccessChange(key, name, oldValue, newValue);
}
}
private void HandleEaseOfAccessChange(string key, string name, object oldValue, object newValue)
{
var args = new SentinelEventArgs();
logger.Warn($@"The ease of access registry value '{key}\{name}' has changed from '{oldValue}' to '{newValue}'!");
Task.Run(() => EaseOfAccessChanged?.Invoke(args)).ContinueWith((_) =>
{
if (args.Allow)
{
registry.StopMonitoring(key, name);
}
});
}
}
}

View file

@ -1,175 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.Threading.Tasks;
using System.Timers;
using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.System.Events;
using SafeExamBrowser.WindowsApi.Contracts;
namespace SafeExamBrowser.Monitoring.System.Components
{
internal class StickyKeys
{
private readonly ILogger logger;
private readonly INativeMethods nativeMethods;
private readonly Timer timer;
private IStickyKeysState original;
internal event SentinelEventHandler Changed;
internal StickyKeys(ILogger logger, INativeMethods nativeMethods)
{
this.logger = logger;
this.nativeMethods = nativeMethods;
this.timer = new Timer();
}
internal bool Disable()
{
var success = nativeMethods.TryGetStickyKeys(out var state);
if (success)
{
success = nativeMethods.DisableStickyKeys();
if (success)
{
original = state;
logger.Info($"Disabled sticky keys (original state: {ToString(state)}).");
}
else
{
logger.Error($"Failed to disable sticky keys (original state: {ToString(state)})!");
}
}
else
{
logger.Error("Failed to retrieve sticky keys state!");
}
return success;
}
internal bool Enable()
{
var success = nativeMethods.TryGetStickyKeys(out var state);
if (success)
{
success = nativeMethods.EnableStickyKeys();
if (success)
{
original = state;
logger.Info($"Enabled sticky keys (original state: {ToString(state)}).");
}
else
{
logger.Error($"Failed to enable sticky keys (original state: {ToString(state)})!");
}
}
else
{
logger.Error("Failed to retrieve sticky keys state!");
}
return success;
}
internal bool Revert()
{
var success = true;
if (original != default)
{
success = nativeMethods.TrySetStickyKeys(original);
if (success)
{
logger.Info($"Successfully reverted sticky keys (original state: {ToString(original)}).");
}
else
{
logger.Error($"Failed to revert sticky keys (original state: {ToString(original)})!");
}
}
original = default;
return success;
}
internal void StartMonitoring()
{
timer.AutoReset = true;
timer.Elapsed += Timer_Elapsed;
timer.Interval = 1000;
timer.Start();
logger.Info("Started monitoring sticky keys.");
}
internal void StopMonitoring()
{
timer.Elapsed -= Timer_Elapsed;
timer.Stop();
logger.Info("Stopped monitoring sticky keys.");
}
private void Timer_Elapsed(object sender, ElapsedEventArgs e)
{
if (nativeMethods.TryGetStickyKeys(out var state))
{
if (state.IsEnabled || state.IsHotkeyActive)
{
HandleStickyKeysChange(state);
}
}
else
{
logger.Error("Failed to monitor sticky keys state!");
}
}
private void HandleStickyKeysChange(IStickyKeysState state)
{
var args = new SentinelEventArgs();
logger.Warn($"The sticky keys state has changed: {ToString(state)}.");
Task.Run(() => Changed?.Invoke(args)).ContinueWith((_) =>
{
if (args.Allow)
{
StopMonitoring();
}
});
if (nativeMethods.DisableStickyKeys())
{
logger.Info($"Disabled sticky keys.");
}
else
{
logger.Error($"Failed to disable sticky keys!");
}
}
private string ToString(IStickyKeysState state)
{
var availability = state.IsAvailable ? "available" : "unavailable";
var status = state.IsEnabled ? "enabled" : "disabled";
var hotkey = state.IsHotkeyActive ? "active" : "inactive";
return $"functionality {availability} and {status}, hotkey {hotkey}";
}
}
}

View file

@ -10,47 +10,46 @@ using System;
using System.Threading.Tasks; using System.Threading.Tasks;
using Microsoft.Win32; using Microsoft.Win32;
using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.System;
using SafeExamBrowser.Monitoring.Contracts.System.Events; using SafeExamBrowser.Monitoring.Contracts.System.Events;
namespace SafeExamBrowser.Monitoring.System.Components namespace SafeExamBrowser.Monitoring.System
{ {
internal class SystemEvents public class SystemMonitor : ISystemMonitor
{ {
private readonly ILogger logger; private readonly ILogger logger;
internal event SessionChangedEventHandler SessionChanged; public event SessionChangedEventHandler SessionChanged;
internal SystemEvents(ILogger logger) public SystemMonitor(ILogger logger)
{ {
this.logger = logger; this.logger = logger;
} }
internal void StartMonitoring() public void Start()
{ {
Microsoft.Win32.SystemEvents.EventsThreadShutdown += SystemEvents_EventsThreadShutdown; SystemEvents.EventsThreadShutdown += SystemEvents_EventsThreadShutdown;
Microsoft.Win32.SystemEvents.InstalledFontsChanged += SystemEvents_InstalledFontsChanged; SystemEvents.InstalledFontsChanged += SystemEvents_InstalledFontsChanged;
Microsoft.Win32.SystemEvents.PowerModeChanged += SystemEvents_PowerModeChanged; SystemEvents.PowerModeChanged += SystemEvents_PowerModeChanged;
Microsoft.Win32.SystemEvents.SessionEnded += SystemEvents_SessionEnded; SystemEvents.SessionEnded += SystemEvents_SessionEnded;
Microsoft.Win32.SystemEvents.SessionEnding += SystemEvents_SessionEnding; SystemEvents.SessionEnding += SystemEvents_SessionEnding;
Microsoft.Win32.SystemEvents.SessionSwitch += SystemEvents_SessionChanged; SystemEvents.SessionSwitch += SystemEvents_SessionChanged;
Microsoft.Win32.SystemEvents.TimeChanged += SystemEvents_TimeChanged; SystemEvents.TimeChanged += SystemEvents_TimeChanged;
Microsoft.Win32.SystemEvents.UserPreferenceChanged += SystemEvents_UserPreferenceChanged; SystemEvents.UserPreferenceChanged += SystemEvents_UserPreferenceChanged;
logger.Info("Started monitoring the operating system.");
logger.Info("Started monitoring system events.");
} }
internal void StopMonitoring() public void Stop()
{ {
Microsoft.Win32.SystemEvents.EventsThreadShutdown -= SystemEvents_EventsThreadShutdown; SystemEvents.EventsThreadShutdown -= SystemEvents_EventsThreadShutdown;
Microsoft.Win32.SystemEvents.InstalledFontsChanged -= SystemEvents_InstalledFontsChanged; SystemEvents.InstalledFontsChanged -= SystemEvents_InstalledFontsChanged;
Microsoft.Win32.SystemEvents.PowerModeChanged -= SystemEvents_PowerModeChanged; SystemEvents.PowerModeChanged -= SystemEvents_PowerModeChanged;
Microsoft.Win32.SystemEvents.SessionEnded -= SystemEvents_SessionEnded; SystemEvents.SessionEnded -= SystemEvents_SessionEnded;
Microsoft.Win32.SystemEvents.SessionEnding -= SystemEvents_SessionEnding; SystemEvents.SessionEnding -= SystemEvents_SessionEnding;
Microsoft.Win32.SystemEvents.SessionSwitch -= SystemEvents_SessionChanged; SystemEvents.SessionSwitch -= SystemEvents_SessionChanged;
Microsoft.Win32.SystemEvents.TimeChanged -= SystemEvents_TimeChanged; SystemEvents.TimeChanged -= SystemEvents_TimeChanged;
Microsoft.Win32.SystemEvents.UserPreferenceChanged -= SystemEvents_UserPreferenceChanged; SystemEvents.UserPreferenceChanged -= SystemEvents_UserPreferenceChanged;
logger.Info("Stopped monitoring the operating system.");
logger.Info("Stopped monitoring system events.");
} }
private void SystemEvents_EventsThreadShutdown(object sender, EventArgs e) private void SystemEvents_EventsThreadShutdown(object sender, EventArgs e)

View file

@ -1,95 +0,0 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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 SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.System;
using SafeExamBrowser.Monitoring.Contracts.System.Events;
using SafeExamBrowser.Monitoring.System.Components;
using SafeExamBrowser.SystemComponents.Contracts.Registry;
using SafeExamBrowser.WindowsApi.Contracts;
namespace SafeExamBrowser.Monitoring.System
{
public class SystemSentinel : ISystemSentinel
{
private readonly Cursors cursors;
private readonly EaseOfAccess easeOfAccess;
private readonly StickyKeys stickyKeys;
private readonly SystemEvents systemEvents;
public event SentinelEventHandler CursorChanged;
public event SentinelEventHandler EaseOfAccessChanged;
public event SentinelEventHandler StickyKeysChanged;
public event SessionChangedEventHandler SessionChanged;
public SystemSentinel(ILogger logger, INativeMethods nativeMethods, IRegistry registry)
{
cursors = new Cursors(logger, registry);
easeOfAccess = new EaseOfAccess(logger, registry);
stickyKeys = new StickyKeys(logger, nativeMethods);
systemEvents = new SystemEvents(logger);
}
public bool DisableStickyKeys()
{
return stickyKeys.Disable();
}
public bool EnableStickyKeys()
{
return stickyKeys.Enable();
}
public bool RevertStickyKeys()
{
return stickyKeys.Revert();
}
public void StartMonitoringCursors()
{
cursors.CursorChanged += (args) => CursorChanged?.Invoke(args);
cursors.StartMonitoring();
}
public void StartMonitoringEaseOfAccess()
{
easeOfAccess.EaseOfAccessChanged += (args) => EaseOfAccessChanged?.Invoke(args);
easeOfAccess.StartMonitoring();
}
public void StartMonitoringStickyKeys()
{
stickyKeys.Changed += (args) => StickyKeysChanged?.Invoke(args);
stickyKeys.StartMonitoring();
}
public void StartMonitoringSystemEvents()
{
systemEvents.SessionChanged += () => SessionChanged?.Invoke();
systemEvents.StartMonitoring();
}
public void StopMonitoring()
{
cursors.StopMonitoring();
easeOfAccess.StopMonitoring();
stickyKeys.StopMonitoring();
systemEvents.StopMonitoring();
}
public bool VerifyCursors()
{
return cursors.Verify();
}
public bool VerifyEaseOfAccess()
{
return easeOfAccess.Verify();
}
}
}

View file

@ -0,0 +1,255 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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.IO;
using System.Reflection;
using System.Threading;
using System.Windows;
using Microsoft.Web.WebView2.Wpf;
using SafeExamBrowser.Configuration.Contracts;
using SafeExamBrowser.Core.Contracts.Resources.Icons;
using SafeExamBrowser.I18n.Contracts;
using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Server.Contracts.Events.Proctoring;
using SafeExamBrowser.Settings.Proctoring;
using SafeExamBrowser.SystemComponents.Contracts;
using SafeExamBrowser.UserInterface.Contracts;
using SafeExamBrowser.UserInterface.Contracts.Proctoring;
namespace SafeExamBrowser.Proctoring.JitsiMeet
{
internal class JitsiMeetImplementation : ProctoringImplementation
{
private readonly AppConfig appConfig;
private readonly IFileSystem fileSystem;
private readonly IModuleLogger logger;
private readonly ProctoringSettings settings;
private readonly IText text;
private readonly IUserInterfaceFactory uiFactory;
private ProctoringControl control;
private string filePath;
private WindowVisibility initialVisibility;
private IProctoringWindow window;
internal override string Name => nameof(JitsiMeet);
internal JitsiMeetImplementation(
AppConfig appConfig,
IFileSystem fileSystem,
IModuleLogger logger,
ProctoringSettings settings,
IText text,
IUserInterfaceFactory uiFactory)
{
this.appConfig = appConfig;
this.fileSystem = fileSystem;
this.logger = logger;
this.settings = settings;
this.text = text;
this.uiFactory = uiFactory;
}
internal override void Initialize()
{
var start = true;
initialVisibility = settings.WindowVisibility;
settings.JitsiMeet.ServerUrl = Sanitize(settings.JitsiMeet.ServerUrl);
start &= !string.IsNullOrWhiteSpace(settings.JitsiMeet.RoomName);
start &= !string.IsNullOrWhiteSpace(settings.JitsiMeet.ServerUrl);
if (start)
{
logger.Info($"Initialized proctoring: All settings are valid, starting automatically...");
Start();
}
else
{
ShowNotificationInactive();
logger.Info($"Initialized proctoring: Not all settings are valid or a server session is active, not starting automatically.");
}
}
internal override void ProctoringConfigurationReceived(bool allowChat, bool receiveAudio, bool receiveVideo)
{
logger.Info("Proctoring configuration received.");
settings.JitsiMeet.AllowChat = allowChat;
settings.JitsiMeet.ReceiveAudio = receiveAudio;
settings.JitsiMeet.ReceiveVideo = receiveVideo;
if (allowChat || receiveVideo)
{
settings.WindowVisibility = WindowVisibility.AllowToHide;
}
else
{
settings.WindowVisibility = initialVisibility;
}
Stop();
Start();
logger.Info($"Successfully updated configuration: {nameof(allowChat)}={allowChat}, {nameof(receiveAudio)}={receiveAudio}, {nameof(receiveVideo)}={receiveVideo}.");
}
internal override void ProctoringInstructionReceived(InstructionEventArgs args)
{
if (args is JitsiMeetInstruction instruction)
{
logger.Info($"Proctoring instruction received: {instruction.Method}");
if (instruction.Method == InstructionMethod.Join)
{
settings.JitsiMeet.RoomName = instruction.RoomName;
settings.JitsiMeet.ServerUrl = instruction.ServerUrl;
settings.JitsiMeet.Token = instruction.Token;
Stop();
Start();
}
else
{
Stop();
}
logger.Info("Successfully processed instruction.");
}
}
internal override void Start()
{
Application.Current.Dispatcher.Invoke(() =>
{
try
{
var content = LoadContent(settings);
filePath = Path.Combine(appConfig.TemporaryDirectory, $"{Path.GetRandomFileName()}_index.html");
fileSystem.Save(content, filePath);
control = new ProctoringControl(logger.CloneFor(nameof(ProctoringControl)), settings);
control.CreationProperties = new CoreWebView2CreationProperties { UserDataFolder = appConfig.TemporaryDirectory };
control.EnsureCoreWebView2Async().ContinueWith(_ =>
{
control.Dispatcher.Invoke(() =>
{
control.CoreWebView2.Navigate(filePath);
});
});
window = uiFactory.CreateProctoringWindow(control);
window.SetTitle(settings.JitsiMeet.Enabled ? settings.JitsiMeet.Subject : "");
window.Show();
if (settings.WindowVisibility == WindowVisibility.AllowToShow || settings.WindowVisibility == WindowVisibility.Hidden)
{
window.Hide();
}
ShowNotificationActive();
logger.Info("Started proctoring.");
}
catch (Exception e)
{
logger.Error($"Failed to start proctoring! Reason: {e.Message}", e);
}
});
}
internal override void Stop()
{
if (control != default && window != default)
{
control.Dispatcher.Invoke(() =>
{
control.ExecuteScriptAsync("api.executeCommand('hangup'); api.dispose();");
Thread.Sleep(2000);
window.Close();
control = default;
window = default;
fileSystem.Delete(filePath);
ShowNotificationInactive();
logger.Info("Stopped proctoring.");
});
}
}
internal override void Terminate()
{
Stop();
logger.Info("Terminated proctoring.");
}
protected override void ActivateNotification()
{
if (settings.WindowVisibility == WindowVisibility.Visible)
{
window?.BringToForeground();
}
else if (settings.WindowVisibility == WindowVisibility.AllowToHide || settings.WindowVisibility == WindowVisibility.AllowToShow)
{
window?.Toggle();
}
}
private string LoadContent(ProctoringSettings settings)
{
var assembly = Assembly.GetAssembly(typeof(ProctoringController));
var path = $"{typeof(ProctoringController).Namespace}.JitsiMeet.index.html";
using (var stream = assembly.GetManifestResourceStream(path))
using (var reader = new StreamReader(stream))
{
var html = reader.ReadToEnd();
html = html.Replace("%%_ALLOW_CHAT_%%", settings.JitsiMeet.AllowChat ? "chat" : "");
html = html.Replace("%%_ALLOW_CLOSED_CAPTIONS_%%", settings.JitsiMeet.AllowClosedCaptions ? "closedcaptions" : "");
html = html.Replace("%%_ALLOW_RAISE_HAND_%%", settings.JitsiMeet.AllowRaiseHand ? "raisehand" : "");
html = html.Replace("%%_ALLOW_RECORDING_%%", settings.JitsiMeet.AllowRecording ? "recording" : "");
html = html.Replace("%%_ALLOW_TILE_VIEW", settings.JitsiMeet.AllowTileView ? "tileview" : "");
html = html.Replace("'%_AUDIO_MUTED_%'", settings.JitsiMeet.AudioMuted && settings.WindowVisibility != WindowVisibility.Hidden ? "true" : "false");
html = html.Replace("'%_AUDIO_ONLY_%'", settings.JitsiMeet.AudioOnly ? "true" : "false");
html = html.Replace("'%_VIDEO_MUTED_%'", settings.JitsiMeet.VideoMuted && settings.WindowVisibility != WindowVisibility.Hidden ? "true" : "false");
return html;
}
}
private string Sanitize(string serverUrl)
{
return serverUrl?.Replace($"{Uri.UriSchemeHttp}{Uri.SchemeDelimiter}", "").Replace($"{Uri.UriSchemeHttps}{Uri.SchemeDelimiter}", "");
}
private void ShowNotificationActive()
{
CanActivate = true;
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/Proctoring_Active.xaml") };
Tooltip = text.Get(TextKey.Notification_ProctoringActiveTooltip);
InvokeNotificationChanged();
}
private void ShowNotificationInactive()
{
CanActivate = false;
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/Proctoring_Inactive.xaml") };
Tooltip = text.Get(TextKey.Notification_ProctoringInactiveTooltip);
InvokeNotificationChanged();
}
}
}

View file

@ -0,0 +1,69 @@
<html>
<head>
<meta charset="utf-8" />
</head>
<body style="margin: 0">
<div id="placeholder" />
<script src='https://meet.jit.si/external_api.js'></script>
<script type="text/javascript">
var api;
function startMeeting(credentials) {
var configOverwrite = {
disableProfile: true,
prejoinPageEnabled: false,
startAudioOnly: '%_AUDIO_ONLY_%',
startWithAudioMuted: '%_AUDIO_MUTED_%',
startWithVideoMuted: '%_VIDEO_MUTED_%'
};
var interfaceOverwrite = {
JITSI_WATERMARK_LINK: '',
SHOW_JITSI_WATERMARK: false,
TOOLBAR_BUTTONS: [
'microphone', 'camera', '%%_ALLOW_CLOSED_CAPTIONS_%%', /*'desktop',*/ 'embedmeeting', 'fullscreen',
'fodeviceselection', /*'hangup',*/ 'profile', '%%_ALLOW_CHAT_%%', '%%_ALLOW_RECORDING_%%',
'livestreaming', 'etherpad', /*'sharedvideo',*/ 'settings', '%%_ALLOW_RAISE_HAND_%%',
'videoquality', 'filmstrip', 'invite', 'feedback', 'stats', 'shortcuts',
'%%_ALLOW_TILE_VIEW_%%', 'select-background', 'download', 'help', /*'mute-everyone',*/ 'mute-video-everyone', 'security'
]
};
var options = {
configOverwrite: configOverwrite,
height: '100%',
interfaceConfigOverwrite: interfaceOverwrite,
jwt: credentials.token,
parentNode: document.querySelector('#placeholder'),
roomName: credentials.roomName,
width: '100%'
};
api = new JitsiMeetExternalAPI(credentials.domain, options);
api.addListener('audioMuteStatusChanged', args => {
if (args.muted) {
api.executeCommand('toggleAudio');
}
});
api.addListener('videoMuteStatusChanged', args => {
if (args.muted) {
api.executeCommand('toggleVideo');
}
});
api.executeCommand('subject', credentials.subject);
}
function webMessageReceived(args) {
if ('credentials' in args.data) {
startMeeting(args.data.credentials);
}
}
window.addEventListener('unload', () => {
api.executeCommand('hangup');
api.dispose();
});
window.chrome.webview.addEventListener('message', webMessageReceived);
window.chrome.webview.postMessage('credentials');
</script>
</body>
</html>

View file

@ -0,0 +1,104 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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 Microsoft.Web.WebView2.Core;
using Microsoft.Web.WebView2.Wpf;
using Newtonsoft.Json.Linq;
using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Settings.Proctoring;
using SafeExamBrowser.UserInterface.Contracts.Proctoring;
using SafeExamBrowser.UserInterface.Contracts.Proctoring.Events;
namespace SafeExamBrowser.Proctoring
{
internal class ProctoringControl : WebView2, IProctoringControl
{
private readonly ILogger logger;
private readonly ProctoringSettings settings;
public event FullScreenChangedEventHandler FullScreenChanged;
internal ProctoringControl(ILogger logger, ProctoringSettings settings)
{
this.logger = logger;
this.settings = settings;
CoreWebView2InitializationCompleted += ProctoringControl_CoreWebView2InitializationCompleted;
}
private void ProctoringControl_CoreWebView2InitializationCompleted(object sender, CoreWebView2InitializationCompletedEventArgs e)
{
if (e.IsSuccess)
{
CoreWebView2.Settings.AreDefaultContextMenusEnabled = false;
CoreWebView2.Settings.AreDevToolsEnabled = false;
CoreWebView2.Settings.IsStatusBarEnabled = false;
CoreWebView2.ContainsFullScreenElementChanged += CoreWebView2_ContainsFullScreenElementChanged;
CoreWebView2.PermissionRequested += CoreWebView2_PermissionRequested;
CoreWebView2.WebMessageReceived += CoreWebView2_WebMessageReceived;
logger.Info("Successfully initialized.");
}
else
{
logger.Error("Failed to initialize!", e.InitializationException);
}
}
private void CoreWebView2_ContainsFullScreenElementChanged(object sender, object e)
{
FullScreenChanged?.Invoke(CoreWebView2.ContainsFullScreenElement);
logger.Debug($"Full screen {(CoreWebView2.ContainsFullScreenElement ? "activated" : "deactivated")}.");
}
private void CoreWebView2_PermissionRequested(object sender, CoreWebView2PermissionRequestedEventArgs e)
{
if (e.PermissionKind == CoreWebView2PermissionKind.Camera || e.PermissionKind == CoreWebView2PermissionKind.Microphone)
{
e.State = CoreWebView2PermissionState.Allow;
logger.Info($"Granted access to {e.PermissionKind}.");
}
else
{
logger.Info($"Denied access to {e.PermissionKind}.");
}
}
private void CoreWebView2_WebMessageReceived(object sender, CoreWebView2WebMessageReceivedEventArgs e)
{
var message = e.TryGetWebMessageAsString();
logger.Debug($"Received web message '{message}'.");
switch (message)
{
case "credentials":
SendCredentials();
break;
}
}
private void SendCredentials()
{
var message = new JObject();
var credentials = new JObject();
if (settings.JitsiMeet.Enabled)
{
credentials.Add(new JProperty("domain", settings.JitsiMeet.ServerUrl));
credentials.Add(new JProperty("roomName", settings.JitsiMeet.RoomName));
credentials.Add(new JProperty("subject", settings.JitsiMeet.ShowMeetingName ? settings.JitsiMeet.Subject : ""));
credentials.Add(new JProperty("token", settings.JitsiMeet.Token));
}
message.Add("credentials", credentials);
logger.Debug("Sending credentials to proctoring client.");
CoreWebView2.PostWebMessageAsJson(message.ToString());
}
}
}

View file

@ -12,6 +12,7 @@ using SafeExamBrowser.Configuration.Contracts;
using SafeExamBrowser.I18n.Contracts; using SafeExamBrowser.I18n.Contracts;
using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.Applications; using SafeExamBrowser.Monitoring.Contracts.Applications;
using SafeExamBrowser.Proctoring.JitsiMeet;
using SafeExamBrowser.Proctoring.ScreenProctoring; using SafeExamBrowser.Proctoring.ScreenProctoring;
using SafeExamBrowser.Proctoring.ScreenProctoring.Service; using SafeExamBrowser.Proctoring.ScreenProctoring.Service;
using SafeExamBrowser.Settings.Proctoring; using SafeExamBrowser.Settings.Proctoring;
@ -56,6 +57,13 @@ namespace SafeExamBrowser.Proctoring
{ {
var implementations = new List<ProctoringImplementation>(); var implementations = new List<ProctoringImplementation>();
if (settings.JitsiMeet.Enabled)
{
var logger = this.logger.CloneFor(nameof(JitsiMeet));
implementations.Add(new JitsiMeetImplementation(appConfig, fileSystem, logger, settings, text, uiFactory));
}
if (settings.ScreenProctoring.Enabled) if (settings.ScreenProctoring.Enabled)
{ {
var logger = this.logger.CloneFor(nameof(ScreenProctoring)); var logger = this.logger.CloneFor(nameof(ScreenProctoring));

View file

@ -62,6 +62,15 @@
<Reference Include="KGySoft.Drawing.Core, Version=8.1.0.0, Culture=neutral, PublicKeyToken=b45eba277439ddfe, processorArchitecture=MSIL"> <Reference Include="KGySoft.Drawing.Core, Version=8.1.0.0, Culture=neutral, PublicKeyToken=b45eba277439ddfe, processorArchitecture=MSIL">
<HintPath>..\packages\KGySoft.Drawing.Core.8.1.0\lib\net46\KGySoft.Drawing.Core.dll</HintPath> <HintPath>..\packages\KGySoft.Drawing.Core.8.1.0\lib\net46\KGySoft.Drawing.Core.dll</HintPath>
</Reference> </Reference>
<Reference Include="Microsoft.Web.WebView2.Core, Version=1.0.2365.46, Culture=neutral, PublicKeyToken=2a8ab48044d2601e, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.Web.WebView2.1.0.2365.46\lib\net45\Microsoft.Web.WebView2.Core.dll</HintPath>
</Reference>
<Reference Include="Microsoft.Web.WebView2.WinForms, Version=1.0.2365.46, Culture=neutral, PublicKeyToken=2a8ab48044d2601e, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.Web.WebView2.1.0.2365.46\lib\net45\Microsoft.Web.WebView2.WinForms.dll</HintPath>
</Reference>
<Reference Include="Microsoft.Web.WebView2.Wpf, Version=1.0.2365.46, Culture=neutral, PublicKeyToken=2a8ab48044d2601e, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.Web.WebView2.1.0.2365.46\lib\net45\Microsoft.Web.WebView2.Wpf.dll</HintPath>
</Reference>
<Reference Include="Newtonsoft.Json, Version=13.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL"> <Reference Include="Newtonsoft.Json, Version=13.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL">
<HintPath>..\packages\Newtonsoft.Json.13.0.3\lib\net45\Newtonsoft.Json.dll</HintPath> <HintPath>..\packages\Newtonsoft.Json.13.0.3\lib\net45\Newtonsoft.Json.dll</HintPath>
</Reference> </Reference>
@ -78,6 +87,8 @@
<Reference Include="WindowsBase" /> <Reference Include="WindowsBase" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<Compile Include="JitsiMeet\JitsiMeetImplementation.cs" />
<Compile Include="ProctoringControl.cs" />
<Compile Include="ProctoringController.cs" /> <Compile Include="ProctoringController.cs" />
<Compile Include="ProctoringFactory.cs" /> <Compile Include="ProctoringFactory.cs" />
<Compile Include="ProctoringImplementation.cs" /> <Compile Include="ProctoringImplementation.cs" />
@ -165,9 +176,19 @@
<Name>SafeExamBrowser.WindowsApi.Contracts</Name> <Name>SafeExamBrowser.WindowsApi.Contracts</Name>
</ProjectReference> </ProjectReference>
</ItemGroup> </ItemGroup>
<ItemGroup>
<EmbeddedResource Include="JitsiMeet\index.html" />
</ItemGroup>
<ItemGroup> <ItemGroup>
<None Include="packages.config" /> <None Include="packages.config" />
</ItemGroup> </ItemGroup>
<ItemGroup /> <ItemGroup />
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" /> <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
<Import Project="..\packages\Microsoft.Web.WebView2.1.0.2365.46\build\Microsoft.Web.WebView2.targets" Condition="Exists('..\packages\Microsoft.Web.WebView2.1.0.2365.46\build\Microsoft.Web.WebView2.targets')" />
<Target Name="EnsureNuGetPackageBuildImports" BeforeTargets="PrepareForBuild">
<PropertyGroup>
<ErrorText>This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}.</ErrorText>
</PropertyGroup>
<Error Condition="!Exists('..\packages\Microsoft.Web.WebView2.1.0.2365.46\build\Microsoft.Web.WebView2.targets')" Text="$([System.String]::Format('$(ErrorText)', '..\packages\Microsoft.Web.WebView2.1.0.2365.46\build\Microsoft.Web.WebView2.targets'))" />
</Target>
</Project> </Project>

View file

@ -26,6 +26,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
private string applicationInfo; private string applicationInfo;
private string browserInfo; private string browserInfo;
private string browserInfoWithoutUrls;
private TimeSpan elapsed; private TimeSpan elapsed;
private string triggerInfo; private string triggerInfo;
private string urls; private string urls;
@ -75,7 +76,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
CaptureMouseTrigger(mouse); CaptureMouseTrigger(mouse);
} }
logger.Debug($"Captured metadata: {applicationInfo} / {browserInfo} / {urlCount} URL(s) / {triggerInfo} / {windowTitle}."); logger.Debug($"Captured metadata: {applicationInfo} / {browserInfoWithoutUrls} / {triggerInfo} / {urlCount} URL(s) / {windowTitle}.");
} }
private void CaptureApplicationData() private void CaptureApplicationData()
@ -100,7 +101,8 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
{ {
var windows = browser.GetWindows(); var windows = browser.GetWindows();
browserInfo = string.Join(", ", windows.Select(w => $"{(w.IsMainWindow ? "Main" : "Additional")} Window: {w.Title}")); browserInfo = string.Join(", ", windows.Select(w => $"{(w.IsMainWindow ? "Main" : "Additional")} Window: {w.Title} ({w.Url})"));
browserInfoWithoutUrls = string.Join(", ", windows.Select(w => $"{(w.IsMainWindow ? "Main" : "Additional")} Window: {w.Title}"));
urls = string.Join(", ", windows.Select(w => w.Url)); urls = string.Join(", ", windows.Select(w => w.Url));
urlCount = windows.Count(); urlCount = windows.Count();
} }
@ -113,19 +115,10 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
private void CaptureKeyboardTrigger(KeyboardTrigger keyboard) private void CaptureKeyboardTrigger(KeyboardTrigger keyboard)
{ {
var flags = Enum.GetValues(typeof(KeyModifier)) var flags = Enum.GetValues(typeof(KeyModifier)).OfType<KeyModifier>().Where(m => m != KeyModifier.None && keyboard.Modifier.HasFlag(m));
.OfType<KeyModifier>()
.Where(m => m != KeyModifier.None && keyboard.Modifier.HasFlag(m) && !keyboard.Key.ToString().Contains(m.ToString()));
var modifiers = flags.Any() ? string.Join(" + ", flags) + " + " : string.Empty; var modifiers = flags.Any() ? string.Join(" + ", flags) + " + " : string.Empty;
if (flags.Any()) triggerInfo = $"'{modifiers}{keyboard.Key}' has been {keyboard.State.ToString().ToLower()}.";
{
triggerInfo = $"'{modifiers}{keyboard.Key}' has been {keyboard.State.ToString().ToLower()}.";
}
else
{
triggerInfo = $"A key has been {keyboard.State.ToString().ToLower()}.";
}
} }
private void CaptureMouseTrigger(MouseTrigger mouse) private void CaptureMouseTrigger(MouseTrigger mouse)
@ -163,6 +156,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
{ {
applicationInfo = "-"; applicationInfo = "-";
browserInfo = "-"; browserInfo = "-";
browserInfoWithoutUrls = "-";
triggerInfo = "-"; triggerInfo = "-";
urls = "-"; urls = "-";
windowTitle = "-"; windowTitle = "-";

View file

@ -166,7 +166,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
{ {
logger.Info("Connecting to service..."); logger.Info("Connecting to service...");
var connect = service.Connect(settings.ClientId, settings.ClientSecret, settings.ServiceUrl); var connect = service.Connect(settings.ServiceUrl);
if (connect.Success) if (connect.Success)
{ {
@ -191,11 +191,11 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
} }
} }
private void UpdateNotification(bool active) private void UpdateNotification(bool live)
{ {
CanActivate = false; CanActivate = false;
if (active) if (live)
{ {
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ScreenProctoring_Active.xaml") }; IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ScreenProctoring_Active.xaml") };
Tooltip = text.Get(TextKey.Notification_ProctoringActiveTooltip); Tooltip = text.Get(TextKey.Notification_ProctoringActiveTooltip);

View file

@ -8,7 +8,6 @@
using System; using System;
using System.Net.Http; using System.Net.Http;
using System.Text;
namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
{ {
@ -16,20 +15,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
{ {
internal static string ToLogString(this HttpResponseMessage response) internal static string ToLogString(this HttpResponseMessage response)
{ {
return response == default ? "No Response" : $"{(int) response.StatusCode} {response.StatusCode} {response.ReasonPhrase}"; return $"{(int?) response?.StatusCode} {response?.StatusCode} {response?.ReasonPhrase}";
}
internal static string ToSummary(this Exception exception)
{
var trimChars = new[] { '.', '!' };
var summary = new StringBuilder(exception.Message?.TrimEnd(trimChars));
for (var inner = exception.InnerException; inner != default; inner = inner.InnerException)
{
summary.Append($" -> {inner.Message?.TrimEnd(trimChars)}");
}
return summary.ToString();
} }
internal static long ToUnixTimestamp(this DateTime date) internal static long ToUnixTimestamp(this DateTime date)

View file

@ -17,11 +17,8 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
{ {
} }
internal bool TryExecute(string clientId, string clientSecret, out string message) internal bool TryExecute(out string message)
{ {
ClientId = clientId;
ClientSecret = clientSecret;
return TryRetrieveOAuth2Token(out message); return TryRetrieveOAuth2Token(out message);
} }
} }

View file

@ -21,21 +21,29 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
{ {
private const int ATTEMPTS = 5; private const int ATTEMPTS = 5;
private static string connectionToken;
private static string oauth2Token; private static string oauth2Token;
private readonly HttpClient httpClient; private readonly HttpClient httpClient;
private bool hadException;
protected readonly Api api; protected readonly Api api;
protected readonly ILogger logger; protected readonly ILogger logger;
protected readonly Parser parser; protected readonly Parser parser;
protected static string ClientId { get; set; }
protected static string ClientSecret { get; set; }
protected (string, string) Authorization => (Header.AUTHORIZATION, $"Bearer {oauth2Token}"); protected (string, string) Authorization => (Header.AUTHORIZATION, $"Bearer {oauth2Token}");
internal static string ConnectionToken
{
get { return connectionToken; }
set { connectionToken = value; }
}
internal static string Oauth2Token
{
get { return oauth2Token; }
set { oauth2Token = value; }
}
protected Request(Api api, HttpClient httpClient, ILogger logger, Parser parser) protected Request(Api api, HttpClient httpClient, ILogger logger, Parser parser)
{ {
this.api = api; this.api = api;
@ -76,16 +84,12 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
} }
catch (TaskCanceledException) catch (TaskCanceledException)
{ {
logger.Warn($"Request {request.Method} '{request.RequestUri}' did not complete within {httpClient.Timeout}!"); logger.Error($"Request {request.Method} '{request.RequestUri}' did not complete within {httpClient.Timeout}ms!");
break; break;
} }
catch (Exception e) catch (Exception e)
{ {
if (IsFirstException()) logger.Error($"Request {request.Method} '{request.RequestUri}' has failed!", e);
{
logger.Warn($"Request {request.Method} '{request.RequestUri}' has failed: {e.ToSummary()}!");
}
} }
} }
@ -94,7 +98,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
protected bool TryRetrieveOAuth2Token(out string message) protected bool TryRetrieveOAuth2Token(out string message)
{ {
var secret = Convert.ToBase64String(Encoding.UTF8.GetBytes($"{ClientId}:{ClientSecret}")); var secret = Convert.ToBase64String(Encoding.UTF8.GetBytes("test:test"));
var authorization = (Header.AUTHORIZATION, $"Basic {secret}"); var authorization = (Header.AUTHORIZATION, $"Basic {secret}");
var content = "grant_type=client_credentials&scope=read write"; var content = "grant_type=client_credentials&scope=read write";
var success = TryExecute(HttpMethod.Post, api.AccessTokenEndpoint, out var response, content, ContentType.URL_ENCODED, authorization); var success = TryExecute(HttpMethod.Post, api.AccessTokenEndpoint, out var response, content, ContentType.URL_ENCODED, authorization);
@ -150,15 +154,6 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
return request; return request;
} }
private bool IsFirstException()
{
var isFirst = !hadException;
hadException = true;
return isFirst;
}
private (string name, string value)[] UpdateOAuth2Token((string name, string value)[] headers) private (string name, string value)[] UpdateOAuth2Token((string name, string value)[] headers)
{ {
var result = new List<(string name, string value)>(); var result = new List<(string name, string value)>();

View file

@ -7,7 +7,6 @@
*/ */
using System; using System;
using System.Net;
using System.Net.Http; using System.Net.Http;
using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Proctoring.ScreenProctoring.Data; using SafeExamBrowser.Proctoring.ScreenProctoring.Data;
@ -25,8 +24,8 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
internal bool TryExecute(MetaData metaData, ScreenShot screenShot, string sessionId, out int health, out string message) internal bool TryExecute(MetaData metaData, ScreenShot screenShot, string sessionId, out int health, out string message)
{ {
var imageFormat = (Header.IMAGE_FORMAT, ToString(screenShot.Format)); var imageFormat = (Header.IMAGE_FORMAT, ToString(screenShot.Format));
var metdataJson = (Header.METADATA, WebUtility.UrlEncode(metaData.ToJson())); var metdataJson = (Header.METADATA, metaData.ToJson());
var timestamp = (Header.TIMESTAMP, screenShot.CaptureTime.ToUnixTimestamp().ToString()); var timestamp = (Header.TIMESTAMP, DateTime.Now.ToUnixTimestamp().ToString());
var url = api.ScreenShotEndpoint.Replace(Api.SESSION_ID, sessionId); var url = api.ScreenShotEndpoint.Replace(Api.SESSION_ID, sessionId);
var success = TryExecute(HttpMethod.Post, url, out var response, screenShot.Data, ContentType.OCTET_STREAM, Authorization, imageFormat, metdataJson, timestamp); var success = TryExecute(HttpMethod.Post, url, out var response, screenShot.Data, ContentType.OCTET_STREAM, Authorization, imageFormat, metdataJson, timestamp);

View file

@ -33,16 +33,12 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service
this.parser = new Parser(logger); this.parser = new Parser(logger);
} }
internal ServiceResponse Connect(string clientId, string clientSecret, string serviceUrl) internal ServiceResponse Connect(string serviceUrl)
{ {
httpClient = new HttpClient httpClient = new HttpClient { BaseAddress = new Uri(serviceUrl) };
{
BaseAddress = new Uri(serviceUrl),
Timeout = TimeSpan.FromSeconds(10)
};
var request = new OAuth2TokenRequest(api, httpClient, logger, parser); var request = new OAuth2TokenRequest(api, httpClient, logger, parser);
var success = request.TryExecute(clientId, clientSecret, out var message); var success = request.TryExecute(out var message);
if (success) if (success)
{ {
@ -85,7 +81,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service
} }
else else
{ {
logger.Warn("Failed to query health!"); logger.Error("Failed to query health!");
} }
return new ServiceResponse<int>(success, health, message); return new ServiceResponse<int>(success, health, message);

View file

@ -62,7 +62,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
var progress = 0; var progress = 0;
var total = previous; var total = previous;
while (HasRemainingWork() && service.IsConnected && (!networkIssue || recovering || DateTime.Now < resume)) while (HasRemainingWork() && service.IsConnected && (!networkIssue || recovering))
{ {
var remaining = buffer.Count + cache.Count; var remaining = buffer.Count + cache.Count;
@ -80,7 +80,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
updateStatus(new RemainingWorkUpdatedEventArgs updateStatus(new RemainingWorkUpdatedEventArgs
{ {
IsWaiting = recovering || networkIssue, IsWaiting = recovering,
Next = buffer.TryPeek(out _, out var schedule, out _) ? schedule : default(DateTime?), Next = buffer.TryPeek(out _, out var schedule, out _) ? schedule : default(DateTime?),
Progress = progress, Progress = progress,
Resume = resume, Resume = resume,
@ -121,7 +121,6 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
thread.Start(); thread.Start();
timer.AutoReset = false; timer.AutoReset = false;
timer.Elapsed += Timer_Elapsed;
timer.Interval = FIFTEEN_SECONDS; timer.Interval = FIFTEEN_SECONDS;
} }
@ -196,6 +195,19 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
private void ExecuteCaching() private void ExecuteCaching()
{ {
const int THREE_MINUTES = 180;
if (!recovering)
{
recovering = true;
resume = DateTime.Now.AddSeconds(random.Next(0, THREE_MINUTES));
timer.Elapsed += Timer_Elapsed;
timer.Start();
logger.Warn($"Activating local caching and suspending transmission due to bad service health (value: {health}, resume: {resume:HH:mm:ss}).");
}
CacheFromBuffer(); CacheFromBuffer();
CacheFromQueue(); CacheFromQueue();
} }
@ -225,7 +237,9 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
else else
{ {
timer.Stop(); timer.Stop();
logger.Info($"Recovery terminated, deactivating local caching and resuming transmission."); timer.Elapsed -= Timer_Elapsed;
logger.Info($"Deactivating local caching and resuming transmission due to improved service health (value: {health}).");
} }
} }
@ -264,7 +278,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
} }
else else
{ {
buffer.Enqueue(metaData, CalculateSchedule(metaData), screenShot); buffer.Enqueue(metaData, DateTime.Now, screenShot);
} }
} }
} }
@ -282,18 +296,10 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
var hasItem = buffer.TryPeek(out var metaData, out var schedule, out var screenShot); var hasItem = buffer.TryPeek(out var metaData, out var schedule, out var screenShot);
var ready = schedule <= DateTime.Now; var ready = schedule <= DateTime.Now;
if (hasItem && ready) if (hasItem && ready && TryTransmit(metaData, screenShot))
{ {
buffer.Dequeue(); buffer.Dequeue();
screenShot.Dispose();
if (TryTransmit(metaData, screenShot))
{
screenShot.Dispose();
}
else
{
buffer.Enqueue(metaData, CalculateSchedule(metaData), screenShot);
}
} }
} }
@ -307,7 +313,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
} }
else else
{ {
buffer.Enqueue(metaData, CalculateSchedule(metaData), screenShot); buffer.Enqueue(metaData, DateTime.Now, screenShot);
} }
} }
} }
@ -322,7 +328,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
} }
else else
{ {
buffer.Enqueue(metaData, CalculateSchedule(metaData), screenShot); buffer.Enqueue(metaData, DateTime.Now, screenShot);
} }
} }
} }
@ -348,11 +354,14 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
if (service.IsConnected) if (service.IsConnected)
{ {
var response = service.Send(metaData, screenShot); var response = service.Send(metaData, screenShot);
var value = response.Success ? response.Value : BAD;
health = UpdateHealth(value);
networkIssue = !response.Success; networkIssue = !response.Success;
success = response.Success; success = response.Success;
if (response.Success)
{
health = UpdateHealth(response.Value);
}
} }
else else
{ {
@ -367,52 +376,30 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
if (service.IsConnected) if (service.IsConnected)
{ {
var response = service.GetHealth(); var response = service.GetHealth();
var value = response.Success ? response.Value : BAD;
health = UpdateHealth(value);
networkIssue = !response.Success; networkIssue = !response.Success;
if (response.Success)
{
health = UpdateHealth(response.Value);
}
} }
else else
{ {
logger.Warn("Cannot query health as service is disconnected!"); logger.Warn("Cannot query health as service is disconnected!");
} }
if (!timer.Enabled) timer.Start();
{
timer.Start();
}
} }
private int UpdateHealth(int value) private int UpdateHealth(int value)
{ {
const int THREE_MINUTES = 180;
var previous = health; var previous = health;
var current = value > BAD ? BAD : (value < GOOD ? GOOD : value); var current = value > BAD ? BAD : (value < GOOD ? GOOD : value);
if (previous != current) if (previous != current)
{ {
logger.Info($"Service health {(previous < current ? "deteriorated" : "improved")} from {previous} to {current}."); logger.Info($"Service health {(previous < current ? "deteriorated" : "improved")} from {previous} to {current}.");
if (current == BAD)
{
recovering = false;
resume = DateTime.Now.AddSeconds(random.Next(0, THREE_MINUTES));
if (!timer.Enabled)
{
timer.Start();
}
logger.Warn($"Activating local caching and suspending transmission due to bad service health (resume: {resume:HH:mm:ss}).");
}
else if (previous == BAD)
{
recovering = true;
resume = DateTime.Now < resume ? resume : DateTime.Now.AddSeconds(random.Next(0, THREE_MINUTES));
logger.Info($"Starting recovery while maintaining local caching (resume: {resume:HH:mm:ss}).");
}
} }
return current; return current;

View file

@ -3,5 +3,6 @@
<package id="KGySoft.CoreLibraries" version="8.1.0" targetFramework="net48" /> <package id="KGySoft.CoreLibraries" version="8.1.0" targetFramework="net48" />
<package id="KGySoft.Drawing" version="8.1.0" targetFramework="net48" /> <package id="KGySoft.Drawing" version="8.1.0" targetFramework="net48" />
<package id="KGySoft.Drawing.Core" version="8.1.0" targetFramework="net48" /> <package id="KGySoft.Drawing.Core" version="8.1.0" targetFramework="net48" />
<package id="Microsoft.Web.WebView2" version="1.0.2365.46" targetFramework="net48" />
<package id="Newtonsoft.Json" version="13.0.3" targetFramework="net48" /> <package id="Newtonsoft.Json" version="13.0.3" targetFramework="net48" />
</packages> </packages>

View file

@ -42,22 +42,21 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
[TestMethod] [TestMethod]
public void Perform_MustShowDisclaimerWhenProctoringEnabled() public void Perform_MustShowDisclaimerWhenProctoringEnabled()
{ {
var count = 0; var disclaimerShown = false;
settings.Proctoring.ScreenProctoring.Enabled = true;
settings.Proctoring.Enabled = true;
sut.ActionRequired += (args) => sut.ActionRequired += (args) =>
{ {
if (args is MessageEventArgs m) if (args is MessageEventArgs m)
{ {
count++; disclaimerShown = true;
m.Result = MessageBoxResult.Ok; m.Result = MessageBoxResult.Ok;
} }
}; };
var result = sut.Perform(); var result = sut.Perform();
Assert.AreEqual(1, count); Assert.IsTrue(disclaimerShown);
Assert.AreEqual(OperationResult.Success, result); Assert.AreEqual(OperationResult.Success, result);
} }
@ -66,8 +65,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
{ {
var disclaimerShown = false; var disclaimerShown = false;
settings.Proctoring.ScreenProctoring.Enabled = true; settings.Proctoring.Enabled = true;
sut.ActionRequired += (args) => sut.ActionRequired += (args) =>
{ {
if (args is MessageEventArgs m) if (args is MessageEventArgs m)
@ -77,7 +75,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
} }
}; };
var result = sut.Repeat(); var result = sut.Perform();
Assert.IsTrue(disclaimerShown); Assert.IsTrue(disclaimerShown);
Assert.AreEqual(OperationResult.Aborted, result); Assert.AreEqual(OperationResult.Aborted, result);
@ -88,6 +86,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
{ {
var disclaimerShown = false; var disclaimerShown = false;
settings.Proctoring.Enabled = false;
sut.ActionRequired += (args) => sut.ActionRequired += (args) =>
{ {
if (args is MessageEventArgs m) if (args is MessageEventArgs m)
@ -106,22 +105,21 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
[TestMethod] [TestMethod]
public void Repeat_MustShowDisclaimerWhenProctoringEnabled() public void Repeat_MustShowDisclaimerWhenProctoringEnabled()
{ {
var count = 0; var disclaimerShown = false;
settings.Proctoring.ScreenProctoring.Enabled = true;
settings.Proctoring.Enabled = true;
sut.ActionRequired += (args) => sut.ActionRequired += (args) =>
{ {
if (args is MessageEventArgs m) if (args is MessageEventArgs m)
{ {
count++; disclaimerShown = true;
m.Result = MessageBoxResult.Ok; m.Result = MessageBoxResult.Ok;
} }
}; };
var result = sut.Perform(); var result = sut.Repeat();
Assert.AreEqual(1, count); Assert.IsTrue(disclaimerShown);
Assert.AreEqual(OperationResult.Success, result); Assert.AreEqual(OperationResult.Success, result);
} }
@ -130,8 +128,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
{ {
var disclaimerShown = false; var disclaimerShown = false;
settings.Proctoring.ScreenProctoring.Enabled = true; settings.Proctoring.Enabled = true;
sut.ActionRequired += (args) => sut.ActionRequired += (args) =>
{ {
if (args is MessageEventArgs m) if (args is MessageEventArgs m)
@ -152,6 +149,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
{ {
var disclaimerShown = false; var disclaimerShown = false;
settings.Proctoring.Enabled = false;
sut.ActionRequired += (args) => sut.ActionRequired += (args) =>
{ {
if (args is MessageEventArgs m) if (args is MessageEventArgs m)
@ -172,6 +170,7 @@ namespace SafeExamBrowser.Runtime.UnitTests.Operations
{ {
var disclaimerShown = false; var disclaimerShown = false;
settings.Proctoring.Enabled = true;
sut.ActionRequired += (args) => sut.ActionRequired += (args) =>
{ {
if (args is MessageEventArgs m) if (args is MessageEventArgs m)

View file

@ -0,0 +1,100 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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 Microsoft.VisualStudio.TestTools.UnitTesting;
using Moq;
using SafeExamBrowser.Configuration.Contracts;
using SafeExamBrowser.Core.Contracts.OperationModel;
using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Runtime.Operations;
using SafeExamBrowser.Settings;
using SafeExamBrowser.Settings.Security;
namespace SafeExamBrowser.Runtime.UnitTests.Operations
{
[TestClass]
public class ProctoringWorkaroundOperationTests
{
private SessionContext context;
private Mock<ILogger> logger;
private AppSettings settings;
private ProctoringWorkaroundOperation sut;
[TestInitialize]
public void Initialize()
{
context = new SessionContext();
logger = new Mock<ILogger>();
settings = new AppSettings();
context.Next = new SessionConfiguration();
context.Next.Settings = settings;
sut = new ProctoringWorkaroundOperation(logger.Object, context);
}
[TestMethod]
public void Perform_MustSwitchToDisableExplorerShellIfProctoringActive()
{
settings.Proctoring.Enabled = true;
settings.Security.KioskMode = KioskMode.CreateNewDesktop;
var result = sut.Perform();
Assert.AreEqual(KioskMode.DisableExplorerShell, settings.Security.KioskMode);
Assert.AreEqual(OperationResult.Success, result);
}
[TestMethod]
public void Perform_MustDoNothingIfProctoringNotActive()
{
settings.Proctoring.Enabled = false;
settings.Security.KioskMode = KioskMode.None;
var result = sut.Perform();
Assert.AreEqual(KioskMode.None, settings.Security.KioskMode);
Assert.AreEqual(OperationResult.Success, result);
}
[TestMethod]
public void Repeat_MustSwitchToDisableExplorerShellIfProctoringActive()
{
settings.Proctoring.Enabled = true;
settings.Security.KioskMode = KioskMode.CreateNewDesktop;
var result = sut.Repeat();
Assert.AreEqual(KioskMode.DisableExplorerShell, settings.Security.KioskMode);
Assert.AreEqual(OperationResult.Success, result);
}
[TestMethod]
public void Repeat_MustDoNothingIfProctoringNotActive()
{
settings.Proctoring.Enabled = false;
settings.Security.KioskMode = KioskMode.None;
var result = sut.Repeat();
Assert.AreEqual(KioskMode.None, settings.Security.KioskMode);
Assert.AreEqual(OperationResult.Success, result);
}
[TestMethod]
public void Revert_MustDoNothing()
{
settings.Proctoring.Enabled = true;
settings.Security.KioskMode = KioskMode.None;
var result = sut.Revert();
Assert.AreEqual(KioskMode.None, settings.Security.KioskMode);
Assert.AreEqual(OperationResult.Success, result);
}
}
}

View file

@ -11,10 +11,10 @@ using Moq;
using SafeExamBrowser.Configuration.Contracts; using SafeExamBrowser.Configuration.Contracts;
using SafeExamBrowser.Core.Contracts.OperationModel; using SafeExamBrowser.Core.Contracts.OperationModel;
using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts;
using SafeExamBrowser.Runtime.Operations; using SafeExamBrowser.Runtime.Operations;
using SafeExamBrowser.Runtime.Operations.Events; using SafeExamBrowser.Runtime.Operations.Events;
using SafeExamBrowser.Settings; using SafeExamBrowser.Settings;
using SafeExamBrowser.SystemComponents.Contracts;
namespace SafeExamBrowser.Runtime.UnitTests.Operations namespace SafeExamBrowser.Runtime.UnitTests.Operations
{ {

View file

@ -11,10 +11,10 @@ using Moq;
using SafeExamBrowser.Configuration.Contracts; using SafeExamBrowser.Configuration.Contracts;
using SafeExamBrowser.Core.Contracts.OperationModel; using SafeExamBrowser.Core.Contracts.OperationModel;
using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts;
using SafeExamBrowser.Runtime.Operations; using SafeExamBrowser.Runtime.Operations;
using SafeExamBrowser.Settings; using SafeExamBrowser.Settings;
using SafeExamBrowser.Settings.Security; using SafeExamBrowser.Settings.Security;
using SafeExamBrowser.SystemComponents.Contracts;
namespace SafeExamBrowser.Runtime.UnitTests.Operations namespace SafeExamBrowser.Runtime.UnitTests.Operations
{ {

View file

@ -145,6 +145,7 @@
<Compile Include="Operations\DisclaimerOperationTests.cs" /> <Compile Include="Operations\DisclaimerOperationTests.cs" />
<Compile Include="Operations\DisplayMonitorOperationTests.cs" /> <Compile Include="Operations\DisplayMonitorOperationTests.cs" />
<Compile Include="Operations\KioskModeOperationTests.cs" /> <Compile Include="Operations\KioskModeOperationTests.cs" />
<Compile Include="Operations\ProctoringWorkaroundOperationTests.cs" />
<Compile Include="Operations\RemoteSessionOperationTests.cs" /> <Compile Include="Operations\RemoteSessionOperationTests.cs" />
<Compile Include="Operations\ServerOperationTests.cs" /> <Compile Include="Operations\ServerOperationTests.cs" />
<Compile Include="Operations\ServiceOperationTests.cs" /> <Compile Include="Operations\ServiceOperationTests.cs" />

View file

@ -25,9 +25,7 @@ using SafeExamBrowser.I18n;
using SafeExamBrowser.I18n.Contracts; using SafeExamBrowser.I18n.Contracts;
using SafeExamBrowser.Logging; using SafeExamBrowser.Logging;
using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring;
using SafeExamBrowser.Monitoring.Display; using SafeExamBrowser.Monitoring.Display;
using SafeExamBrowser.Monitoring.System;
using SafeExamBrowser.Runtime.Communication; using SafeExamBrowser.Runtime.Communication;
using SafeExamBrowser.Runtime.Operations; using SafeExamBrowser.Runtime.Operations;
using SafeExamBrowser.Server; using SafeExamBrowser.Server;
@ -84,7 +82,6 @@ namespace SafeExamBrowser.Runtime
var remoteSessionDetector = new RemoteSessionDetector(ModuleLogger(nameof(RemoteSessionDetector))); var remoteSessionDetector = new RemoteSessionDetector(ModuleLogger(nameof(RemoteSessionDetector)));
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 runtimeWindow = uiFactory.CreateRuntimeWindow(appConfig); var runtimeWindow = uiFactory.CreateRuntimeWindow(appConfig);
var sentinel = new SystemSentinel(ModuleLogger(nameof(SystemSentinel)), nativeMethods, registry);
var server = new ServerProxy(appConfig, keyGenerator, ModuleLogger(nameof(ServerProxy)), systemInfo, userInfo); var server = new ServerProxy(appConfig, keyGenerator, ModuleLogger(nameof(ServerProxy)), systemInfo, userInfo);
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();
@ -104,11 +101,12 @@ namespace SafeExamBrowser.Runtime
sessionOperations.Enqueue(new VersionRestrictionOperation(logger, sessionContext, text)); sessionOperations.Enqueue(new VersionRestrictionOperation(logger, sessionContext, text));
sessionOperations.Enqueue(new DisclaimerOperation(logger, sessionContext)); sessionOperations.Enqueue(new DisclaimerOperation(logger, sessionContext));
sessionOperations.Enqueue(new RemoteSessionOperation(remoteSessionDetector, logger, sessionContext)); sessionOperations.Enqueue(new RemoteSessionOperation(remoteSessionDetector, logger, sessionContext));
sessionOperations.Enqueue(new SessionIntegrityOperation(logger, sentinel, sessionContext)); sessionOperations.Enqueue(new SessionIntegrityOperation(logger, registry, sessionContext));
sessionOperations.Enqueue(new VirtualMachineOperation(vmDetector, logger, sessionContext)); sessionOperations.Enqueue(new VirtualMachineOperation(vmDetector, logger, sessionContext));
sessionOperations.Enqueue(new DisplayMonitorOperation(displayMonitor, logger, sessionContext, text)); sessionOperations.Enqueue(new DisplayMonitorOperation(displayMonitor, logger, sessionContext, text));
sessionOperations.Enqueue(new ServiceOperation(logger, runtimeHost, serviceProxy, sessionContext, THIRTY_SECONDS, userInfo)); sessionOperations.Enqueue(new ServiceOperation(logger, runtimeHost, serviceProxy, sessionContext, 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 ProctoringWorkaroundOperation(logger, sessionContext));
sessionOperations.Enqueue(new KioskModeOperation(desktopFactory, desktopMonitor, explorerShell, logger, processFactory, sessionContext)); sessionOperations.Enqueue(new KioskModeOperation(desktopFactory, desktopMonitor, 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));
sessionOperations.Enqueue(new SessionActivationOperation(logger, sessionContext)); sessionOperations.Enqueue(new SessionActivationOperation(logger, sessionContext));

View file

@ -103,7 +103,7 @@ namespace SafeExamBrowser.Runtime.Operations
var logFilePath = $"{'"' + Convert.ToBase64String(Encoding.UTF8.GetBytes(Context.Next.AppConfig.ClientLogFilePath)) + '"'}"; var logFilePath = $"{'"' + Convert.ToBase64String(Encoding.UTF8.GetBytes(Context.Next.AppConfig.ClientLogFilePath)) + '"'}";
var logLevel = Context.Next.Settings.LogLevel.ToString(); var logLevel = Context.Next.Settings.LogLevel.ToString();
var runtimeHostUri = Context.Next.AppConfig.RuntimeAddress; var runtimeHostUri = Context.Next.AppConfig.RuntimeAddress;
var uiMode = Context.Next.Settings.UserInterface.Mode.ToString(); var uiMode = Context.Next.Settings.UserInterfaceMode.ToString();
var clientReady = false; var clientReady = false;
var clientReadyEvent = new AutoResetEvent(false); var clientReadyEvent = new AutoResetEvent(false);

View file

@ -32,13 +32,13 @@ namespace SafeExamBrowser.Runtime.Operations
protected abstract void InvokeActionRequired(ActionRequiredEventArgs args); protected abstract void InvokeActionRequired(ActionRequiredEventArgs args);
protected LoadStatus? TryLoadSettings(Uri uri, UriSource source, out PasswordParameters passwordParams, out AppSettings settings, string currentPassword = default) protected LoadStatus? TryLoadSettings(Uri uri, UriSource source, out PasswordParameters passwordParams, out AppSettings settings, string currentPassword = default(string))
{ {
passwordParams = new PasswordParameters { Password = string.Empty, IsHash = true }; passwordParams = new PasswordParameters { Password = string.Empty, IsHash = true };
var status = configuration.TryLoadSettings(uri, out settings, passwordParams); var status = configuration.TryLoadSettings(uri, out settings, passwordParams);
if (status == LoadStatus.PasswordNeeded && currentPassword != default) if (status == LoadStatus.PasswordNeeded && currentPassword != default(string))
{ {
passwordParams.Password = currentPassword; passwordParams.Password = currentPassword;
passwordParams.IsHash = true; passwordParams.IsHash = true;
@ -46,7 +46,7 @@ namespace SafeExamBrowser.Runtime.Operations
status = configuration.TryLoadSettings(uri, out settings, passwordParams); status = configuration.TryLoadSettings(uri, out settings, passwordParams);
} }
for (var attempts = 0; attempts < 5 && status == LoadStatus.PasswordNeeded; attempts++) for (int attempts = 0; attempts < 5 && status == LoadStatus.PasswordNeeded; attempts++)
{ {
var isLocalConfig = source == UriSource.AppData || source == UriSource.ProgramData; var isLocalConfig = source == UriSource.AppData || source == UriSource.ProgramData;
var purpose = isLocalConfig ? PasswordRequestPurpose.LocalSettings : PasswordRequestPurpose.Settings; var purpose = isLocalConfig ? PasswordRequestPurpose.LocalSettings : PasswordRequestPurpose.Settings;

View file

@ -164,18 +164,13 @@ namespace SafeExamBrowser.Runtime.Operations
result = OperationResult.Aborted; result = OperationResult.Aborted;
} }
if (result == OperationResult.Success && Context.Current.IsBrowserResource)
{
HandleReconfigurationByBrowserResource();
}
fileSystem.Delete(uri.LocalPath); fileSystem.Delete(uri.LocalPath);
logger.Info($"Deleted temporary configuration file '{uri}'."); logger.Info($"Deleted temporary configuration file '{uri}'.");
return result; return result;
} }
private OperationResult DetermineLoadResult(Uri uri, UriSource source, AppSettings settings, LoadStatus status, PasswordParameters passwordParams, string currentPassword = default) private OperationResult DetermineLoadResult(Uri uri, UriSource source, AppSettings settings, LoadStatus status, PasswordParameters passwordParams, string currentPassword = default(string))
{ {
var result = OperationResult.Failed; var result = OperationResult.Failed;
@ -210,7 +205,6 @@ namespace SafeExamBrowser.Runtime.Operations
private OperationResult HandleBrowserResource(Uri uri) private OperationResult HandleBrowserResource(Uri uri)
{ {
Context.Next.IsBrowserResource = true;
Context.Next.Settings.Applications.Blacklist.Clear(); Context.Next.Settings.Applications.Blacklist.Clear();
Context.Next.Settings.Applications.Whitelist.Clear(); Context.Next.Settings.Applications.Whitelist.Clear();
Context.Next.Settings.Display.AllowedDisplays = 10; Context.Next.Settings.Display.AllowedDisplays = 10;
@ -228,7 +222,7 @@ namespace SafeExamBrowser.Runtime.Operations
return OperationResult.Success; return OperationResult.Success;
} }
private OperationResult HandleClientConfiguration(Uri uri, PasswordParameters passwordParams, string currentPassword = default) private OperationResult HandleClientConfiguration(Uri uri, PasswordParameters passwordParams, string currentPassword = default(string))
{ {
var isFirstSession = Context.Current == null; var isFirstSession = Context.Current == null;
var success = TryConfigureClient(uri, passwordParams, currentPassword); var success = TryConfigureClient(uri, passwordParams, currentPassword);
@ -246,12 +240,6 @@ namespace SafeExamBrowser.Runtime.Operations
return result; return result;
} }
private void HandleReconfigurationByBrowserResource()
{
Context.Next.Settings.Browser.DeleteCookiesOnStartup = false;
logger.Info("Some browser settings were overridden in order to retain a potential LMS / web application session.");
}
private void HandleStartUrlQuery(Uri uri, UriSource source) private void HandleStartUrlQuery(Uri uri, UriSource source)
{ {
if (source == UriSource.Reconfiguration && Uri.TryCreate(Context.ReconfigurationUrl, UriKind.Absolute, out var reconfigurationUri)) if (source == UriSource.Reconfiguration && Uri.TryCreate(Context.ReconfigurationUrl, UriKind.Absolute, out var reconfigurationUri))
@ -259,13 +247,13 @@ namespace SafeExamBrowser.Runtime.Operations
uri = reconfigurationUri; uri = reconfigurationUri;
} }
if (uri != default && uri.Query.LastIndexOf('?') > 0) if (uri != default(Uri) && uri.Query.LastIndexOf('?') > 0)
{ {
Context.Next.Settings.Browser.StartUrlQuery = uri.Query.Substring(uri.Query.LastIndexOf('?')); Context.Next.Settings.Browser.StartUrlQuery = uri.Query.Substring(uri.Query.LastIndexOf('?'));
} }
} }
private bool? TryConfigureClient(Uri uri, PasswordParameters passwordParams, string currentPassword = default) private bool? TryConfigureClient(Uri uri, PasswordParameters passwordParams, string currentPassword = default(string))
{ {
var mustAuthenticate = IsRequiredToAuthenticateForClientConfiguration(passwordParams, currentPassword); var mustAuthenticate = IsRequiredToAuthenticateForClientConfiguration(passwordParams, currentPassword);
@ -316,9 +304,9 @@ namespace SafeExamBrowser.Runtime.Operations
return success; return success;
} }
private bool IsRequiredToAuthenticateForClientConfiguration(PasswordParameters passwordParams, string currentPassword = default) private bool IsRequiredToAuthenticateForClientConfiguration(PasswordParameters passwordParams, string currentPassword = default(string))
{ {
var mustAuthenticate = currentPassword != default; var mustAuthenticate = currentPassword != default(string);
if (mustAuthenticate) if (mustAuthenticate)
{ {
@ -346,7 +334,7 @@ namespace SafeExamBrowser.Runtime.Operations
{ {
var authenticated = false; var authenticated = false;
for (var attempts = 0; attempts < 5 && !authenticated; attempts++) for (int attempts = 0; attempts < 5 && !authenticated; attempts++)
{ {
var success = TryGetPassword(PasswordRequestPurpose.LocalAdministrator, out var password); var success = TryGetPassword(PasswordRequestPurpose.LocalAdministrator, out var password);
@ -396,8 +384,8 @@ namespace SafeExamBrowser.Runtime.Operations
{ {
var isValidUri = false; var isValidUri = false;
uri = default; uri = null;
source = default; source = default(UriSource);
if (commandLineArgs?.Length > 1) if (commandLineArgs?.Length > 1)
{ {

View file

@ -31,11 +31,21 @@ namespace SafeExamBrowser.Runtime.Operations
{ {
var result = OperationResult.Success; var result = OperationResult.Success;
if (Context.Next.Settings.Proctoring.ScreenProctoring.Enabled) if (Context.Next.Settings.Proctoring.JitsiMeet.Enabled)
{
result = ShowVideoProctoringDisclaimer();
}
if (result == OperationResult.Success && Context.Next.Settings.Proctoring.ScreenProctoring.Enabled)
{ {
result = ShowScreenProctoringDisclaimer(); result = ShowScreenProctoringDisclaimer();
} }
if (result == OperationResult.Success && Context.Next.Settings.Proctoring.Zoom.Enabled)
{
result = ShowZoomError();
}
return result; return result;
} }
@ -43,11 +53,21 @@ namespace SafeExamBrowser.Runtime.Operations
{ {
var result = OperationResult.Success; var result = OperationResult.Success;
if (Context.Next.Settings.Proctoring.ScreenProctoring.Enabled) if (Context.Next.Settings.Proctoring.JitsiMeet.Enabled)
{
result = ShowVideoProctoringDisclaimer();
}
if (result == OperationResult.Success && Context.Next.Settings.Proctoring.ScreenProctoring.Enabled)
{ {
result = ShowScreenProctoringDisclaimer(); result = ShowScreenProctoringDisclaimer();
} }
if (result == OperationResult.Success && Context.Next.Settings.Proctoring.Zoom.Enabled)
{
result = ShowZoomError();
}
return result; return result;
} }
@ -82,5 +102,50 @@ namespace SafeExamBrowser.Runtime.Operations
return OperationResult.Aborted; return OperationResult.Aborted;
} }
} }
private OperationResult ShowVideoProctoringDisclaimer()
{
var args = new MessageEventArgs
{
Action = MessageBoxAction.OkCancel,
Icon = MessageBoxIcon.Information,
Message = TextKey.MessageBox_VideoProctoringDisclaimer,
Title = TextKey.MessageBox_VideoProctoringDisclaimerTitle
};
StatusChanged?.Invoke(TextKey.OperationStatus_WaitDisclaimerConfirmation);
ActionRequired?.Invoke(args);
if (args.Result == MessageBoxResult.Ok)
{
logger.Info("The user confirmed the video proctoring disclaimer.");
return OperationResult.Success;
}
else
{
logger.Warn("The user did not confirm the video proctoring disclaimer! Aborting session initialization...");
return OperationResult.Aborted;
}
}
private OperationResult ShowZoomError()
{
var args = new MessageEventArgs
{
Action = MessageBoxAction.Ok,
Icon = MessageBoxIcon.Error,
Message = TextKey.MessageBox_ZoomNotSupported,
Title = TextKey.MessageBox_ZoomNotSupportedTitle
};
logger.Error("Zoom proctoring is enabled but not supported! Aborting session initialization...");
StatusChanged?.Invoke(TextKey.OperationStatus_WaitErrorConfirmation);
ActionRequired?.Invoke(args);
return OperationResult.Aborted;
}
} }
} }

View file

@ -0,0 +1,49 @@
/*
* Copyright (c) 2024 ETH Zürich, IT Services
*
* 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 SafeExamBrowser.Core.Contracts.OperationModel;
using SafeExamBrowser.Core.Contracts.OperationModel.Events;
using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Settings.Security;
namespace SafeExamBrowser.Runtime.Operations
{
internal class ProctoringWorkaroundOperation : SessionOperation
{
private readonly ILogger logger;
public override event ActionRequiredEventHandler ActionRequired { add { } remove { } }
public override event StatusChangedEventHandler StatusChanged { add { } remove { } }
public ProctoringWorkaroundOperation(ILogger logger, SessionContext context) : base(context)
{
this.logger = logger;
}
public override OperationResult Perform()
{
if (Context.Next.Settings.Proctoring.JitsiMeet.Enabled && Context.Next.Settings.Security.KioskMode == KioskMode.CreateNewDesktop)
{
Context.Next.Settings.Security.KioskMode = KioskMode.DisableExplorerShell;
logger.Info("Switched kiosk mode to Disable Explorer Shell due to remote proctoring being enabled.");
}
return OperationResult.Success;
}
public override OperationResult Repeat()
{
return Perform();
}
public override OperationResult Revert()
{
return OperationResult.Success;
}
}
}

View file

@ -10,8 +10,8 @@ using SafeExamBrowser.Core.Contracts.OperationModel;
using SafeExamBrowser.Core.Contracts.OperationModel.Events; using SafeExamBrowser.Core.Contracts.OperationModel.Events;
using SafeExamBrowser.I18n.Contracts; using SafeExamBrowser.I18n.Contracts;
using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts;
using SafeExamBrowser.Runtime.Operations.Events; using SafeExamBrowser.Runtime.Operations.Events;
using SafeExamBrowser.SystemComponents.Contracts;
using SafeExamBrowser.UserInterface.Contracts.MessageBox; using SafeExamBrowser.UserInterface.Contracts.MessageBox;
namespace SafeExamBrowser.Runtime.Operations namespace SafeExamBrowser.Runtime.Operations

View file

@ -169,7 +169,6 @@ namespace SafeExamBrowser.Runtime.Operations
if (status == LoadStatus.Success) if (status == LoadStatus.Success)
{ {
var browserSettings = Context.Next.Settings.Browser;
var serverSettings = Context.Next.Settings.Server; var serverSettings = Context.Next.Settings.Server;
Context.Next.AppConfig.ServerApi = info.Api; Context.Next.AppConfig.ServerApi = info.Api;
@ -179,7 +178,6 @@ namespace SafeExamBrowser.Runtime.Operations
Context.Next.Settings = settings; Context.Next.Settings = settings;
Context.Next.Settings.Browser.StartUrl = exam.Url; Context.Next.Settings.Browser.StartUrl = exam.Url;
Context.Next.Settings.Browser.StartUrlQuery = browserSettings.StartUrlQuery;
Context.Next.Settings.Server = serverSettings; Context.Next.Settings.Server = serverSettings;
Context.Next.Settings.SessionMode = SessionMode.Server; Context.Next.Settings.SessionMode = SessionMode.Server;

View file

@ -6,26 +6,31 @@
* file, You can obtain one at http://mozilla.org/MPL/2.0/. * file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/ */
using System;
using System.Linq;
using SafeExamBrowser.Core.Contracts.OperationModel; using SafeExamBrowser.Core.Contracts.OperationModel;
using SafeExamBrowser.Core.Contracts.OperationModel.Events; using SafeExamBrowser.Core.Contracts.OperationModel.Events;
using SafeExamBrowser.I18n.Contracts; using SafeExamBrowser.I18n.Contracts;
using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts.System; using SafeExamBrowser.SystemComponents.Contracts.Registry;
namespace SafeExamBrowser.Runtime.Operations namespace SafeExamBrowser.Runtime.Operations
{ {
internal class SessionIntegrityOperation : SessionOperation internal class SessionIntegrityOperation : SessionOperation
{ {
private static readonly string USER_PATH = $@"{Environment.ExpandEnvironmentVariables("%LocalAppData%")}\Microsoft\Windows\Cursors\";
private static readonly string SYSTEM_PATH = $@"{Environment.ExpandEnvironmentVariables("%SystemRoot%")}\Cursors\";
private readonly ILogger logger; private readonly ILogger logger;
private readonly ISystemSentinel sentinel; private readonly IRegistry registry;
public override event ActionRequiredEventHandler ActionRequired { add { } remove { } } public override event ActionRequiredEventHandler ActionRequired { add { } remove { } }
public override event StatusChangedEventHandler StatusChanged; public override event StatusChangedEventHandler StatusChanged;
public SessionIntegrityOperation(ILogger logger, ISystemSentinel sentinel, SessionContext context) : base(context) public SessionIntegrityOperation(ILogger logger, IRegistry registry, SessionContext context) : base(context)
{ {
this.logger = logger; this.logger = logger;
this.sentinel = sentinel; this.registry = registry;
} }
public override OperationResult Perform() public override OperationResult Perform()
@ -34,11 +39,8 @@ namespace SafeExamBrowser.Runtime.Operations
StatusChanged?.Invoke(TextKey.OperationStatus_VerifySessionIntegrity); StatusChanged?.Invoke(TextKey.OperationStatus_VerifySessionIntegrity);
success &= InitializeStickyKeys(); success &= VerifyCursorConfiguration();
success &= VerifyCursors(); success &= VerifyEaseOfAccessConfiguration();
success &= VerifyEaseOfAccess();
LogResult(success);
return success ? OperationResult.Success : OperationResult.Failed; return success ? OperationResult.Success : OperationResult.Failed;
} }
@ -49,60 +51,43 @@ namespace SafeExamBrowser.Runtime.Operations
StatusChanged?.Invoke(TextKey.OperationStatus_VerifySessionIntegrity); StatusChanged?.Invoke(TextKey.OperationStatus_VerifySessionIntegrity);
success &= InitializeStickyKeys(); success &= VerifyCursorConfiguration();
success &= VerifyCursors(); success &= VerifyEaseOfAccessConfiguration();
success &= VerifyEaseOfAccess();
LogResult(success);
return success ? OperationResult.Success : OperationResult.Failed; return success ? OperationResult.Success : OperationResult.Failed;
} }
public override OperationResult Revert() public override OperationResult Revert()
{ {
FinalizeStickyKeys();
return OperationResult.Success; return OperationResult.Success;
} }
private void FinalizeStickyKeys() private bool VerifyCursorConfiguration()
{
sentinel.RevertStickyKeys();
}
private bool InitializeStickyKeys()
{
var success = true;
sentinel.RevertStickyKeys();
if (!Context.Next.Settings.Security.AllowStickyKeys)
{
success = sentinel.DisableStickyKeys();
}
return success;
}
private void LogResult(bool success)
{
if (success)
{
logger.Info("Successfully ensured session integrity.");
}
else
{
logger.Error("Failed to ensure session integrity! Aborting session initialization...");
}
}
private bool VerifyCursors()
{ {
var success = true; var success = true;
if (Context.Next.Settings.Security.VerifyCursorConfiguration) if (Context.Next.Settings.Security.VerifyCursorConfiguration)
{ {
success = sentinel.VerifyCursors(); logger.Info($"Attempting to verify cursor configuration...");
success = registry.TryGetNames(RegistryValue.UserHive.Cursors_Key, out var cursors);
if (success)
{
foreach (var cursor in cursors.Where(c => !string.IsNullOrWhiteSpace(c)))
{
success &= VerifyCursor(cursor);
}
}
if (success)
{
logger.Info("Cursor configuration successfully verified.");
}
else
{
logger.Warn("Failed to verify cursor configuration or configuration is compromised! Aborting session initialization...");
}
} }
else else
{ {
@ -112,22 +97,65 @@ namespace SafeExamBrowser.Runtime.Operations
return success; return success;
} }
private bool VerifyEaseOfAccess() private bool VerifyCursor(string cursor)
{ {
var success = sentinel.VerifyEaseOfAccess(); var success = true;
success &= registry.TryRead(RegistryValue.UserHive.Cursors_Key, cursor, out var value);
success &= !(value is string) || (value is string path && (string.IsNullOrWhiteSpace(path) || IsValidCursorPath(path)));
if (!success) if (!success)
{ {
if (Context.Current?.Settings.Service.IgnoreService == false) if (value != default)
{
logger.Warn($"Configuration of cursor '{cursor}' is compromised: '{value}'!");
}
else
{
logger.Warn($"Failed to verify configuration of cursor '{cursor}'!");
}
}
return success;
}
private bool IsValidCursorPath(string path)
{
return path.StartsWith(USER_PATH, StringComparison.OrdinalIgnoreCase) || path.StartsWith(SYSTEM_PATH, StringComparison.OrdinalIgnoreCase);
}
private bool VerifyEaseOfAccessConfiguration()
{
var success = false;
logger.Info($"Attempting to verify ease of access configuration...");
if (registry.TryRead(RegistryValue.MachineHive.EaseOfAccess_Key, RegistryValue.MachineHive.EaseOfAccess_Name, out var value))
{
if (value is string s && string.IsNullOrWhiteSpace(s))
{ {
logger.Info($"Ease of access configuration is compromised but service was active in the current session.");
success = true; success = true;
logger.Info("Ease of access configuration successfully verified.");
} }
else if (!Context.Next.Settings.Service.IgnoreService) else if (!Context.Next.Settings.Service.IgnoreService)
{ {
logger.Info($"Ease of access configuration is compromised but service will be active in the next session.");
success = true; success = true;
logger.Info($"Ease of access configuration is compromised ('{value}'), but service will be active in the next session.");
} }
else if (Context.Current?.Settings.Service.IgnoreService == false)
{
success = true;
logger.Info($"Ease of access configuration is set ('{value}'), but service was active in the current session.");
}
else
{
logger.Warn($"Ease of access configuration is compromised: '{value}'! Aborting session initialization...");
}
}
else
{
success = true;
logger.Info("Ease of access configuration successfully verified (value does not exist).");
} }
return success; return success;

View file

@ -10,17 +10,17 @@ using SafeExamBrowser.Core.Contracts.OperationModel;
using SafeExamBrowser.Core.Contracts.OperationModel.Events; using SafeExamBrowser.Core.Contracts.OperationModel.Events;
using SafeExamBrowser.I18n.Contracts; using SafeExamBrowser.I18n.Contracts;
using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Logging.Contracts;
using SafeExamBrowser.Monitoring.Contracts;
using SafeExamBrowser.Runtime.Operations.Events; using SafeExamBrowser.Runtime.Operations.Events;
using SafeExamBrowser.Settings.Security; using SafeExamBrowser.Settings.Security;
using SafeExamBrowser.SystemComponents.Contracts;
using SafeExamBrowser.UserInterface.Contracts.MessageBox; using SafeExamBrowser.UserInterface.Contracts.MessageBox;
namespace SafeExamBrowser.Runtime.Operations namespace SafeExamBrowser.Runtime.Operations
{ {
internal class VirtualMachineOperation : SessionOperation internal class VirtualMachineOperation : SessionOperation
{ {
private readonly IVirtualMachineDetector detector; private IVirtualMachineDetector detector;
private readonly ILogger logger; private ILogger logger;
public override event ActionRequiredEventHandler ActionRequired; public override event ActionRequiredEventHandler ActionRequired;
public override event StatusChangedEventHandler StatusChanged; public override event StatusChangedEventHandler StatusChanged;

View file

@ -109,6 +109,7 @@
<Compile Include="Operations\ApplicationIntegrityOperation.cs" /> <Compile Include="Operations\ApplicationIntegrityOperation.cs" />
<Compile Include="Operations\Events\VersionRestrictionMessageArgs.cs" /> <Compile Include="Operations\Events\VersionRestrictionMessageArgs.cs" />
<Compile Include="Operations\KioskModeOperation.cs" /> <Compile Include="Operations\KioskModeOperation.cs" />
<Compile Include="Operations\ProctoringWorkaroundOperation.cs" />
<Compile Include="Operations\RemoteSessionOperation.cs" /> <Compile Include="Operations\RemoteSessionOperation.cs" />
<Compile Include="Operations\ServerOperation.cs" /> <Compile Include="Operations\ServerOperation.cs" />
<Compile Include="Operations\ServiceOperation.cs" /> <Compile Include="Operations\ServiceOperation.cs" />

View file

@ -8,7 +8,6 @@
using System; using System;
using System.Net.Http; using System.Net.Http;
using System.Text;
using SafeExamBrowser.Settings.Logging; using SafeExamBrowser.Settings.Logging;
namespace SafeExamBrowser.Server namespace SafeExamBrowser.Server
@ -17,7 +16,7 @@ namespace SafeExamBrowser.Server
{ {
internal static string ToLogString(this HttpResponseMessage response) internal static string ToLogString(this HttpResponseMessage response)
{ {
return response == default ? "No Response" : $"{(int) response.StatusCode} {response.StatusCode} {response.ReasonPhrase}"; return $"{(int?) response?.StatusCode} {response?.StatusCode} {response?.ReasonPhrase}";
} }
internal static string ToLogType(this LogLevel severity) internal static string ToLogType(this LogLevel severity)
@ -37,19 +36,6 @@ namespace SafeExamBrowser.Server
return "UNKNOWN"; return "UNKNOWN";
} }
internal static string ToSummary(this Exception exception)
{
var trimChars = new[] { '.', '!' };
var summary = new StringBuilder(exception.Message?.TrimEnd(trimChars));
for (var inner = exception.InnerException; inner != default; inner = inner.InnerException)
{
summary.Append($" -> {inner.Message?.TrimEnd(trimChars)}");
}
return summary.ToString();
}
internal static long ToUnixTimestamp(this DateTime date) internal static long ToUnixTimestamp(this DateTime date)
{ {
return new DateTimeOffset(date).ToUnixTimeMilliseconds(); return new DateTimeOffset(date).ToUnixTimeMilliseconds();

View file

@ -26,8 +26,6 @@ namespace SafeExamBrowser.Server.Requests
private readonly HttpClient httpClient; private readonly HttpClient httpClient;
private bool hadException;
protected readonly ApiVersion1 api; protected readonly ApiVersion1 api;
protected readonly ILogger logger; protected readonly ILogger logger;
protected readonly Parser parser; protected readonly Parser parser;
@ -75,7 +73,7 @@ namespace SafeExamBrowser.Server.Requests
{ {
response = httpClient.SendAsync(request).GetAwaiter().GetResult(); response = httpClient.SendAsync(request).GetAwaiter().GetResult();
if (PerformLoggingFor(request)) if (request.RequestUri.AbsolutePath != api.LogEndpoint && request.RequestUri.AbsolutePath != api.PingEndpoint)
{ {
logger.Debug($"Completed request: {request.Method} '{request.RequestUri}' -> {response.ToLogString()}"); logger.Debug($"Completed request: {request.Method} '{request.RequestUri}' -> {response.ToLogString()}");
} }
@ -92,19 +90,12 @@ namespace SafeExamBrowser.Server.Requests
} }
catch (TaskCanceledException) catch (TaskCanceledException)
{ {
if (PerformLoggingFor(request)) logger.Debug($"Request {request.Method} '{request.RequestUri}' did not complete within {settings.RequestTimeout}ms!");
{
logger.Warn($"Request {request.Method} '{request.RequestUri}' did not complete within {settings.RequestTimeout}ms!");
}
break; break;
} }
catch (Exception e) catch (Exception e)
{ {
if (PerformLoggingFor(request) && IsFirstException()) logger.Debug($"Request {request.Method} '{request.RequestUri}' failed due to {e}");
{
logger.Warn($"Request {request.Method} '{request.RequestUri}' has failed: {e.ToSummary()}!");
}
} }
} }
@ -177,20 +168,6 @@ namespace SafeExamBrowser.Server.Requests
return request; return request;
} }
private bool IsFirstException()
{
var isFirst = !hadException;
hadException = true;
return isFirst;
}
private bool PerformLoggingFor(HttpRequestMessage request)
{
return request.RequestUri.AbsolutePath != api.LogEndpoint && request.RequestUri.AbsolutePath != api.PingEndpoint;
}
private (string name, string value)[] UpdateOAuth2Token((string name, string value)[] headers) private (string name, string value)[] UpdateOAuth2Token((string name, string value)[] headers)
{ {
var result = new List<(string name, string value)>(); var result = new List<(string name, string value)>();

View file

@ -23,7 +23,8 @@ namespace SafeExamBrowser.Server.Requests
internal bool TryExecute(Exam exam, out string message, out string appSignatureKeySalt, out string browserExamKey) internal bool TryExecute(Exam exam, out string message, out string appSignatureKeySalt, out string browserExamKey)
{ {
var content = $"examId={exam.Id}"; var content = $"examId={exam.Id}";
var success = TryExecute(HttpMethod.Put, api.HandshakeEndpoint, out var response, content, ContentType.URL_ENCODED, Authorization, Token); var method = new HttpMethod("PATCH");
var success = TryExecute(method, api.HandshakeEndpoint, out var response, content, ContentType.URL_ENCODED, Authorization, Token);
appSignatureKeySalt = default; appSignatureKeySalt = default;
browserExamKey = default; browserExamKey = default;

Some files were not shown because too many files have changed in this diff Show more