2024-02-21 18:37:23 +01:00
|
|
|
|
/*
|
2024-03-05 18:37:42 +01:00
|
|
|
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
2024-02-21 18:37:23 +01:00
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using SafeExamBrowser.Browser.Contracts;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
2024-02-22 18:04:00 +01:00
|
|
|
|
using SafeExamBrowser.Settings.Proctoring;
|
2024-02-21 18:37:23 +01:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts.Events;
|
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Proctoring.ScreenProctoring.Data
|
|
|
|
|
{
|
|
|
|
|
internal class MetaDataAggregator
|
|
|
|
|
{
|
|
|
|
|
private readonly IApplicationMonitor applicationMonitor;
|
|
|
|
|
private readonly IBrowserApplication browser;
|
|
|
|
|
private readonly ILogger logger;
|
2024-02-22 18:04:00 +01:00
|
|
|
|
private readonly MetaDataSettings settings;
|
2024-02-21 18:37:23 +01:00
|
|
|
|
|
|
|
|
|
private string applicationInfo;
|
|
|
|
|
private string browserInfo;
|
2024-02-22 17:29:17 +01:00
|
|
|
|
private string browserInfoWithoutUrls;
|
2024-02-21 18:37:23 +01:00
|
|
|
|
private TimeSpan elapsed;
|
|
|
|
|
private string triggerInfo;
|
|
|
|
|
private string urls;
|
2024-02-22 17:29:17 +01:00
|
|
|
|
private int urlCount;
|
2024-02-21 18:37:23 +01:00
|
|
|
|
private string windowTitle;
|
|
|
|
|
|
|
|
|
|
internal MetaData Data => new MetaData
|
|
|
|
|
{
|
|
|
|
|
ApplicationInfo = applicationInfo,
|
|
|
|
|
BrowserInfo = browserInfo,
|
|
|
|
|
Elapsed = elapsed,
|
|
|
|
|
TriggerInfo = triggerInfo,
|
|
|
|
|
Urls = urls,
|
|
|
|
|
WindowTitle = windowTitle
|
|
|
|
|
};
|
|
|
|
|
|
2024-02-22 18:04:00 +01:00
|
|
|
|
internal MetaDataAggregator(
|
|
|
|
|
IApplicationMonitor applicationMonitor,
|
|
|
|
|
IBrowserApplication browser,
|
|
|
|
|
TimeSpan elapsed,
|
|
|
|
|
ILogger logger,
|
|
|
|
|
MetaDataSettings settings)
|
2024-02-21 18:37:23 +01:00
|
|
|
|
{
|
|
|
|
|
this.applicationMonitor = applicationMonitor;
|
|
|
|
|
this.browser = browser;
|
|
|
|
|
this.elapsed = elapsed;
|
|
|
|
|
this.logger = logger;
|
2024-02-22 18:04:00 +01:00
|
|
|
|
this.settings = settings;
|
2024-02-21 18:37:23 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal void Capture(IntervalTrigger interval = default, KeyboardTrigger keyboard = default, MouseTrigger mouse = default)
|
|
|
|
|
{
|
2024-02-22 18:04:00 +01:00
|
|
|
|
Initialize();
|
2024-02-21 18:37:23 +01:00
|
|
|
|
CaptureApplicationData();
|
|
|
|
|
CaptureBrowserData();
|
|
|
|
|
|
|
|
|
|
if (interval != default)
|
|
|
|
|
{
|
|
|
|
|
CaptureIntervalTrigger(interval);
|
|
|
|
|
}
|
|
|
|
|
else if (keyboard != default)
|
|
|
|
|
{
|
|
|
|
|
CaptureKeyboardTrigger(keyboard);
|
|
|
|
|
}
|
|
|
|
|
else if (mouse != default)
|
|
|
|
|
{
|
|
|
|
|
CaptureMouseTrigger(mouse);
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-22 17:29:17 +01:00
|
|
|
|
logger.Debug($"Captured metadata: {applicationInfo} / {browserInfoWithoutUrls} / {triggerInfo} / {urlCount} URL(s) / {windowTitle}.");
|
2024-02-21 18:37:23 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void CaptureApplicationData()
|
|
|
|
|
{
|
|
|
|
|
if (applicationMonitor.TryGetActiveApplication(out var application))
|
|
|
|
|
{
|
2024-02-22 18:04:00 +01:00
|
|
|
|
if (settings.CaptureApplicationData)
|
|
|
|
|
{
|
|
|
|
|
applicationInfo = BuildApplicationInfo(application);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (settings.CaptureWindowTitle)
|
|
|
|
|
{
|
|
|
|
|
windowTitle = string.IsNullOrEmpty(application.Window.Title) ? "-" : application.Window.Title;
|
|
|
|
|
}
|
2024-02-21 18:37:23 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void CaptureBrowserData()
|
|
|
|
|
{
|
2024-02-22 18:04:00 +01:00
|
|
|
|
if (settings.CaptureBrowserData)
|
|
|
|
|
{
|
|
|
|
|
var windows = browser.GetWindows();
|
2024-02-21 18:37:23 +01:00
|
|
|
|
|
2024-02-22 18:04:00 +01:00
|
|
|
|
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));
|
|
|
|
|
urlCount = windows.Count();
|
|
|
|
|
}
|
2024-02-21 18:37:23 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void CaptureIntervalTrigger(IntervalTrigger interval)
|
|
|
|
|
{
|
|
|
|
|
triggerInfo = $"Maximum interval of {interval.ConfigurationValue}ms has been reached.";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void CaptureKeyboardTrigger(KeyboardTrigger keyboard)
|
|
|
|
|
{
|
2024-07-25 16:49:52 +02:00
|
|
|
|
var flags = Enum.GetValues(typeof(KeyModifier))
|
|
|
|
|
.OfType<KeyModifier>()
|
|
|
|
|
.Where(m => m != KeyModifier.None && keyboard.Modifier.HasFlag(m) && !keyboard.Key.ToString().Contains(m.ToString()));
|
2024-02-21 18:37:23 +01:00
|
|
|
|
var modifiers = flags.Any() ? string.Join(" + ", flags) + " + " : string.Empty;
|
|
|
|
|
|
2024-07-25 16:49:52 +02:00
|
|
|
|
if (flags.Any())
|
|
|
|
|
{
|
|
|
|
|
triggerInfo = $"'{modifiers}{keyboard.Key}' has been {keyboard.State.ToString().ToLower()}.";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
triggerInfo = $"A key has been {keyboard.State.ToString().ToLower()}.";
|
|
|
|
|
}
|
2024-02-21 18:37:23 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void CaptureMouseTrigger(MouseTrigger mouse)
|
|
|
|
|
{
|
|
|
|
|
if (mouse.Info.IsTouch)
|
|
|
|
|
{
|
|
|
|
|
triggerInfo = $"Tap as {mouse.Button} mouse button has been {mouse.State.ToString().ToLower()} at ({mouse.Info.X}/{mouse.Info.Y}).";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
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();
|
|
|
|
|
}
|
2024-02-22 18:04:00 +01:00
|
|
|
|
|
|
|
|
|
private void Initialize()
|
|
|
|
|
{
|
|
|
|
|
applicationInfo = "-";
|
|
|
|
|
browserInfo = "-";
|
|
|
|
|
browserInfoWithoutUrls = "-";
|
|
|
|
|
triggerInfo = "-";
|
|
|
|
|
urls = "-";
|
|
|
|
|
windowTitle = "-";
|
|
|
|
|
}
|
2024-02-21 18:37:23 +01:00
|
|
|
|
}
|
|
|
|
|
}
|