2021-02-10 23:21:48 +01:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2021 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/.
|
|
|
|
|
*/
|
|
|
|
|
|
2021-03-10 21:26:45 +01:00
|
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Reflection;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
using System.Windows;
|
2021-05-11 20:02:54 +02:00
|
|
|
|
using Microsoft.Web.WebView2.Wpf;
|
2021-03-17 00:05:29 +01:00
|
|
|
|
using SafeExamBrowser.Configuration.Contracts;
|
2021-03-18 23:12:07 +01:00
|
|
|
|
using SafeExamBrowser.Core.Contracts.Notifications;
|
2021-03-25 13:49:45 +01:00
|
|
|
|
using SafeExamBrowser.Core.Contracts.Notifications.Events;
|
2021-03-18 23:12:07 +01:00
|
|
|
|
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
2021-03-25 13:49:45 +01:00
|
|
|
|
using SafeExamBrowser.I18n.Contracts;
|
2021-03-10 21:26:45 +01:00
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2021-02-10 23:21:48 +01:00
|
|
|
|
using SafeExamBrowser.Proctoring.Contracts;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
using SafeExamBrowser.Server.Contracts;
|
2021-02-19 22:03:52 +01:00
|
|
|
|
using SafeExamBrowser.Settings.Proctoring;
|
2021-03-17 00:05:29 +01:00
|
|
|
|
using SafeExamBrowser.SystemComponents.Contracts;
|
2021-02-19 22:03:52 +01:00
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts;
|
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts.Proctoring;
|
2021-02-10 23:21:48 +01:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Proctoring
|
|
|
|
|
{
|
2021-03-18 23:12:07 +01:00
|
|
|
|
public class ProctoringController : IProctoringController, INotification
|
2021-02-10 23:21:48 +01:00
|
|
|
|
{
|
2021-03-17 00:05:29 +01:00
|
|
|
|
private readonly AppConfig appConfig;
|
|
|
|
|
private readonly IFileSystem fileSystem;
|
2021-03-10 21:26:45 +01:00
|
|
|
|
private readonly IModuleLogger logger;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
private readonly IServerProxy server;
|
2021-03-25 13:49:45 +01:00
|
|
|
|
private readonly IText text;
|
2021-02-19 22:03:52 +01:00
|
|
|
|
private readonly IUserInterfaceFactory uiFactory;
|
2021-02-10 23:21:48 +01:00
|
|
|
|
|
2021-03-17 00:05:29 +01:00
|
|
|
|
private string filePath;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
private ProctoringControl control;
|
2021-03-23 21:12:47 +01:00
|
|
|
|
private ProctoringSettings settings;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
private IProctoringWindow window;
|
2021-05-19 01:35:01 +02:00
|
|
|
|
private WindowVisibility windowVisibility;
|
2021-02-19 22:03:52 +01:00
|
|
|
|
|
2021-03-25 13:49:45 +01:00
|
|
|
|
public IconResource IconResource { get; set; }
|
2021-04-12 19:57:11 +02:00
|
|
|
|
public string Tooltip { get; set; }
|
2021-03-18 23:12:07 +01:00
|
|
|
|
|
2021-03-25 13:49:45 +01:00
|
|
|
|
public event NotificationChangedEventHandler NotificationChanged;
|
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
public ProctoringController(
|
|
|
|
|
AppConfig appConfig,
|
|
|
|
|
IFileSystem fileSystem,
|
|
|
|
|
IModuleLogger logger,
|
|
|
|
|
IServerProxy server,
|
|
|
|
|
IText text,
|
|
|
|
|
IUserInterfaceFactory uiFactory)
|
2021-02-19 22:03:52 +01:00
|
|
|
|
{
|
2021-03-17 00:05:29 +01:00
|
|
|
|
this.appConfig = appConfig;
|
|
|
|
|
this.fileSystem = fileSystem;
|
2021-03-10 21:26:45 +01:00
|
|
|
|
this.logger = logger;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
this.server = server;
|
2021-03-25 13:49:45 +01:00
|
|
|
|
this.text = text;
|
2021-02-19 22:03:52 +01:00
|
|
|
|
this.uiFactory = uiFactory;
|
2021-03-25 13:49:45 +01:00
|
|
|
|
|
|
|
|
|
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ProctoringNotification_Inactive.xaml") };
|
2021-04-12 19:57:11 +02:00
|
|
|
|
Tooltip = text.Get(TextKey.Notification_ProctoringInactiveTooltip);
|
2021-02-19 22:03:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-03-18 23:12:07 +01:00
|
|
|
|
public void Activate()
|
|
|
|
|
{
|
2021-03-23 21:12:47 +01:00
|
|
|
|
if (settings.WindowVisibility == WindowVisibility.Visible)
|
|
|
|
|
{
|
|
|
|
|
window?.BringToForeground();
|
|
|
|
|
}
|
|
|
|
|
else if (settings.WindowVisibility == WindowVisibility.AllowToHide || settings.WindowVisibility == WindowVisibility.AllowToShow)
|
|
|
|
|
{
|
2021-04-12 10:59:31 +02:00
|
|
|
|
window?.Toggle();
|
2021-03-23 21:12:47 +01:00
|
|
|
|
}
|
2021-03-18 23:12:07 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-02-19 22:03:52 +01:00
|
|
|
|
public void Initialize(ProctoringSettings settings)
|
|
|
|
|
{
|
2021-04-12 10:59:31 +02:00
|
|
|
|
var start = false;
|
|
|
|
|
|
2021-03-23 21:12:47 +01:00
|
|
|
|
this.settings = settings;
|
2021-05-19 01:35:01 +02:00
|
|
|
|
this.windowVisibility = settings.WindowVisibility;
|
2021-03-23 21:12:47 +01:00
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
server.ProctoringConfigurationReceived += Server_ProctoringConfigurationReceived;
|
|
|
|
|
server.ProctoringInstructionReceived += Server_ProctoringInstructionReceived;
|
2021-02-19 22:03:52 +01:00
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
if (settings.JitsiMeet.Enabled)
|
|
|
|
|
{
|
2021-05-19 02:43:07 +02:00
|
|
|
|
this.settings.JitsiMeet.ServerUrl = Sanitize(settings.JitsiMeet.ServerUrl);
|
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
start = !string.IsNullOrWhiteSpace(settings.JitsiMeet.RoomName);
|
|
|
|
|
start &= !string.IsNullOrWhiteSpace(settings.JitsiMeet.ServerUrl);
|
|
|
|
|
}
|
|
|
|
|
else if (settings.Zoom.Enabled)
|
|
|
|
|
{
|
|
|
|
|
start = !string.IsNullOrWhiteSpace(settings.Zoom.ApiKey);
|
|
|
|
|
start &= !string.IsNullOrWhiteSpace(settings.Zoom.ApiSecret);
|
|
|
|
|
start &= settings.Zoom.MeetingNumber != default(int);
|
|
|
|
|
start &= !string.IsNullOrWhiteSpace(settings.Zoom.UserName);
|
|
|
|
|
}
|
2021-03-17 00:05:29 +01:00
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
if (start)
|
|
|
|
|
{
|
|
|
|
|
StartProctoring();
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-03-04 16:05:22 +01:00
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
public void Terminate()
|
|
|
|
|
{
|
|
|
|
|
StopProctoring();
|
|
|
|
|
}
|
2021-03-23 21:12:47 +01:00
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
private void Server_ProctoringInstructionReceived(string roomName, string serverUrl, string token)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Proctoring instruction received.");
|
2021-03-04 16:05:22 +01:00
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
settings.JitsiMeet.RoomName = roomName;
|
2021-05-19 02:43:07 +02:00
|
|
|
|
settings.JitsiMeet.ServerUrl = Sanitize(serverUrl);
|
2021-04-12 10:59:31 +02:00
|
|
|
|
settings.JitsiMeet.Token = token;
|
2021-03-25 13:49:45 +01:00
|
|
|
|
|
2021-04-12 15:59:42 +02:00
|
|
|
|
StopProctoring();
|
2021-04-12 10:59:31 +02:00
|
|
|
|
StartProctoring();
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-12 15:59:42 +02:00
|
|
|
|
private void Server_ProctoringConfigurationReceived(bool allowChat, bool receiveAudio, bool receiveVideo)
|
2021-04-12 10:59:31 +02:00
|
|
|
|
{
|
|
|
|
|
logger.Info("Proctoring configuration received.");
|
|
|
|
|
|
2021-04-12 15:59:42 +02:00
|
|
|
|
settings.JitsiMeet.AllowChat = allowChat;
|
|
|
|
|
settings.JitsiMeet.ReceiveAudio = receiveAudio;
|
|
|
|
|
settings.JitsiMeet.ReceiveVideo = receiveVideo;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
|
2021-04-12 15:59:42 +02:00
|
|
|
|
if (allowChat || receiveVideo)
|
|
|
|
|
{
|
|
|
|
|
settings.WindowVisibility = WindowVisibility.AllowToHide;
|
|
|
|
|
}
|
2021-05-19 01:35:01 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
settings.WindowVisibility = windowVisibility;
|
|
|
|
|
}
|
2021-04-12 15:59:42 +02:00
|
|
|
|
|
|
|
|
|
StopProctoring();
|
|
|
|
|
StartProctoring();
|
2021-02-19 22:03:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
private void StartProctoring()
|
2021-02-19 22:03:52 +01:00
|
|
|
|
{
|
2021-04-12 10:59:31 +02:00
|
|
|
|
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)));
|
2021-05-11 20:02:54 +02:00
|
|
|
|
control.CreationProperties = new CoreWebView2CreationProperties { UserDataFolder = appConfig.TemporaryDirectory };
|
2021-04-12 10:59:31 +02:00
|
|
|
|
control.EnsureCoreWebView2Async().ContinueWith(_ =>
|
|
|
|
|
{
|
|
|
|
|
control.Dispatcher.Invoke(() =>
|
|
|
|
|
{
|
|
|
|
|
control.CoreWebView2.Navigate(filePath);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
window = uiFactory.CreateProctoringWindow(control);
|
|
|
|
|
window.SetTitle(settings.JitsiMeet.Enabled ? settings.JitsiMeet.Subject : settings.Zoom.UserName);
|
|
|
|
|
window.Show();
|
|
|
|
|
|
|
|
|
|
if (settings.WindowVisibility == WindowVisibility.AllowToShow || settings.WindowVisibility == WindowVisibility.Hidden)
|
|
|
|
|
{
|
|
|
|
|
window.Hide();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IconResource = new XamlIconResource { Uri = new Uri("pack://application:,,,/SafeExamBrowser.UserInterface.Desktop;component/Images/ProctoringNotification_Active.xaml") };
|
2021-04-12 19:57:11 +02:00
|
|
|
|
Tooltip = text.Get(TextKey.Notification_ProctoringActiveTooltip);
|
2021-04-12 10:59:31 +02:00
|
|
|
|
NotificationChanged?.Invoke();
|
|
|
|
|
|
|
|
|
|
logger.Info($"Started proctoring with {(settings.JitsiMeet.Enabled ? "Jitsi Meet" : "Zoom")}.");
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error($"Failed to start proctoring! Reason: {e.Message}", e);
|
|
|
|
|
}
|
|
|
|
|
});
|
2021-03-23 21:12:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
private void StopProctoring()
|
2021-03-23 21:12:47 +01:00
|
|
|
|
{
|
2021-05-12 16:15:49 +02:00
|
|
|
|
if (control != default(ProctoringControl) && window != default(IProctoringWindow))
|
2021-04-12 10:59:31 +02:00
|
|
|
|
{
|
2021-05-12 16:15:49 +02:00
|
|
|
|
control.Dispatcher.Invoke(() =>
|
|
|
|
|
{
|
|
|
|
|
control.ExecuteScriptAsync("api.executeCommand('hangup'); api.dispose();");
|
|
|
|
|
window.Close();
|
|
|
|
|
control = default(ProctoringControl);
|
|
|
|
|
window = default(IProctoringWindow);
|
|
|
|
|
fileSystem.Delete(filePath);
|
|
|
|
|
|
|
|
|
|
logger.Info("Stopped proctoring.");
|
|
|
|
|
});
|
2021-04-12 10:59:31 +02:00
|
|
|
|
}
|
2021-02-19 22:03:52 +01:00
|
|
|
|
}
|
2021-03-10 21:26:45 +01:00
|
|
|
|
|
|
|
|
|
private string LoadContent(ProctoringSettings settings)
|
|
|
|
|
{
|
|
|
|
|
var provider = settings.JitsiMeet.Enabled ? "JitsiMeet" : "Zoom";
|
|
|
|
|
var assembly = Assembly.GetAssembly(typeof(ProctoringController));
|
|
|
|
|
var path = $"{typeof(ProctoringController).Namespace}.{provider}.index.html";
|
|
|
|
|
|
|
|
|
|
using (var stream = assembly.GetManifestResourceStream(path))
|
|
|
|
|
using (var reader = new StreamReader(stream))
|
|
|
|
|
{
|
|
|
|
|
var html = reader.ReadToEnd();
|
|
|
|
|
|
|
|
|
|
if (settings.JitsiMeet.Enabled)
|
|
|
|
|
{
|
2021-04-12 10:59:31 +02:00
|
|
|
|
html = html.Replace("%%_ALLOW_CHAT_%%", settings.JitsiMeet.AllowChat ? "chat" : "");
|
|
|
|
|
html = html.Replace("%%_ALLOW_CLOSED_CAPTIONS_%%", settings.JitsiMeet.AllowCloseCaptions ? "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("%%_SUBJECT_%%", settings.JitsiMeet.ShowMeetingName ? settings.JitsiMeet.Subject : " ");
|
2021-03-10 21:26:45 +01:00
|
|
|
|
html = html.Replace("%%_DOMAIN_%%", settings.JitsiMeet.ServerUrl);
|
|
|
|
|
html = html.Replace("%%_ROOM_NAME_%%", settings.JitsiMeet.RoomName);
|
|
|
|
|
html = html.Replace("%%_TOKEN_%%", settings.JitsiMeet.Token);
|
2021-04-12 10:59:31 +02:00
|
|
|
|
html = html.Replace("'%_VIDEO_MUTED_%'", settings.JitsiMeet.VideoMuted && settings.WindowVisibility != WindowVisibility.Hidden ? "true" : "false");
|
2021-03-10 21:26:45 +01:00
|
|
|
|
}
|
|
|
|
|
else if (settings.Zoom.Enabled)
|
|
|
|
|
{
|
|
|
|
|
html = html.Replace("%%_API_KEY_%%", settings.Zoom.ApiKey);
|
|
|
|
|
html = html.Replace("%%_API_SECRET_%%", settings.Zoom.ApiSecret);
|
|
|
|
|
html = html.Replace("123456789", Convert.ToString(settings.Zoom.MeetingNumber));
|
|
|
|
|
html = html.Replace("%%_USER_NAME_%%", settings.Zoom.UserName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return html;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-05-19 02:43:07 +02:00
|
|
|
|
|
|
|
|
|
private string Sanitize(string serverUrl)
|
|
|
|
|
{
|
|
|
|
|
return serverUrl?.Replace($"{Uri.UriSchemeHttp}{Uri.SchemeDelimiter}", "").Replace($"{Uri.UriSchemeHttps}{Uri.SchemeDelimiter}", "");
|
|
|
|
|
}
|
2021-02-10 23:21:48 +01:00
|
|
|
|
}
|
|
|
|
|
}
|