SEBSP-23: Implemented scaffolding for network redundancy.
This commit is contained in:
parent
731a748552
commit
a213ec0f7d
13 changed files with 739 additions and 196 deletions
|
@ -39,7 +39,7 @@ namespace SafeExamBrowser.Proctoring
|
||||||
internal abstract void Stop();
|
internal abstract void Stop();
|
||||||
internal abstract void Terminate();
|
internal abstract void Terminate();
|
||||||
|
|
||||||
protected abstract void ActivateNotification();
|
protected virtual void ActivateNotification() { }
|
||||||
protected abstract void TerminateNotification();
|
protected virtual void TerminateNotification() { }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -94,23 +94,27 @@
|
||||||
<Compile Include="ScreenProctoring\Data\IntervalTrigger.cs" />
|
<Compile Include="ScreenProctoring\Data\IntervalTrigger.cs" />
|
||||||
<Compile Include="ScreenProctoring\Data\KeyboardTrigger.cs" />
|
<Compile Include="ScreenProctoring\Data\KeyboardTrigger.cs" />
|
||||||
<Compile Include="ScreenProctoring\Data\MouseTrigger.cs" />
|
<Compile Include="ScreenProctoring\Data\MouseTrigger.cs" />
|
||||||
|
<Compile Include="ScreenProctoring\Events\DataCollectedEventHandler.cs" />
|
||||||
<Compile Include="ScreenProctoring\Imaging\Extensions.cs" />
|
<Compile Include="ScreenProctoring\Imaging\Extensions.cs" />
|
||||||
<Compile Include="ScreenProctoring\Data\Metadata.cs" />
|
<Compile Include="ScreenProctoring\Data\Metadata.cs" />
|
||||||
<Compile Include="ScreenProctoring\Imaging\ProcessingOrder.cs" />
|
<Compile Include="ScreenProctoring\Imaging\ProcessingOrder.cs" />
|
||||||
<Compile Include="ScreenProctoring\Imaging\ScreenShot.cs" />
|
<Compile Include="ScreenProctoring\Imaging\ScreenShot.cs" />
|
||||||
<Compile Include="ScreenProctoring\ScreenProctoringImplementation.cs" />
|
<Compile Include="ScreenProctoring\ScreenProctoringImplementation.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Api.cs" />
|
<Compile Include="ScreenProctoring\Service\Api.cs" />
|
||||||
|
<Compile Include="ScreenProctoring\DataCollector.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Parser.cs" />
|
<Compile Include="ScreenProctoring\Service\Parser.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Requests\ContentType.cs" />
|
<Compile Include="ScreenProctoring\Service\Requests\ContentType.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Requests\CreateSessionRequest.cs" />
|
<Compile Include="ScreenProctoring\Service\Requests\CreateSessionRequest.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Requests\Header.cs" />
|
<Compile Include="ScreenProctoring\Service\Requests\Header.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Requests\Extensions.cs" />
|
<Compile Include="ScreenProctoring\Service\Requests\Extensions.cs" />
|
||||||
|
<Compile Include="ScreenProctoring\Service\Requests\HealthRequest.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Requests\TerminateSessionRequest.cs" />
|
<Compile Include="ScreenProctoring\Service\Requests\TerminateSessionRequest.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Requests\OAuth2TokenRequest.cs" />
|
<Compile Include="ScreenProctoring\Service\Requests\OAuth2TokenRequest.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Requests\Request.cs" />
|
<Compile Include="ScreenProctoring\Service\Requests\Request.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\Requests\ScreenShotRequest.cs" />
|
<Compile Include="ScreenProctoring\Service\Requests\ScreenShotRequest.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\ServiceProxy.cs" />
|
<Compile Include="ScreenProctoring\Service\ServiceProxy.cs" />
|
||||||
<Compile Include="ScreenProctoring\Service\ServiceResponse.cs" />
|
<Compile Include="ScreenProctoring\Service\ServiceResponse.cs" />
|
||||||
|
<Compile Include="ScreenProctoring\TransmissionSpooler.cs" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="..\SafeExamBrowser.Applications.Contracts\SafeExamBrowser.Applications.Contracts.csproj">
|
<ProjectReference Include="..\SafeExamBrowser.Applications.Contracts\SafeExamBrowser.Applications.Contracts.csproj">
|
||||||
|
|
|
@ -7,7 +7,6 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
|
@ -28,14 +27,16 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
|
||||||
|
|
||||||
internal string ApplicationInfo { get; private set; }
|
internal string ApplicationInfo { get; private set; }
|
||||||
internal string BrowserInfo { get; private set; }
|
internal string BrowserInfo { get; private set; }
|
||||||
|
internal TimeSpan Elapsed { get; private set; }
|
||||||
internal string TriggerInfo { get; private set; }
|
internal string TriggerInfo { get; private set; }
|
||||||
internal string Urls { get; private set; }
|
internal string Urls { get; private set; }
|
||||||
internal string WindowTitle { get; private set; }
|
internal string WindowTitle { get; private set; }
|
||||||
|
|
||||||
internal Metadata(IApplicationMonitor applicationMonitor, IBrowserApplication browser, ILogger logger)
|
internal Metadata(IApplicationMonitor applicationMonitor, IBrowserApplication browser, TimeSpan elapsed, ILogger logger)
|
||||||
{
|
{
|
||||||
this.applicationMonitor = applicationMonitor;
|
this.applicationMonitor = applicationMonitor;
|
||||||
this.browser = browser;
|
this.browser = browser;
|
||||||
|
this.Elapsed = elapsed;
|
||||||
this.logger = logger;
|
this.logger = logger;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -57,6 +58,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
|
||||||
CaptureMouseTrigger(mouse);
|
CaptureMouseTrigger(mouse);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: Can only log URLs when allowed by policy in browser configuration!
|
||||||
logger.Debug($"Captured metadata: {ApplicationInfo} / {BrowserInfo} / {TriggerInfo} / {Urls} / {WindowTitle}.");
|
logger.Debug($"Captured metadata: {ApplicationInfo} / {BrowserInfo} / {TriggerInfo} / {Urls} / {WindowTitle}.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -79,7 +81,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
|
||||||
if (applicationMonitor.TryGetActiveApplication(out var application))
|
if (applicationMonitor.TryGetActiveApplication(out var application))
|
||||||
{
|
{
|
||||||
ApplicationInfo = BuildApplicationInfo(application);
|
ApplicationInfo = BuildApplicationInfo(application);
|
||||||
WindowTitle = BuildWindowTitle(application);
|
WindowTitle = string.IsNullOrEmpty(application.Window.Title) ? "-" : application.Window.Title;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -88,46 +90,12 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private string BuildApplicationInfo(ActiveApplication application)
|
|
||||||
{
|
|
||||||
var info = new StringBuilder();
|
|
||||||
|
|
||||||
info.Append(application.Process.Name);
|
|
||||||
|
|
||||||
if (application.Process.OriginalName != default)
|
|
||||||
{
|
|
||||||
info.Append($" ({application.Process.OriginalName}{(application.Process.Signature == default ? ")" : "")}");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (application.Process.Signature != default)
|
|
||||||
{
|
|
||||||
info.Append($"{(application.Process.OriginalName == default ? "(" : ", ")}{application.Process.Signature})");
|
|
||||||
}
|
|
||||||
|
|
||||||
return info.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private string BuildWindowTitle(ActiveApplication application)
|
|
||||||
{
|
|
||||||
return string.IsNullOrEmpty(application.Window.Title) ? "-" : application.Window.Title;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void CaptureBrowserData()
|
private void CaptureBrowserData()
|
||||||
{
|
{
|
||||||
var windows = browser.GetWindows();
|
var windows = browser.GetWindows();
|
||||||
|
|
||||||
BrowserInfo = BuildBrowserInfo(windows);
|
BrowserInfo = string.Join(", ", windows.Select(w => $"{(w.IsMainWindow ? "Main" : "Additional")} Window: {w.Title} ({w.Url})"));
|
||||||
Urls = BuildUrls(windows);
|
Urls = string.Join(", ", windows.Select(w => w.Url));
|
||||||
}
|
|
||||||
|
|
||||||
private string BuildUrls(IEnumerable<IBrowserWindow> windows)
|
|
||||||
{
|
|
||||||
return string.Join(", ", windows.Select(w => w.Url));
|
|
||||||
}
|
|
||||||
|
|
||||||
private string BuildBrowserInfo(IEnumerable<IBrowserWindow> windows)
|
|
||||||
{
|
|
||||||
return string.Join(", ", windows.Select(w => $"{(w.IsMainWindow ? "Main" : "Additional")} Window: {w.Title} ({w.Url})"));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void CaptureIntervalTrigger(IntervalTrigger interval)
|
private void CaptureIntervalTrigger(IntervalTrigger interval)
|
||||||
|
@ -154,5 +122,24 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
|
||||||
TriggerInfo = $"{mouse.Button} mouse button has been {mouse.State.ToString().ToLower()} at ({mouse.Info.X}/{mouse.Info.Y}).";
|
TriggerInfo = $"{mouse.Button} mouse button has been {mouse.State.ToString().ToLower()} at ({mouse.Info.X}/{mouse.Info.Y}).";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private string BuildApplicationInfo(ActiveApplication application)
|
||||||
|
{
|
||||||
|
var info = new StringBuilder();
|
||||||
|
|
||||||
|
info.Append(application.Process.Name);
|
||||||
|
|
||||||
|
if (application.Process.OriginalName != default)
|
||||||
|
{
|
||||||
|
info.Append($" ({application.Process.OriginalName}{(application.Process.Signature == default ? ")" : "")}");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (application.Process.Signature != default)
|
||||||
|
{
|
||||||
|
info.Append($"{(application.Process.OriginalName == default ? "(" : ", ")}{application.Process.Signature})");
|
||||||
|
}
|
||||||
|
|
||||||
|
return info.ToString();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
176
SafeExamBrowser.Proctoring/ScreenProctoring/DataCollector.cs
Normal file
176
SafeExamBrowser.Proctoring/ScreenProctoring/DataCollector.cs
Normal file
|
@ -0,0 +1,176 @@
|
||||||
|
/*
|
||||||
|
* 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;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Timers;
|
||||||
|
using System.Windows.Input;
|
||||||
|
using SafeExamBrowser.Browser.Contracts;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
||||||
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Data;
|
||||||
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Events;
|
||||||
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Imaging;
|
||||||
|
using SafeExamBrowser.Settings.Proctoring;
|
||||||
|
using SafeExamBrowser.WindowsApi.Contracts;
|
||||||
|
using SafeExamBrowser.WindowsApi.Contracts.Events;
|
||||||
|
using MouseButton = SafeExamBrowser.WindowsApi.Contracts.Events.MouseButton;
|
||||||
|
using MouseButtonState = SafeExamBrowser.WindowsApi.Contracts.Events.MouseButtonState;
|
||||||
|
using Timer = System.Timers.Timer;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
||||||
|
{
|
||||||
|
internal class DataCollector
|
||||||
|
{
|
||||||
|
private readonly object @lock = new object();
|
||||||
|
|
||||||
|
private readonly IApplicationMonitor applicationMonitor;
|
||||||
|
private readonly IBrowserApplication browser;
|
||||||
|
private readonly IModuleLogger logger;
|
||||||
|
private readonly INativeMethods nativeMethods;
|
||||||
|
private readonly ScreenProctoringSettings settings;
|
||||||
|
private readonly Timer timer;
|
||||||
|
|
||||||
|
private DateTime last;
|
||||||
|
private Guid? keyboardHookId;
|
||||||
|
private Guid? mouseHookId;
|
||||||
|
|
||||||
|
internal event DataCollectedEventHandler DataCollected;
|
||||||
|
|
||||||
|
internal DataCollector(
|
||||||
|
IApplicationMonitor applicationMonitor,
|
||||||
|
IBrowserApplication browser,
|
||||||
|
IModuleLogger logger,
|
||||||
|
INativeMethods nativeMethods,
|
||||||
|
ScreenProctoringSettings settings)
|
||||||
|
{
|
||||||
|
this.applicationMonitor = applicationMonitor;
|
||||||
|
this.browser = browser;
|
||||||
|
this.logger = logger;
|
||||||
|
this.nativeMethods = nativeMethods;
|
||||||
|
this.settings = settings;
|
||||||
|
this.timer = new Timer();
|
||||||
|
}
|
||||||
|
|
||||||
|
internal void Start()
|
||||||
|
{
|
||||||
|
last = DateTime.Now;
|
||||||
|
|
||||||
|
keyboardHookId = nativeMethods.RegisterKeyboardHook(KeyboardHookCallback);
|
||||||
|
mouseHookId = nativeMethods.RegisterMouseHook(MouseHookCallback);
|
||||||
|
|
||||||
|
timer.AutoReset = false;
|
||||||
|
timer.Elapsed += MaxIntervalElapsed;
|
||||||
|
timer.Interval = settings.MaxInterval;
|
||||||
|
timer.Start();
|
||||||
|
|
||||||
|
logger.Debug("Started.");
|
||||||
|
}
|
||||||
|
|
||||||
|
internal void Stop()
|
||||||
|
{
|
||||||
|
last = DateTime.Now;
|
||||||
|
|
||||||
|
if (keyboardHookId.HasValue)
|
||||||
|
{
|
||||||
|
nativeMethods.DeregisterKeyboardHook(keyboardHookId.Value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mouseHookId.HasValue)
|
||||||
|
{
|
||||||
|
nativeMethods.DeregisterMouseHook(mouseHookId.Value);
|
||||||
|
}
|
||||||
|
|
||||||
|
keyboardHookId = default;
|
||||||
|
mouseHookId = default;
|
||||||
|
|
||||||
|
timer.Elapsed -= MaxIntervalElapsed;
|
||||||
|
timer.Stop();
|
||||||
|
|
||||||
|
logger.Debug("Stopped.");
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool KeyboardHookCallback(int keyCode, KeyModifier modifier, KeyState state)
|
||||||
|
{
|
||||||
|
var trigger = new KeyboardTrigger
|
||||||
|
{
|
||||||
|
Key = KeyInterop.KeyFromVirtualKey(keyCode),
|
||||||
|
Modifier = modifier,
|
||||||
|
State = state
|
||||||
|
};
|
||||||
|
|
||||||
|
TryCollect(keyboard: trigger);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void MaxIntervalElapsed(object sender, ElapsedEventArgs args)
|
||||||
|
{
|
||||||
|
var trigger = new IntervalTrigger
|
||||||
|
{
|
||||||
|
ConfigurationValue = settings.MaxInterval,
|
||||||
|
TimeElapsed = Convert.ToInt32(DateTime.Now.Subtract(last).TotalMilliseconds)
|
||||||
|
};
|
||||||
|
|
||||||
|
TryCollect(interval: trigger);
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool MouseHookCallback(MouseButton button, MouseButtonState state, MouseInformation info)
|
||||||
|
{
|
||||||
|
var trigger = new MouseTrigger
|
||||||
|
{
|
||||||
|
Button = button,
|
||||||
|
Info = info,
|
||||||
|
State = state
|
||||||
|
};
|
||||||
|
|
||||||
|
TryCollect(mouse: trigger);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void TryCollect(IntervalTrigger interval = default, KeyboardTrigger keyboard = default, MouseTrigger mouse = default)
|
||||||
|
{
|
||||||
|
if (MinIntervalElapsed() && Monitor.TryEnter(@lock))
|
||||||
|
{
|
||||||
|
var elapsed = DateTime.Now.Subtract(last);
|
||||||
|
|
||||||
|
last = DateTime.Now;
|
||||||
|
timer.Stop();
|
||||||
|
|
||||||
|
Task.Run(() =>
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var metadata = new Metadata(applicationMonitor, browser, elapsed, logger.CloneFor(nameof(Metadata)));
|
||||||
|
var screenShot = new ScreenShot(logger.CloneFor(nameof(ScreenShot)), settings);
|
||||||
|
|
||||||
|
metadata.Capture(interval, keyboard, mouse);
|
||||||
|
screenShot.Take();
|
||||||
|
screenShot.Compress();
|
||||||
|
|
||||||
|
DataCollected?.Invoke(metadata, screenShot);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
logger.Error("Failed to execute data collection!", e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
timer.Start();
|
||||||
|
Monitor.Exit(@lock);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool MinIntervalElapsed()
|
||||||
|
{
|
||||||
|
return DateTime.Now.Subtract(last) >= new TimeSpan(0, 0, 0, 0, settings.MinInterval);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,15 @@
|
||||||
|
/*
|
||||||
|
* 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 SafeExamBrowser.Proctoring.ScreenProctoring.Data;
|
||||||
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Imaging;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Proctoring.ScreenProctoring.Events
|
||||||
|
{
|
||||||
|
internal delegate void DataCollectedEventHandler(Metadata metadata, ScreenShot screenShot);
|
||||||
|
}
|
|
@ -7,10 +7,6 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Threading;
|
|
||||||
using System.Threading.Tasks;
|
|
||||||
using System.Timers;
|
|
||||||
using System.Windows.Input;
|
|
||||||
using SafeExamBrowser.Browser.Contracts;
|
using SafeExamBrowser.Browser.Contracts;
|
||||||
using SafeExamBrowser.Core.Contracts.Notifications.Events;
|
using SafeExamBrowser.Core.Contracts.Notifications.Events;
|
||||||
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
||||||
|
@ -23,29 +19,17 @@ using SafeExamBrowser.Proctoring.ScreenProctoring.Service;
|
||||||
using SafeExamBrowser.Server.Contracts.Events.Proctoring;
|
using SafeExamBrowser.Server.Contracts.Events.Proctoring;
|
||||||
using SafeExamBrowser.Settings.Proctoring;
|
using SafeExamBrowser.Settings.Proctoring;
|
||||||
using SafeExamBrowser.WindowsApi.Contracts;
|
using SafeExamBrowser.WindowsApi.Contracts;
|
||||||
using SafeExamBrowser.WindowsApi.Contracts.Events;
|
|
||||||
using MouseButton = SafeExamBrowser.WindowsApi.Contracts.Events.MouseButton;
|
|
||||||
using MouseButtonState = SafeExamBrowser.WindowsApi.Contracts.Events.MouseButtonState;
|
|
||||||
using Timer = System.Timers.Timer;
|
|
||||||
|
|
||||||
namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
||||||
{
|
{
|
||||||
internal class ScreenProctoringImplementation : ProctoringImplementation
|
internal class ScreenProctoringImplementation : ProctoringImplementation
|
||||||
{
|
{
|
||||||
private readonly object @lock = new object();
|
private readonly DataCollector collector;
|
||||||
|
|
||||||
private readonly IApplicationMonitor applicationMonitor;
|
|
||||||
private readonly IBrowserApplication browser;
|
|
||||||
private readonly IModuleLogger logger;
|
private readonly IModuleLogger logger;
|
||||||
private readonly INativeMethods nativeMethods;
|
|
||||||
private readonly ServiceProxy service;
|
private readonly ServiceProxy service;
|
||||||
private readonly ScreenProctoringSettings settings;
|
private readonly ScreenProctoringSettings settings;
|
||||||
|
private readonly TransmissionSpooler spooler;
|
||||||
private readonly IText text;
|
private readonly IText text;
|
||||||
private readonly Timer timer;
|
|
||||||
|
|
||||||
private DateTime last;
|
|
||||||
private Guid? keyboardHookId;
|
|
||||||
private Guid? mouseHookId;
|
|
||||||
|
|
||||||
internal override string Name => nameof(ScreenProctoring);
|
internal override string Name => nameof(ScreenProctoring);
|
||||||
|
|
||||||
|
@ -60,14 +44,12 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
||||||
ProctoringSettings settings,
|
ProctoringSettings settings,
|
||||||
IText text)
|
IText text)
|
||||||
{
|
{
|
||||||
this.applicationMonitor = applicationMonitor;
|
this.collector = new DataCollector(applicationMonitor, browser, logger.CloneFor(nameof(DataCollector)), nativeMethods, settings.ScreenProctoring);
|
||||||
this.browser = browser;
|
|
||||||
this.logger = logger;
|
this.logger = logger;
|
||||||
this.nativeMethods = nativeMethods;
|
|
||||||
this.service = service;
|
this.service = service;
|
||||||
this.settings = settings.ScreenProctoring;
|
this.settings = settings.ScreenProctoring;
|
||||||
|
this.spooler = new TransmissionSpooler(logger.CloneFor(nameof(TransmissionSpooler)), service);
|
||||||
this.text = text;
|
this.text = text;
|
||||||
this.timer = new Timer();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
internal override void Initialize()
|
internal override void Initialize()
|
||||||
|
@ -79,9 +61,6 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
||||||
start &= !string.IsNullOrWhiteSpace(settings.GroupId);
|
start &= !string.IsNullOrWhiteSpace(settings.GroupId);
|
||||||
start &= !string.IsNullOrWhiteSpace(settings.ServiceUrl);
|
start &= !string.IsNullOrWhiteSpace(settings.ServiceUrl);
|
||||||
|
|
||||||
timer.AutoReset = false;
|
|
||||||
timer.Interval = settings.MaxInterval;
|
|
||||||
|
|
||||||
if (start)
|
if (start)
|
||||||
{
|
{
|
||||||
logger.Info($"Initialized proctoring: All settings are valid, starting automatically...");
|
logger.Info($"Initialized proctoring: All settings are valid, starting automatically...");
|
||||||
|
@ -91,7 +70,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ShowNotificationInactive();
|
UpdateNotification(false);
|
||||||
logger.Info($"Initialized proctoring: Not all settings are valid or a server session is active, not starting automatically.");
|
logger.Info($"Initialized proctoring: Not all settings are valid or a server session is active, not starting automatically.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -125,60 +104,43 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
||||||
logger.Info("Successfully processed instruction.");
|
logger.Info("Successfully processed instruction.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal override void Start()
|
internal override void Start()
|
||||||
{
|
{
|
||||||
last = DateTime.Now;
|
collector.DataCollected += Collector_DataCollected;
|
||||||
keyboardHookId = nativeMethods.RegisterKeyboardHook(KeyboardHookCallback);
|
collector.Start();
|
||||||
mouseHookId = nativeMethods.RegisterMouseHook(MouseHookCallback);
|
spooler.Start();
|
||||||
|
|
||||||
timer.Elapsed += Timer_Elapsed;
|
UpdateNotification(true);
|
||||||
timer.Start();
|
|
||||||
|
|
||||||
ShowNotificationActive();
|
|
||||||
|
|
||||||
logger.Info($"Started proctoring.");
|
logger.Info($"Started proctoring.");
|
||||||
}
|
}
|
||||||
|
|
||||||
internal override void Stop()
|
internal override void Stop()
|
||||||
{
|
{
|
||||||
if (keyboardHookId.HasValue)
|
collector.Stop();
|
||||||
{
|
collector.DataCollected -= Collector_DataCollected;
|
||||||
nativeMethods.DeregisterKeyboardHook(keyboardHookId.Value);
|
spooler.Stop();
|
||||||
}
|
|
||||||
|
|
||||||
if (mouseHookId.HasValue)
|
TerminateSession();
|
||||||
{
|
UpdateNotification(false);
|
||||||
nativeMethods.DeregisterMouseHook(mouseHookId.Value);
|
|
||||||
}
|
|
||||||
|
|
||||||
keyboardHookId = default;
|
|
||||||
mouseHookId = default;
|
|
||||||
|
|
||||||
timer.Elapsed -= Timer_Elapsed;
|
|
||||||
timer.Stop();
|
|
||||||
|
|
||||||
TerminateServiceSession();
|
|
||||||
ShowNotificationInactive();
|
|
||||||
|
|
||||||
logger.Info("Stopped proctoring.");
|
logger.Info("Stopped proctoring.");
|
||||||
}
|
}
|
||||||
|
|
||||||
internal override void Terminate()
|
internal override void Terminate()
|
||||||
{
|
{
|
||||||
|
// TODO: Cache transmission or user information!
|
||||||
|
|
||||||
Stop();
|
Stop();
|
||||||
TerminateNotification();
|
TerminateNotification();
|
||||||
|
|
||||||
logger.Info("Terminated proctoring.");
|
logger.Info("Terminated proctoring.");
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void ActivateNotification()
|
private void Collector_DataCollected(Metadata metadata, ScreenShot screenShot)
|
||||||
{
|
{
|
||||||
// Nothing to do here for now...
|
spooler.Add(metadata, screenShot);
|
||||||
}
|
|
||||||
|
|
||||||
protected override void TerminateNotification()
|
|
||||||
{
|
|
||||||
// Nothing to do here for now...
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void Connect(string sessionId = default)
|
private void Connect(string sessionId = default)
|
||||||
|
@ -201,52 +163,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool KeyboardHookCallback(int keyCode, KeyModifier modifier, KeyState state)
|
private void TerminateSession()
|
||||||
{
|
|
||||||
var trigger = new KeyboardTrigger
|
|
||||||
{
|
|
||||||
Key = KeyInterop.KeyFromVirtualKey(keyCode),
|
|
||||||
Modifier = modifier,
|
|
||||||
State = state
|
|
||||||
};
|
|
||||||
|
|
||||||
TryExecute(keyboard: trigger);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private bool MouseHookCallback(MouseButton button, MouseButtonState state, MouseInformation info)
|
|
||||||
{
|
|
||||||
var trigger = new MouseTrigger
|
|
||||||
{
|
|
||||||
Button = button,
|
|
||||||
Info = info,
|
|
||||||
State = state
|
|
||||||
};
|
|
||||||
|
|
||||||
TryExecute(mouse: trigger);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void ShowNotificationActive()
|
|
||||||
{
|
|
||||||
// TODO: Replace with actual icon!
|
|
||||||
// TODO: Extend INotification with IsEnabled or CanActivate, as the screen proctoring notification does not have any action or window!
|
|
||||||
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ProctoringNotification_Active.xaml") };
|
|
||||||
Tooltip = text.Get(TextKey.Notification_ProctoringActiveTooltip);
|
|
||||||
NotificationChanged?.Invoke();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void ShowNotificationInactive()
|
|
||||||
{
|
|
||||||
// TODO: Replace with actual icon!
|
|
||||||
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ProctoringNotification_Inactive.xaml") };
|
|
||||||
Tooltip = text.Get(TextKey.Notification_ProctoringInactiveTooltip);
|
|
||||||
NotificationChanged?.Invoke();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void TerminateServiceSession()
|
|
||||||
{
|
{
|
||||||
if (service.IsConnected)
|
if (service.IsConnected)
|
||||||
{
|
{
|
||||||
|
@ -255,60 +172,24 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void Timer_Elapsed(object sender, ElapsedEventArgs args)
|
private void UpdateNotification(bool live)
|
||||||
{
|
{
|
||||||
var trigger = new IntervalTrigger
|
// TODO: Replace with actual icon!
|
||||||
|
// TODO: Extend INotification with IsEnabled or CanActivate!
|
||||||
|
// TODO: Service health, HD space and caching indicators!
|
||||||
|
|
||||||
|
if (live)
|
||||||
{
|
{
|
||||||
ConfigurationValue = settings.MaxInterval,
|
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ProctoringNotification_Active.xaml") };
|
||||||
TimeElapsed = Convert.ToInt32(DateTime.Now.Subtract(last).TotalMilliseconds)
|
Tooltip = text.Get(TextKey.Notification_ProctoringActiveTooltip);
|
||||||
};
|
}
|
||||||
|
else
|
||||||
TryExecute(interval: trigger);
|
{
|
||||||
}
|
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ProctoringNotification_Inactive.xaml") };
|
||||||
|
Tooltip = text.Get(TextKey.Notification_ProctoringInactiveTooltip);
|
||||||
private void TryExecute(IntervalTrigger interval = default, KeyboardTrigger keyboard = default, MouseTrigger mouse = default)
|
|
||||||
{
|
|
||||||
if (MinimumIntervalElapsed() && Monitor.TryEnter(@lock))
|
|
||||||
{
|
|
||||||
last = DateTime.Now;
|
|
||||||
timer.Stop();
|
|
||||||
|
|
||||||
Task.Run(() =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var metadata = new Metadata(applicationMonitor, browser, logger.CloneFor(nameof(Metadata)));
|
|
||||||
|
|
||||||
using (var screenShot = new ScreenShot(logger.CloneFor(nameof(ScreenShot)), settings))
|
|
||||||
{
|
|
||||||
metadata.Capture(interval, keyboard, mouse);
|
|
||||||
screenShot.Take();
|
|
||||||
screenShot.Compress();
|
|
||||||
|
|
||||||
if (service.IsConnected)
|
|
||||||
{
|
|
||||||
service.Send(metadata, screenShot);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
logger.Warn("Cannot send screen shot as service is disconnected!");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
logger.Error("Failed to execute capturing and/or transmission!", e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
timer.Start();
|
|
||||||
Monitor.Exit(@lock);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
private bool MinimumIntervalElapsed()
|
NotificationChanged?.Invoke();
|
||||||
{
|
|
||||||
return DateTime.Now.Subtract(last) >= new TimeSpan(0, 0, 0, 0, settings.MinInterval);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,12 +13,14 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service
|
||||||
internal const string SESSION_ID = "%%_SESSION_ID_%%";
|
internal const string SESSION_ID = "%%_SESSION_ID_%%";
|
||||||
|
|
||||||
internal string AccessTokenEndpoint { get; set; }
|
internal string AccessTokenEndpoint { get; set; }
|
||||||
|
internal string HealthEndpoint { get; set; }
|
||||||
internal string ScreenShotEndpoint { get; set; }
|
internal string ScreenShotEndpoint { get; set; }
|
||||||
internal string SessionEndpoint { get; set; }
|
internal string SessionEndpoint { get; set; }
|
||||||
|
|
||||||
internal Api()
|
internal Api()
|
||||||
{
|
{
|
||||||
AccessTokenEndpoint = "/oauth/token";
|
AccessTokenEndpoint = "/oauth/token";
|
||||||
|
HealthEndpoint = "/health";
|
||||||
ScreenShotEndpoint = $"/seb-api/v1/session/{SESSION_ID}/screenshot";
|
ScreenShotEndpoint = $"/seb-api/v1/session/{SESSION_ID}/screenshot";
|
||||||
SessionEndpoint = "/seb-api/v1/session";
|
SessionEndpoint = "/seb-api/v1/session";
|
||||||
}
|
}
|
||||||
|
|
|
@ -46,6 +46,20 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service
|
||||||
return isExpired;
|
return isExpired;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal bool TryParseHealth(HttpResponseMessage response, out int health)
|
||||||
|
{
|
||||||
|
var success = false;
|
||||||
|
|
||||||
|
health = default;
|
||||||
|
|
||||||
|
if (response.Headers.TryGetValues(Header.HEALTH, out var values))
|
||||||
|
{
|
||||||
|
success = int.TryParse(values.First(), out health);
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
internal bool TryParseOauth2Token(HttpContent content, out string oauth2Token)
|
internal bool TryParseOauth2Token(HttpContent content, out string oauth2Token)
|
||||||
{
|
{
|
||||||
oauth2Token = default;
|
oauth2Token = default;
|
||||||
|
|
|
@ -13,6 +13,7 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
|
||||||
internal const string ACCEPT = "Accept";
|
internal const string ACCEPT = "Accept";
|
||||||
internal const string AUTHORIZATION = "Authorization";
|
internal const string AUTHORIZATION = "Authorization";
|
||||||
internal const string GROUP_ID = "SEB_GROUP_UUID";
|
internal const string GROUP_ID = "SEB_GROUP_UUID";
|
||||||
|
internal const string HEALTH = "sps_server_health";
|
||||||
internal const string IMAGE_FORMAT = "imageFormat";
|
internal const string IMAGE_FORMAT = "imageFormat";
|
||||||
internal const string METADATA = "metaData";
|
internal const string METADATA = "metaData";
|
||||||
internal const string SESSION_ID = "SEB_SESSION_UUID";
|
internal const string SESSION_ID = "SEB_SESSION_UUID";
|
||||||
|
|
|
@ -0,0 +1,36 @@
|
||||||
|
/*
|
||||||
|
* 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.Net.Http;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
|
||||||
|
{
|
||||||
|
internal class HealthRequest : Request
|
||||||
|
{
|
||||||
|
internal HealthRequest(Api api, HttpClient httpClient, ILogger logger, Parser parser) : base(api, httpClient, logger, parser)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(out int health, out string message)
|
||||||
|
{
|
||||||
|
var url = api.HealthEndpoint;
|
||||||
|
var success = TryExecute(HttpMethod.Get, url, out var response);
|
||||||
|
|
||||||
|
health = default;
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
parser.TryParseHealth(response, out health);
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -23,11 +23,11 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service.Requests
|
||||||
|
|
||||||
internal bool TryExecute(Metadata metadata, ScreenShot screenShot, string sessionId, out string message)
|
internal bool TryExecute(Metadata metadata, ScreenShot screenShot, string sessionId, out string message)
|
||||||
{
|
{
|
||||||
var data = (Header.METADATA, metadata.ToJson());
|
|
||||||
var imageFormat = (Header.IMAGE_FORMAT, ToString(screenShot.Format));
|
var imageFormat = (Header.IMAGE_FORMAT, ToString(screenShot.Format));
|
||||||
|
var metdataJson = (Header.METADATA, metadata.ToJson());
|
||||||
var timestamp = (Header.TIMESTAMP, DateTime.Now.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, data, imageFormat, timestamp);
|
var success = TryExecute(HttpMethod.Post, url, out var response, screenShot.Data, ContentType.OCTET_STREAM, Authorization, imageFormat, metdataJson, timestamp);
|
||||||
|
|
||||||
message = response.ToLogString();
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
|
|
@ -70,6 +70,23 @@ namespace SafeExamBrowser.Proctoring.ScreenProctoring.Service
|
||||||
return new ServiceResponse(success, message);
|
return new ServiceResponse(success, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal ServiceResponse<int> GetHealth()
|
||||||
|
{
|
||||||
|
var request = new HealthRequest(api, httpClient, logger, parser);
|
||||||
|
var success = request.TryExecute(out var health, out var message);
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
logger.Info($"Successfully queried health (value: {health}).");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
logger.Error("Failed to query health!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return new ServiceResponse<int>(success, health, message);
|
||||||
|
}
|
||||||
|
|
||||||
internal ServiceResponse Send(Metadata metadata, ScreenShot screenShot)
|
internal ServiceResponse Send(Metadata metadata, ScreenShot screenShot)
|
||||||
{
|
{
|
||||||
var request = new ScreenShotRequest(api, httpClient, logger, parser);
|
var request = new ScreenShotRequest(api, httpClient, logger, parser);
|
||||||
|
|
|
@ -0,0 +1,410 @@
|
||||||
|
/*
|
||||||
|
* 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;
|
||||||
|
using System.Collections.Concurrent;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Timers;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Data;
|
||||||
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Imaging;
|
||||||
|
using SafeExamBrowser.Proctoring.ScreenProctoring.Service;
|
||||||
|
using Timer = System.Timers.Timer;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Proctoring.ScreenProctoring
|
||||||
|
{
|
||||||
|
internal class TransmissionSpooler
|
||||||
|
{
|
||||||
|
const int BAD = 10;
|
||||||
|
const int GOOD = 0;
|
||||||
|
|
||||||
|
private readonly ILogger logger;
|
||||||
|
private readonly ConcurrentQueue<(Metadata metadata, ScreenShot screenShot)> queue;
|
||||||
|
private readonly Random random;
|
||||||
|
private readonly ServiceProxy service;
|
||||||
|
private readonly Timer timer;
|
||||||
|
|
||||||
|
private Queue<(Metadata metadata, DateTime schedule, ScreenShot screenShot)> buffer;
|
||||||
|
private int health;
|
||||||
|
private bool recovering;
|
||||||
|
private DateTime resume;
|
||||||
|
private Thread thread;
|
||||||
|
private CancellationTokenSource token;
|
||||||
|
|
||||||
|
internal TransmissionSpooler(ILogger logger, ServiceProxy service)
|
||||||
|
{
|
||||||
|
this.buffer = new Queue<(Metadata, DateTime, ScreenShot)>();
|
||||||
|
this.logger = logger;
|
||||||
|
this.queue = new ConcurrentQueue<(Metadata, ScreenShot)>();
|
||||||
|
this.random = new Random();
|
||||||
|
this.service = service;
|
||||||
|
this.timer = new Timer();
|
||||||
|
}
|
||||||
|
|
||||||
|
internal void Add(Metadata metadata, ScreenShot screenShot)
|
||||||
|
{
|
||||||
|
queue.Enqueue((metadata, screenShot));
|
||||||
|
}
|
||||||
|
|
||||||
|
internal void Start()
|
||||||
|
{
|
||||||
|
const int FIFTEEN_SECONDS = 15000;
|
||||||
|
|
||||||
|
logger.Debug("Starting...");
|
||||||
|
|
||||||
|
health = GOOD;
|
||||||
|
recovering = false;
|
||||||
|
resume = default;
|
||||||
|
token = new CancellationTokenSource();
|
||||||
|
|
||||||
|
thread = new Thread(Execute);
|
||||||
|
thread.IsBackground = true;
|
||||||
|
thread.Start();
|
||||||
|
|
||||||
|
timer.AutoReset = false;
|
||||||
|
timer.Elapsed += Timer_Elapsed;
|
||||||
|
timer.Interval = 2000;
|
||||||
|
// TODO: Revert!
|
||||||
|
// timer.Interval = FIFTEEN_SECONDS;
|
||||||
|
timer.Start();
|
||||||
|
}
|
||||||
|
|
||||||
|
internal void Stop()
|
||||||
|
{
|
||||||
|
const int TEN_SECONDS = 10000;
|
||||||
|
|
||||||
|
if (thread != default)
|
||||||
|
{
|
||||||
|
logger.Debug("Stopping...");
|
||||||
|
|
||||||
|
timer.Elapsed -= Timer_Elapsed;
|
||||||
|
timer.Stop();
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
token.Cancel();
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
logger.Error("Failed to initiate cancellation!", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var success = thread.Join(TEN_SECONDS);
|
||||||
|
|
||||||
|
if (!success)
|
||||||
|
{
|
||||||
|
thread.Abort();
|
||||||
|
logger.Warn($"Aborted since stopping gracefully within {TEN_SECONDS / 1000:N0} seconds failed!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
logger.Error("Failed to stop!", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
resume = default;
|
||||||
|
thread = default;
|
||||||
|
token = default;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void Execute()
|
||||||
|
{
|
||||||
|
logger.Debug("Ready.");
|
||||||
|
|
||||||
|
while (!token.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
if (health == BAD)
|
||||||
|
{
|
||||||
|
ExecuteCacheOnly();
|
||||||
|
}
|
||||||
|
else if (recovering)
|
||||||
|
{
|
||||||
|
ExecuteRecovery();
|
||||||
|
}
|
||||||
|
else if (health == GOOD)
|
||||||
|
{
|
||||||
|
ExecuteNormally();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ExecuteDeferred();
|
||||||
|
}
|
||||||
|
|
||||||
|
Thread.Sleep(50);
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.Debug("Stopped.");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ExecuteCacheOnly()
|
||||||
|
{
|
||||||
|
const int THREE_MINUTES = 180;
|
||||||
|
|
||||||
|
if (!recovering)
|
||||||
|
{
|
||||||
|
recovering = true;
|
||||||
|
resume = DateTime.Now.AddSeconds(random.Next(0, THREE_MINUTES));
|
||||||
|
|
||||||
|
logger.Warn($"Activating local caching and suspending transmission due to bad service health (value: {health}, resume: {resume:HH:mm:ss}).");
|
||||||
|
}
|
||||||
|
|
||||||
|
CacheBuffer();
|
||||||
|
CacheFromQueue();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ExecuteDeferred()
|
||||||
|
{
|
||||||
|
Schedule(health);
|
||||||
|
|
||||||
|
if (TryPeekFromBuffer(out _, out var schedule, out _) && schedule <= DateTime.Now)
|
||||||
|
{
|
||||||
|
TryTransmitFromBuffer();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ExecuteNormally()
|
||||||
|
{
|
||||||
|
TryTransmitFromBuffer();
|
||||||
|
TryTransmitFromCache();
|
||||||
|
TryTransmitFromQueue();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ExecuteRecovery()
|
||||||
|
{
|
||||||
|
CacheFromQueue();
|
||||||
|
recovering = DateTime.Now < resume;
|
||||||
|
|
||||||
|
if (!recovering)
|
||||||
|
{
|
||||||
|
logger.Info($"Deactivating local caching and resuming transmission due to improved service health (value: {health}).");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void Buffer(Metadata metadata, DateTime schedule, ScreenShot screenShot)
|
||||||
|
{
|
||||||
|
buffer.Enqueue((metadata, schedule, screenShot));
|
||||||
|
buffer = new Queue<(Metadata, DateTime, ScreenShot)>(buffer.OrderBy((b) => b.schedule));
|
||||||
|
|
||||||
|
// TODO: Remove!
|
||||||
|
PrintBuffer();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void PrintBuffer()
|
||||||
|
{
|
||||||
|
logger.Log("-------------------------------------------------------------------------------------------------------");
|
||||||
|
logger.Info($"Buffer: {buffer.Count}");
|
||||||
|
|
||||||
|
foreach (var (m, t, s) in buffer)
|
||||||
|
{
|
||||||
|
logger.Log($"\t\t{t} ({m.Elapsed} {s.Data.Length})");
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.Log("-------------------------------------------------------------------------------------------------------");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void CacheBuffer()
|
||||||
|
{
|
||||||
|
foreach (var (metadata, _, screenShot) in buffer)
|
||||||
|
{
|
||||||
|
using (screenShot)
|
||||||
|
{
|
||||||
|
Cache(metadata, screenShot);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Revert!
|
||||||
|
// buffer.Clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void CacheFromQueue()
|
||||||
|
{
|
||||||
|
if (TryDequeue(out var metadata, out var screenShot))
|
||||||
|
{
|
||||||
|
using (screenShot)
|
||||||
|
{
|
||||||
|
Cache(metadata, screenShot);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void Cache(Metadata metadata, ScreenShot screenShot)
|
||||||
|
{
|
||||||
|
// TODO: Implement caching!
|
||||||
|
//var directory = Dispatcher.Invoke(() => OutputPath.Text);
|
||||||
|
//var extension = screenShot.Format.ToString().ToLower();
|
||||||
|
//var path = Path.Combine(directory, $"{DateTime.Now:HH\\hmm\\mss\\sfff\\m\\s}.{extension}");
|
||||||
|
|
||||||
|
//if (!Directory.Exists(directory))
|
||||||
|
//{
|
||||||
|
// Directory.CreateDirectory(directory);
|
||||||
|
// logger.Debug($"Created local output directory '{directory}'.");
|
||||||
|
//}
|
||||||
|
|
||||||
|
//File.WriteAllBytes(path, screenShot.Data);
|
||||||
|
//logger.Debug($"Screen shot saved as '{path}'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void Schedule(int health)
|
||||||
|
{
|
||||||
|
if (TryDequeue(out var metadata, out var screenShot))
|
||||||
|
{
|
||||||
|
var schedule = DateTime.Now.AddMilliseconds((health + 1) * metadata.Elapsed.TotalMilliseconds);
|
||||||
|
|
||||||
|
Buffer(metadata, schedule, screenShot);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool TryDequeue(out Metadata metadata, out ScreenShot screenShot)
|
||||||
|
{
|
||||||
|
metadata = default;
|
||||||
|
screenShot = default;
|
||||||
|
|
||||||
|
if (queue.TryDequeue(out var item))
|
||||||
|
{
|
||||||
|
metadata = item.metadata;
|
||||||
|
screenShot = item.screenShot;
|
||||||
|
}
|
||||||
|
|
||||||
|
return metadata != default && screenShot != default;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool TryPeekFromBuffer(out Metadata metadata, out DateTime schedule, out ScreenShot screenShot)
|
||||||
|
{
|
||||||
|
metadata = default;
|
||||||
|
schedule = default;
|
||||||
|
screenShot = default;
|
||||||
|
|
||||||
|
if (buffer.Any())
|
||||||
|
{
|
||||||
|
(metadata, schedule, screenShot) = buffer.Peek();
|
||||||
|
}
|
||||||
|
|
||||||
|
return metadata != default && screenShot != default;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool TryTransmitFromBuffer()
|
||||||
|
{
|
||||||
|
var success = false;
|
||||||
|
|
||||||
|
if (TryPeekFromBuffer(out var metadata, out _, out var screenShot))
|
||||||
|
{
|
||||||
|
// TODO: Exception after sending of screenshot, most likely due to concurrent disposal!!
|
||||||
|
success = TryTransmit(metadata, screenShot);
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
buffer.Dequeue();
|
||||||
|
screenShot.Dispose();
|
||||||
|
|
||||||
|
// TODO: Revert!
|
||||||
|
PrintBuffer();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool TryTransmitFromCache()
|
||||||
|
{
|
||||||
|
var success = false;
|
||||||
|
|
||||||
|
// TODO: Implement transmission from cache!
|
||||||
|
//if (Cache.Any())
|
||||||
|
//{
|
||||||
|
//
|
||||||
|
//}
|
||||||
|
//else
|
||||||
|
//{
|
||||||
|
// success = true;
|
||||||
|
//}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool TryTransmitFromQueue()
|
||||||
|
{
|
||||||
|
var success = false;
|
||||||
|
|
||||||
|
if (TryDequeue(out var metadata, out var screenShot))
|
||||||
|
{
|
||||||
|
success = TryTransmit(metadata, screenShot);
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
screenShot.Dispose();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Buffer(metadata, DateTime.Now, screenShot);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool TryTransmit(Metadata metadata, ScreenShot screenShot)
|
||||||
|
{
|
||||||
|
var success = false;
|
||||||
|
|
||||||
|
if (service.IsConnected)
|
||||||
|
{
|
||||||
|
success = service.Send(metadata, screenShot).Success;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
logger.Warn("Cannot send screen shot as service is disconnected!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
private readonly Random temp = new Random();
|
||||||
|
|
||||||
|
private void Timer_Elapsed(object sender, ElapsedEventArgs e)
|
||||||
|
{
|
||||||
|
// TODO: Revert!
|
||||||
|
//if (service.IsConnected)
|
||||||
|
//{
|
||||||
|
// var response = service.GetHealth();
|
||||||
|
|
||||||
|
// if (response.Success)
|
||||||
|
// {
|
||||||
|
// var previous = health;
|
||||||
|
|
||||||
|
// health = response.Value > BAD ? BAD : (response.Value < GOOD ? GOOD : response.Value);
|
||||||
|
|
||||||
|
// if (previous != health)
|
||||||
|
// {
|
||||||
|
// logger.Info($"Service health {(previous < health ? "deteriorated" : "improved")} from {previous} to {health}.");
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//}
|
||||||
|
//else
|
||||||
|
//{
|
||||||
|
// logger.Warn("Cannot query health as service is disconnected!");
|
||||||
|
//}
|
||||||
|
|
||||||
|
var previous = health;
|
||||||
|
|
||||||
|
health += temp.Next(-3, 5);
|
||||||
|
health = health < GOOD ? GOOD : (health > BAD ? BAD : health);
|
||||||
|
|
||||||
|
if (previous != health)
|
||||||
|
{
|
||||||
|
logger.Info($"Service health {(previous < health ? "deteriorated" : "improved")} from {previous} to {health}.");
|
||||||
|
}
|
||||||
|
|
||||||
|
timer.Start();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue