2024-02-01 17:36:11 +01:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
using System;
|
2024-02-13 11:04:36 +01:00
|
|
|
|
using SafeExamBrowser.Browser.Contracts;
|
2024-02-21 18:37:23 +01:00
|
|
|
|
using SafeExamBrowser.Configuration.Contracts;
|
2024-02-01 17:36:11 +01:00
|
|
|
|
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
|
|
|
|
using SafeExamBrowser.I18n.Contracts;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2024-02-13 11:04:36 +01:00
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
|
|
|
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Data;
|
2024-02-01 17:36:11 +01:00
|
|
|
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Imaging;
|
|
|
|
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Service;
|
|
|
|
|
using SafeExamBrowser.Server.Contracts.Events.Proctoring;
|
|
|
|
|
using SafeExamBrowser.Settings.Proctoring;
|
2024-02-06 10:45:45 +01:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts;
|
2024-02-01 17:36:11 +01:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
|
|
|
|
{
|
|
|
|
|
internal class ScreenProctoringImplementation : ProctoringImplementation
|
|
|
|
|
{
|
2024-02-16 19:42:41 +01:00
|
|
|
|
private readonly DataCollector collector;
|
2024-02-01 17:36:11 +01:00
|
|
|
|
private readonly IModuleLogger logger;
|
|
|
|
|
private readonly ServiceProxy service;
|
|
|
|
|
private readonly ScreenProctoringSettings settings;
|
2024-02-16 19:42:41 +01:00
|
|
|
|
private readonly TransmissionSpooler spooler;
|
2024-02-01 17:36:11 +01:00
|
|
|
|
private readonly IText text;
|
2024-02-06 10:45:45 +01:00
|
|
|
|
|
2024-02-01 17:36:11 +01:00
|
|
|
|
internal override string Name => nameof(ScreenProctoring);
|
|
|
|
|
|
2024-02-06 10:45:45 +01:00
|
|
|
|
internal ScreenProctoringImplementation(
|
2024-02-21 18:37:23 +01:00
|
|
|
|
AppConfig appConfig,
|
2024-02-13 11:04:36 +01:00
|
|
|
|
IApplicationMonitor applicationMonitor,
|
|
|
|
|
IBrowserApplication browser,
|
2024-02-06 10:45:45 +01:00
|
|
|
|
IModuleLogger logger,
|
|
|
|
|
INativeMethods nativeMethods,
|
|
|
|
|
ServiceProxy service,
|
|
|
|
|
ProctoringSettings settings,
|
|
|
|
|
IText text)
|
2024-02-01 17:36:11 +01:00
|
|
|
|
{
|
2024-02-16 19:42:41 +01:00
|
|
|
|
this.collector = new DataCollector(applicationMonitor, browser, logger.CloneFor(nameof(DataCollector)), nativeMethods, settings.ScreenProctoring);
|
2024-02-01 17:36:11 +01:00
|
|
|
|
this.logger = logger;
|
|
|
|
|
this.service = service;
|
|
|
|
|
this.settings = settings.ScreenProctoring;
|
2024-02-21 18:37:23 +01:00
|
|
|
|
this.spooler = new TransmissionSpooler(appConfig, logger.CloneFor(nameof(TransmissionSpooler)), service);
|
2024-02-01 17:36:11 +01:00
|
|
|
|
this.text = text;
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-29 21:05:43 +01:00
|
|
|
|
internal override void ExecuteRemainingWork()
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Starting execution of remaining work...");
|
|
|
|
|
spooler.ExecuteRemainingWork(InvokeRemainingWorkUpdated);
|
|
|
|
|
logger.Info("Terminated execution of remaining work.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal override bool HasRemainingWork()
|
|
|
|
|
{
|
|
|
|
|
var hasWork = spooler.HasRemainingWork();
|
|
|
|
|
|
|
|
|
|
if (hasWork)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("There is remaining work to be done.");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Info("There is no remaining work to be done.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return hasWork;
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-01 17:36:11 +01:00
|
|
|
|
internal override void Initialize()
|
|
|
|
|
{
|
|
|
|
|
var start = true;
|
|
|
|
|
|
|
|
|
|
start &= !string.IsNullOrWhiteSpace(settings.ClientId);
|
|
|
|
|
start &= !string.IsNullOrWhiteSpace(settings.ClientSecret);
|
|
|
|
|
start &= !string.IsNullOrWhiteSpace(settings.GroupId);
|
|
|
|
|
start &= !string.IsNullOrWhiteSpace(settings.ServiceUrl);
|
|
|
|
|
|
|
|
|
|
if (start)
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"Initialized proctoring: All settings are valid, starting automatically...");
|
|
|
|
|
|
|
|
|
|
Connect();
|
|
|
|
|
Start();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2024-02-16 19:42:41 +01:00
|
|
|
|
UpdateNotification(false);
|
2024-02-01 17:36:11 +01:00
|
|
|
|
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)
|
|
|
|
|
{
|
|
|
|
|
// Nothing to do here for now...
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal override void ProctoringInstructionReceived(InstructionEventArgs args)
|
|
|
|
|
{
|
|
|
|
|
if (args is ScreenProctoringInstruction instruction)
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"Proctoring instruction received: {instruction.Method}.");
|
|
|
|
|
|
|
|
|
|
if (instruction.Method == InstructionMethod.Join)
|
|
|
|
|
{
|
|
|
|
|
settings.ClientId = instruction.ClientId;
|
|
|
|
|
settings.ClientSecret = instruction.ClientSecret;
|
|
|
|
|
settings.GroupId = instruction.GroupId;
|
|
|
|
|
settings.ServiceUrl = instruction.ServiceUrl;
|
|
|
|
|
|
|
|
|
|
Connect(instruction.SessionId);
|
|
|
|
|
Start();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Stop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
logger.Info("Successfully processed instruction.");
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-02-16 19:42:41 +01:00
|
|
|
|
|
2024-02-01 17:36:11 +01:00
|
|
|
|
internal override void Start()
|
|
|
|
|
{
|
2024-02-16 19:42:41 +01:00
|
|
|
|
collector.DataCollected += Collector_DataCollected;
|
|
|
|
|
collector.Start();
|
|
|
|
|
spooler.Start();
|
2024-02-06 10:45:45 +01:00
|
|
|
|
|
2024-02-16 19:42:41 +01:00
|
|
|
|
UpdateNotification(true);
|
2024-02-01 17:36:11 +01:00
|
|
|
|
|
|
|
|
|
logger.Info($"Started proctoring.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal override void Stop()
|
|
|
|
|
{
|
2024-02-16 19:42:41 +01:00
|
|
|
|
collector.Stop();
|
|
|
|
|
collector.DataCollected -= Collector_DataCollected;
|
|
|
|
|
spooler.Stop();
|
2024-02-06 10:45:45 +01:00
|
|
|
|
|
2024-02-16 19:42:41 +01:00
|
|
|
|
TerminateSession();
|
|
|
|
|
UpdateNotification(false);
|
2024-02-01 17:36:11 +01:00
|
|
|
|
|
|
|
|
|
logger.Info("Stopped proctoring.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal override void Terminate()
|
|
|
|
|
{
|
2024-02-06 10:45:45 +01:00
|
|
|
|
Stop();
|
2024-02-01 17:36:11 +01:00
|
|
|
|
|
|
|
|
|
logger.Info("Terminated proctoring.");
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-21 18:37:23 +01:00
|
|
|
|
private void Collector_DataCollected(MetaData metaData, ScreenShot screenShot)
|
2024-02-01 17:36:11 +01:00
|
|
|
|
{
|
2024-02-21 18:37:23 +01:00
|
|
|
|
spooler.Add(metaData, screenShot);
|
2024-02-01 17:36:11 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Connect(string sessionId = default)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Connecting to service...");
|
|
|
|
|
|
|
|
|
|
var connect = service.Connect(settings.ServiceUrl);
|
|
|
|
|
|
|
|
|
|
if (connect.Success)
|
|
|
|
|
{
|
|
|
|
|
if (sessionId == default)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Creating session...");
|
|
|
|
|
service.CreateSession(settings.GroupId);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
service.SessionId = sessionId;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-16 19:42:41 +01:00
|
|
|
|
private void TerminateSession()
|
2024-02-01 17:36:11 +01:00
|
|
|
|
{
|
|
|
|
|
if (service.IsConnected)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Terminating session...");
|
|
|
|
|
service.TerminateSession();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-16 19:42:41 +01:00
|
|
|
|
private void UpdateNotification(bool live)
|
2024-02-01 17:36:11 +01:00
|
|
|
|
{
|
2024-02-23 18:32:44 +01:00
|
|
|
|
CanActivate = false;
|
2024-02-06 10:45:45 +01:00
|
|
|
|
|
2024-02-16 19:42:41 +01:00
|
|
|
|
if (live)
|
2024-02-01 17:36:11 +01:00
|
|
|
|
{
|
2024-02-23 18:32:44 +01:00
|
|
|
|
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ScreenProctoring_Active.xaml") };
|
2024-02-16 19:42:41 +01:00
|
|
|
|
Tooltip = text.Get(TextKey.Notification_ProctoringActiveTooltip);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2024-02-23 18:32:44 +01:00
|
|
|
|
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ScreenProctoring_Inactive.xaml") };
|
2024-02-16 19:42:41 +01:00
|
|
|
|
Tooltip = text.Get(TextKey.Notification_ProctoringInactiveTooltip);
|
2024-02-01 17:36:11 +01:00
|
|
|
|
}
|
|
|
|
|
|
2024-02-29 21:05:43 +01:00
|
|
|
|
InvokeNotificationChanged();
|
2024-02-01 17:36:11 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|