SEBWIN-608: Refactored server proxy by extracting request implementations.
This commit is contained in:
parent
ae3755df84
commit
da458bcfb0
26 changed files with 1085 additions and 364 deletions
|
@ -20,14 +20,14 @@ namespace SafeExamBrowser.Configuration.ConfigurationData
|
||||||
{
|
{
|
||||||
internal void Process(IDictionary<string, object> rawData, AppSettings settings)
|
internal void Process(IDictionary<string, object> rawData, AppSettings settings)
|
||||||
{
|
{
|
||||||
AllowBrowserToolbarForReloading(rawData, settings);
|
AllowBrowserToolbarForReloading(settings);
|
||||||
CalculateConfigurationKey(rawData, settings);
|
CalculateConfigurationKey(rawData, settings);
|
||||||
HandleBrowserHomeFunctionality(settings);
|
HandleBrowserHomeFunctionality(settings);
|
||||||
InitializeProctoringSettings(settings);
|
InitializeProctoringSettings(settings);
|
||||||
RemoveLegacyBrowsers(settings);
|
RemoveLegacyBrowsers(settings);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void AllowBrowserToolbarForReloading(IDictionary<string, object> rawData, AppSettings settings)
|
private void AllowBrowserToolbarForReloading(AppSettings settings)
|
||||||
{
|
{
|
||||||
if (settings.Browser.AdditionalWindow.AllowReloading && settings.Browser.AdditionalWindow.ShowReloadButton)
|
if (settings.Browser.AdditionalWindow.AllowReloading && settings.Browser.AdditionalWindow.ShowReloadButton)
|
||||||
{
|
{
|
||||||
|
|
|
@ -49,16 +49,16 @@ namespace SafeExamBrowser.Server.Contracts
|
||||||
/// </summary>
|
/// </summary>
|
||||||
event TerminationRequestedEventHandler TerminationRequested;
|
event TerminationRequestedEventHandler TerminationRequested;
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Attempts to initialize a connection with the server.
|
|
||||||
/// </summary>
|
|
||||||
ServerResponse Connect();
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Sends a lock screen confirm notification to the server.
|
/// Sends a lock screen confirm notification to the server.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
ServerResponse ConfirmLockScreen();
|
ServerResponse ConfirmLockScreen();
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Attempts to initialize a connection with the server.
|
||||||
|
/// </summary>
|
||||||
|
ServerResponse Connect();
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Terminates a connection with the server.
|
/// Terminates a connection with the server.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
@ -99,6 +99,11 @@ namespace SafeExamBrowser.Server.Contracts
|
||||||
/// </summary>
|
/// </summary>
|
||||||
ServerResponse LowerHand();
|
ServerResponse LowerHand();
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Sends a raise hand notification to the server.
|
||||||
|
/// </summary>
|
||||||
|
ServerResponse RaiseHand(string message = default);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Sends the selected exam to the server.
|
/// Sends the selected exam to the server.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
@ -118,10 +123,5 @@ namespace SafeExamBrowser.Server.Contracts
|
||||||
/// Stops sending ping and log data to the server.
|
/// Stops sending ping and log data to the server.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
void StopConnectivity();
|
void StopConnectivity();
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Sends a raise hand notification to the server.
|
|
||||||
/// </summary>
|
|
||||||
ServerResponse RaiseHand(string message = default);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
17
SafeExamBrowser.Server/Data/AttributeType.cs
Normal file
17
SafeExamBrowser.Server/Data/AttributeType.cs
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Data
|
||||||
|
{
|
||||||
|
internal enum AttributeType
|
||||||
|
{
|
||||||
|
None,
|
||||||
|
Hand,
|
||||||
|
LockScreen
|
||||||
|
}
|
||||||
|
}
|
|
@ -18,7 +18,7 @@ namespace SafeExamBrowser.Server.Data
|
||||||
internal string Message { get; set; }
|
internal string Message { get; set; }
|
||||||
internal bool ReceiveAudio { get; set; }
|
internal bool ReceiveAudio { get; set; }
|
||||||
internal bool ReceiveVideo { get; set; }
|
internal bool ReceiveVideo { get; set; }
|
||||||
internal string Type { get; set; }
|
internal AttributeType Type { get; set; }
|
||||||
|
|
||||||
internal Attributes()
|
internal Attributes()
|
||||||
{
|
{
|
||||||
|
|
|
@ -151,9 +151,9 @@ namespace SafeExamBrowser.Server
|
||||||
return connectionToken != default;
|
return connectionToken != default;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal bool TryParseExams(HttpContent content, out IList<Exam> exams)
|
internal bool TryParseExams(HttpContent content, out IEnumerable<Exam> exams)
|
||||||
{
|
{
|
||||||
exams = new List<Exam>();
|
var list = new List<Exam>();
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
@ -161,7 +161,7 @@ namespace SafeExamBrowser.Server
|
||||||
|
|
||||||
foreach (var exam in json.AsJEnumerable())
|
foreach (var exam in json.AsJEnumerable())
|
||||||
{
|
{
|
||||||
exams.Add(new Exam
|
list.Add(new Exam
|
||||||
{
|
{
|
||||||
Id = exam["examId"].Value<string>(),
|
Id = exam["examId"].Value<string>(),
|
||||||
LmsName = exam["lmsType"].Value<string>(),
|
LmsName = exam["lmsType"].Value<string>(),
|
||||||
|
@ -175,6 +175,8 @@ namespace SafeExamBrowser.Server
|
||||||
logger.Error("Failed to parse exams!", e);
|
logger.Error("Failed to parse exams!", e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
exams = list;
|
||||||
|
|
||||||
return exams.Any();
|
return exams.Any();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -271,7 +273,15 @@ namespace SafeExamBrowser.Server
|
||||||
|
|
||||||
if (attributesJson.ContainsKey("type"))
|
if (attributesJson.ContainsKey("type"))
|
||||||
{
|
{
|
||||||
attributes.Type = attributesJson["type"].Value<string>();
|
switch (attributesJson["type"].Value<string>())
|
||||||
|
{
|
||||||
|
case "lockscreen":
|
||||||
|
attributes.Type = AttributeType.LockScreen;
|
||||||
|
break;
|
||||||
|
case "raisehand":
|
||||||
|
attributes.Type = AttributeType.Hand;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
42
SafeExamBrowser.Server/Requests/ApiRequest.cs
Normal file
42
SafeExamBrowser.Server/Requests/ApiRequest.cs
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class ApiRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal ApiRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(out ApiVersion1 api, out string message)
|
||||||
|
{
|
||||||
|
var success = TryExecute(HttpMethod.Get, settings.ApiUrl, out var response);
|
||||||
|
|
||||||
|
api = new ApiVersion1();
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
parser.TryParseApi(response.Content, out api);
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
37
SafeExamBrowser.Server/Requests/AppSignatureKeyRequest.cs
Normal file
37
SafeExamBrowser.Server/Requests/AppSignatureKeyRequest.cs
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class AppSignatureKeyRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal AppSignatureKeyRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(out string message)
|
||||||
|
{
|
||||||
|
var content = $"seb_signature_key={"WINDOWS-TEST-ASK-1234"}";
|
||||||
|
var success = TryExecute(new HttpMethod("PATCH"), api.HandshakeEndpoint, out var response, content, ContentType.URL_ENCODED, Authorization, Token);
|
||||||
|
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
63
SafeExamBrowser.Server/Requests/AvailableExamsRequest.cs
Normal file
63
SafeExamBrowser.Server/Requests/AvailableExamsRequest.cs
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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.Collections.Generic;
|
||||||
|
using System.Net.Http;
|
||||||
|
using SafeExamBrowser.Configuration.Contracts;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Contracts.Data;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
using SafeExamBrowser.SystemComponents.Contracts;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class AvailableExamsRequest : BaseRequest
|
||||||
|
{
|
||||||
|
private readonly AppConfig appConfig;
|
||||||
|
private readonly ISystemInfo systemInfo;
|
||||||
|
private readonly IUserInfo userInfo;
|
||||||
|
|
||||||
|
internal AvailableExamsRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
AppConfig appConfig,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings,
|
||||||
|
ISystemInfo systemInfo,
|
||||||
|
IUserInfo userInfo) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
this.appConfig = appConfig;
|
||||||
|
this.systemInfo = systemInfo;
|
||||||
|
this.userInfo = userInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(string examId, out IEnumerable<Exam> exams, out string message)
|
||||||
|
{
|
||||||
|
var clientInfo = $"client_id={userInfo.GetUserName()}&seb_machine_name={systemInfo.Name}";
|
||||||
|
var versionInfo = $"seb_os_name={systemInfo.OperatingSystemInfo}&seb_version={appConfig.ProgramInformationalVersion}";
|
||||||
|
var content = $"institutionId={settings.Institution}&{clientInfo}&{versionInfo}{(examId == default ? "" : $"&examId={examId}")}";
|
||||||
|
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.HandshakeEndpoint, out var response, content, ContentType.URL_ENCODED, Authorization);
|
||||||
|
|
||||||
|
exams = default;
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
var hasExams = parser.TryParseExams(response.Content, out exams);
|
||||||
|
var hasToken = TryRetrieveConnectionToken(response);
|
||||||
|
|
||||||
|
success = hasExams && hasToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
190
SafeExamBrowser.Server/Requests/BaseRequest.cs
Normal file
190
SafeExamBrowser.Server/Requests/BaseRequest.cs
Normal file
|
@ -0,0 +1,190 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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.Generic;
|
||||||
|
using System.Net;
|
||||||
|
using System.Net.Http;
|
||||||
|
using System.Net.Http.Headers;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal abstract class BaseRequest
|
||||||
|
{
|
||||||
|
private static string connectionToken;
|
||||||
|
private static string oauth2Token;
|
||||||
|
|
||||||
|
private readonly HttpClient httpClient;
|
||||||
|
|
||||||
|
protected readonly ApiVersion1 api;
|
||||||
|
protected readonly ILogger logger;
|
||||||
|
protected readonly Parser parser;
|
||||||
|
protected readonly ServerSettings settings;
|
||||||
|
|
||||||
|
protected (string, string) Authorization => (Header.AUTHORIZATION, $"Bearer {oauth2Token}");
|
||||||
|
protected (string, string) Token => (Header.CONNECTION_TOKEN, connectionToken);
|
||||||
|
|
||||||
|
internal static string ConnectionToken
|
||||||
|
{
|
||||||
|
get { return connectionToken; }
|
||||||
|
set { connectionToken = value; }
|
||||||
|
}
|
||||||
|
|
||||||
|
internal static string Oauth2Token
|
||||||
|
{
|
||||||
|
get { return oauth2Token; }
|
||||||
|
set { oauth2Token = value; }
|
||||||
|
}
|
||||||
|
|
||||||
|
protected BaseRequest(ApiVersion1 api, HttpClient httpClient, ILogger logger, Parser parser, ServerSettings settings)
|
||||||
|
{
|
||||||
|
this.api = api;
|
||||||
|
this.httpClient = httpClient;
|
||||||
|
this.logger = logger;
|
||||||
|
this.parser = parser;
|
||||||
|
this.settings = settings;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected bool TryExecute(
|
||||||
|
HttpMethod method,
|
||||||
|
string url,
|
||||||
|
out HttpResponseMessage response,
|
||||||
|
string content = default,
|
||||||
|
string contentType = default,
|
||||||
|
params (string name, string value)[] headers)
|
||||||
|
{
|
||||||
|
response = default;
|
||||||
|
|
||||||
|
for (var attempt = 0; attempt < settings.RequestAttempts && (response == default || !response.IsSuccessStatusCode); attempt++)
|
||||||
|
{
|
||||||
|
var request = BuildRequest(method, url, content, contentType, headers);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
response = httpClient.SendAsync(request).GetAwaiter().GetResult();
|
||||||
|
|
||||||
|
if (request.RequestUri.AbsolutePath != api.LogEndpoint && request.RequestUri.AbsolutePath != api.PingEndpoint)
|
||||||
|
{
|
||||||
|
logger.Debug($"Completed request: {request.Method} '{request.RequestUri}' -> {response.ToLogString()}");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.StatusCode == HttpStatusCode.Unauthorized && parser.IsTokenExpired(response.Content))
|
||||||
|
{
|
||||||
|
logger.Info("OAuth2 token has expired, attempting to retrieve new one...");
|
||||||
|
|
||||||
|
if (TryRetrieveOAuth2Token(out var message))
|
||||||
|
{
|
||||||
|
headers = UpdateOAuth2Token(headers);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (TaskCanceledException)
|
||||||
|
{
|
||||||
|
logger.Debug($"Request {request.Method} '{request.RequestUri}' did not complete within {settings.RequestTimeout}ms!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
logger.Debug($"Request {request.Method} '{request.RequestUri}' failed due to {e}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return response != default && response.IsSuccessStatusCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected bool TryRetrieveConnectionToken(HttpResponseMessage response)
|
||||||
|
{
|
||||||
|
var success = parser.TryParseConnectionToken(response, out connectionToken);
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
logger.Info("Successfully retrieved connection token.");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
logger.Error("Failed to retrieve connection token!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected bool TryRetrieveOAuth2Token(out string message)
|
||||||
|
{
|
||||||
|
var secret = Convert.ToBase64String(Encoding.UTF8.GetBytes($"{settings.ClientName}:{settings.ClientSecret}"));
|
||||||
|
var authorization = (Header.AUTHORIZATION, $"Basic {secret}");
|
||||||
|
var content = "grant_type=client_credentials&scope=read write";
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.AccessTokenEndpoint, out var response, content, ContentType.URL_ENCODED, authorization);
|
||||||
|
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
if (success && parser.TryParseOauth2Token(response.Content, out oauth2Token))
|
||||||
|
{
|
||||||
|
logger.Info("Successfully retrieved OAuth2 token.");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
logger.Error("Failed to retrieve OAuth2 token!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
private HttpRequestMessage BuildRequest(
|
||||||
|
HttpMethod method,
|
||||||
|
string url,
|
||||||
|
string content = default,
|
||||||
|
string contentType = default,
|
||||||
|
params (string name, string value)[] headers)
|
||||||
|
{
|
||||||
|
var request = new HttpRequestMessage(method, url);
|
||||||
|
|
||||||
|
if (content != default)
|
||||||
|
{
|
||||||
|
request.Content = new StringContent(content, Encoding.UTF8);
|
||||||
|
|
||||||
|
if (contentType != default)
|
||||||
|
{
|
||||||
|
request.Content.Headers.ContentType = MediaTypeHeaderValue.Parse(contentType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
request.Headers.Add(Header.ACCEPT, "application/json, */*");
|
||||||
|
|
||||||
|
foreach (var (name, value) in headers)
|
||||||
|
{
|
||||||
|
request.Headers.Add(name, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return request;
|
||||||
|
}
|
||||||
|
|
||||||
|
private (string name, string value)[] UpdateOAuth2Token((string name, string value)[] headers)
|
||||||
|
{
|
||||||
|
var result = new List<(string name, string value)>();
|
||||||
|
|
||||||
|
foreach (var header in headers)
|
||||||
|
{
|
||||||
|
if (header.name == Header.AUTHORIZATION)
|
||||||
|
{
|
||||||
|
result.Add((Header.AUTHORIZATION, $"Bearer {oauth2Token}"));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
result.Add(header);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.ToArray();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
45
SafeExamBrowser.Server/Requests/ConfirmLockScreenRequest.cs
Normal file
45
SafeExamBrowser.Server/Requests/ConfirmLockScreenRequest.cs
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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.Net.Http;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class ConfirmLockScreenRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal ConfirmLockScreenRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(int lockScreenId, out string message)
|
||||||
|
{
|
||||||
|
var json = new JObject
|
||||||
|
{
|
||||||
|
["numericValue"] = lockScreenId,
|
||||||
|
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
||||||
|
["type"] = "NOTIFICATION_CONFIRMED"
|
||||||
|
};
|
||||||
|
var content = json.ToString();
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, content, ContentType.JSON, Authorization, Token);
|
||||||
|
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
16
SafeExamBrowser.Server/Requests/ContentType.cs
Normal file
16
SafeExamBrowser.Server/Requests/ContentType.cs
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal static class ContentType
|
||||||
|
{
|
||||||
|
internal const string JSON = "application/json;charset=UTF-8";
|
||||||
|
internal const string URL_ENCODED = "application/x-www-form-urlencoded";
|
||||||
|
}
|
||||||
|
}
|
37
SafeExamBrowser.Server/Requests/DisconnectionRequest.cs
Normal file
37
SafeExamBrowser.Server/Requests/DisconnectionRequest.cs
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class DisconnectionRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal DisconnectionRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(out string message)
|
||||||
|
{
|
||||||
|
var content = "delete=true";
|
||||||
|
var success = TryExecute(HttpMethod.Delete, api.HandshakeEndpoint, out var response, content, ContentType.URL_ENCODED, Authorization, Token);
|
||||||
|
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
39
SafeExamBrowser.Server/Requests/ExamConfigurationRequest.cs
Normal file
39
SafeExamBrowser.Server/Requests/ExamConfigurationRequest.cs
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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;
|
||||||
|
using SafeExamBrowser.Server.Contracts.Data;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class ExamConfigurationRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal ExamConfigurationRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(Exam exam, out HttpContent content, out string message)
|
||||||
|
{
|
||||||
|
var url = $"{api.ConfigurationEndpoint}?examId={exam.Id}";
|
||||||
|
var success = TryExecute(HttpMethod.Get, url, out var response, default, default, Authorization, Token);
|
||||||
|
|
||||||
|
content = response.Content;
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
17
SafeExamBrowser.Server/Requests/Header.cs
Normal file
17
SafeExamBrowser.Server/Requests/Header.cs
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal static class Header
|
||||||
|
{
|
||||||
|
internal const string ACCEPT = "Accept";
|
||||||
|
internal const string AUTHORIZATION = "Authorization";
|
||||||
|
internal const string CONNECTION_TOKEN = "SEBConnectionToken";
|
||||||
|
}
|
||||||
|
}
|
46
SafeExamBrowser.Server/Requests/LockScreenRequest.cs
Normal file
46
SafeExamBrowser.Server/Requests/LockScreenRequest.cs
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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.Net.Http;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class LockScreenRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal LockScreenRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(int lockScreenId, string text, out string message)
|
||||||
|
{
|
||||||
|
var json = new JObject
|
||||||
|
{
|
||||||
|
["numericValue"] = lockScreenId,
|
||||||
|
["text"] = $"<lockscreen> {text}",
|
||||||
|
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
||||||
|
["type"] = "NOTIFICATION"
|
||||||
|
};
|
||||||
|
var content = json.ToString();
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, content, ContentType.JSON, Authorization, Token);
|
||||||
|
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
42
SafeExamBrowser.Server/Requests/LogRequest.cs
Normal file
42
SafeExamBrowser.Server/Requests/LogRequest.cs
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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 Newtonsoft.Json.Linq;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class LogRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal LogRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(ILogMessage message)
|
||||||
|
{
|
||||||
|
var json = new JObject
|
||||||
|
{
|
||||||
|
["text"] = message.Message,
|
||||||
|
["timestamp"] = message.DateTime.ToUnixTimestamp(),
|
||||||
|
["type"] = message.Severity.ToLogType()
|
||||||
|
};
|
||||||
|
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out _, json.ToString(), ContentType.JSON, Authorization, Token);
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
45
SafeExamBrowser.Server/Requests/LowerHandRequest.cs
Normal file
45
SafeExamBrowser.Server/Requests/LowerHandRequest.cs
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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.Net.Http;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class LowerHandRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal LowerHandRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(int handId, out string message)
|
||||||
|
{
|
||||||
|
var json = new JObject
|
||||||
|
{
|
||||||
|
["numericValue"] = handId,
|
||||||
|
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
||||||
|
["type"] = "NOTIFICATION_CONFIRMED"
|
||||||
|
};
|
||||||
|
var content = json.ToString();
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, content, ContentType.JSON, Authorization, Token);
|
||||||
|
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
49
SafeExamBrowser.Server/Requests/NetworkAdapterRequest.cs
Normal file
49
SafeExamBrowser.Server/Requests/NetworkAdapterRequest.cs
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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.Net.Http;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Logging;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class NetworkAdapterRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal NetworkAdapterRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(string text, int? value = default)
|
||||||
|
{
|
||||||
|
var json = new JObject
|
||||||
|
{
|
||||||
|
["text"] = text,
|
||||||
|
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
||||||
|
["type"] = LogLevel.Info.ToLogType()
|
||||||
|
};
|
||||||
|
|
||||||
|
if (value != default)
|
||||||
|
{
|
||||||
|
json["numericValue"] = value.Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, json.ToString(), ContentType.JSON, Authorization, Token);
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
32
SafeExamBrowser.Server/Requests/OAuth2TokenRequest.cs
Normal file
32
SafeExamBrowser.Server/Requests/OAuth2TokenRequest.cs
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class OAuth2TokenRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal OAuth2TokenRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(out string message)
|
||||||
|
{
|
||||||
|
return TryRetrieveOAuth2Token(out message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
45
SafeExamBrowser.Server/Requests/PingRequest.cs
Normal file
45
SafeExamBrowser.Server/Requests/PingRequest.cs
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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.Net.Http;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class PingRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal PingRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(int pingNumber, out HttpContent content, out string message, string confirmation = default)
|
||||||
|
{
|
||||||
|
var requestContent = $"timestamp={DateTime.Now.ToUnixTimestamp()}&ping-number={pingNumber}";
|
||||||
|
|
||||||
|
if (confirmation != default)
|
||||||
|
{
|
||||||
|
requestContent = $"{requestContent}&instruction-confirm={confirmation}";
|
||||||
|
}
|
||||||
|
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.PingEndpoint, out var response, requestContent, ContentType.URL_ENCODED, Authorization, Token);
|
||||||
|
|
||||||
|
content = response.Content;
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
45
SafeExamBrowser.Server/Requests/PowerSupplyRequest.cs
Normal file
45
SafeExamBrowser.Server/Requests/PowerSupplyRequest.cs
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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.Net.Http;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Logging;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class PowerSupplyRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal PowerSupplyRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(string text, int value)
|
||||||
|
{
|
||||||
|
var json = new JObject
|
||||||
|
{
|
||||||
|
["numericValue"] = value,
|
||||||
|
["text"] = text,
|
||||||
|
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
||||||
|
["type"] = LogLevel.Info.ToLogType()
|
||||||
|
};
|
||||||
|
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out _, json.ToString(), ContentType.JSON, Authorization, Token);
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
46
SafeExamBrowser.Server/Requests/RaiseHandRequest.cs
Normal file
46
SafeExamBrowser.Server/Requests/RaiseHandRequest.cs
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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.Net.Http;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class RaiseHandRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal RaiseHandRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(int handId, string text, out string message)
|
||||||
|
{
|
||||||
|
var json = new JObject
|
||||||
|
{
|
||||||
|
["numericValue"] = handId,
|
||||||
|
["text"] = $"<raisehand> {text}",
|
||||||
|
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
||||||
|
["type"] = "NOTIFICATION"
|
||||||
|
};
|
||||||
|
var content = json.ToString();
|
||||||
|
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, content, ContentType.JSON, Authorization, Token);
|
||||||
|
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
40
SafeExamBrowser.Server/Requests/SelectExamRequest.cs
Normal file
40
SafeExamBrowser.Server/Requests/SelectExamRequest.cs
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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;
|
||||||
|
using SafeExamBrowser.Server.Contracts.Data;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class SelectExamRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal SelectExamRequest(ApiVersion1 api, HttpClient httpClient, ILogger logger, Parser parser, ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(Exam exam, out string message, out string salt)
|
||||||
|
{
|
||||||
|
var content = $"examId={exam.Id}";
|
||||||
|
var method = new HttpMethod("PATCH");
|
||||||
|
var success = TryExecute(method, api.HandshakeEndpoint, out var response, content, ContentType.URL_ENCODED, Authorization, Token);
|
||||||
|
|
||||||
|
message = response.ToLogString();
|
||||||
|
salt = default;
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
parser.TryParseAppSignatureKeySalt(response, out salt);
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
37
SafeExamBrowser.Server/Requests/SessionIdentifierRequest.cs
Normal file
37
SafeExamBrowser.Server/Requests/SessionIdentifierRequest.cs
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022 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;
|
||||||
|
using SafeExamBrowser.Server.Data;
|
||||||
|
using SafeExamBrowser.Settings.Server;
|
||||||
|
|
||||||
|
namespace SafeExamBrowser.Server.Requests
|
||||||
|
{
|
||||||
|
internal class SessionIdentifierRequest : BaseRequest
|
||||||
|
{
|
||||||
|
internal SessionIdentifierRequest(
|
||||||
|
ApiVersion1 api,
|
||||||
|
HttpClient httpClient,
|
||||||
|
ILogger logger,
|
||||||
|
Parser parser,
|
||||||
|
ServerSettings settings) : base(api, httpClient, logger, parser, settings)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
internal bool TryExecute(string examId, string identifier, out string message)
|
||||||
|
{
|
||||||
|
var content = $"examId={examId}&seb_user_session_id={identifier}";
|
||||||
|
var success = TryExecute(HttpMethod.Put, api.HandshakeEndpoint, out var response, content, ContentType.URL_ENCODED, Authorization, Token);
|
||||||
|
|
||||||
|
message = response.ToLogString();
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -60,11 +60,31 @@
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Compile Include="Data\ApiVersion1.cs" />
|
<Compile Include="Data\ApiVersion1.cs" />
|
||||||
<Compile Include="Data\Attributes.cs" />
|
<Compile Include="Data\Attributes.cs" />
|
||||||
|
<Compile Include="Data\AttributeType.cs" />
|
||||||
<Compile Include="Data\Instructions.cs" />
|
<Compile Include="Data\Instructions.cs" />
|
||||||
<Compile Include="Extensions.cs" />
|
<Compile Include="Extensions.cs" />
|
||||||
<Compile Include="FileSystem.cs" />
|
<Compile Include="FileSystem.cs" />
|
||||||
<Compile Include="Parser.cs" />
|
<Compile Include="Parser.cs" />
|
||||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||||
|
<Compile Include="Requests\ApiRequest.cs" />
|
||||||
|
<Compile Include="Requests\AppSignatureKeyRequest.cs" />
|
||||||
|
<Compile Include="Requests\AvailableExamsRequest.cs" />
|
||||||
|
<Compile Include="Requests\BaseRequest.cs" />
|
||||||
|
<Compile Include="Requests\ConfirmLockScreenRequest.cs" />
|
||||||
|
<Compile Include="Requests\ContentType.cs" />
|
||||||
|
<Compile Include="Requests\DisconnectionRequest.cs" />
|
||||||
|
<Compile Include="Requests\ExamConfigurationRequest.cs" />
|
||||||
|
<Compile Include="Requests\Header.cs" />
|
||||||
|
<Compile Include="Requests\LockScreenRequest.cs" />
|
||||||
|
<Compile Include="Requests\LogRequest.cs" />
|
||||||
|
<Compile Include="Requests\LowerHandRequest.cs" />
|
||||||
|
<Compile Include="Requests\NetworkAdapterRequest.cs" />
|
||||||
|
<Compile Include="Requests\OAuth2TokenRequest.cs" />
|
||||||
|
<Compile Include="Requests\PingRequest.cs" />
|
||||||
|
<Compile Include="Requests\PowerSupplyRequest.cs" />
|
||||||
|
<Compile Include="Requests\RaiseHandRequest.cs" />
|
||||||
|
<Compile Include="Requests\SelectExamRequest.cs" />
|
||||||
|
<Compile Include="Requests\SessionIdentifierRequest.cs" />
|
||||||
<Compile Include="ServerProxy.cs" />
|
<Compile Include="ServerProxy.cs" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
|
|
@ -10,21 +10,17 @@ using System;
|
||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
using System.Net.Http.Headers;
|
|
||||||
using System.Text;
|
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using System.Timers;
|
using System.Timers;
|
||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
using Newtonsoft.Json.Linq;
|
|
||||||
using SafeExamBrowser.Configuration.Contracts;
|
using SafeExamBrowser.Configuration.Contracts;
|
||||||
using SafeExamBrowser.Logging.Contracts;
|
using SafeExamBrowser.Logging.Contracts;
|
||||||
using SafeExamBrowser.Server.Contracts;
|
using SafeExamBrowser.Server.Contracts;
|
||||||
using SafeExamBrowser.Server.Contracts.Data;
|
using SafeExamBrowser.Server.Contracts.Data;
|
||||||
using SafeExamBrowser.Server.Contracts.Events;
|
using SafeExamBrowser.Server.Contracts.Events;
|
||||||
using SafeExamBrowser.Server.Data;
|
using SafeExamBrowser.Server.Data;
|
||||||
using SafeExamBrowser.Settings.Logging;
|
using SafeExamBrowser.Server.Requests;
|
||||||
using SafeExamBrowser.Settings.Server;
|
using SafeExamBrowser.Settings.Server;
|
||||||
using SafeExamBrowser.SystemComponents.Contracts;
|
using SafeExamBrowser.SystemComponents.Contracts;
|
||||||
using SafeExamBrowser.SystemComponents.Contracts.Network;
|
using SafeExamBrowser.SystemComponents.Contracts.Network;
|
||||||
|
@ -49,16 +45,14 @@ namespace SafeExamBrowser.Server
|
||||||
private readonly INetworkAdapter networkAdapter;
|
private readonly INetworkAdapter networkAdapter;
|
||||||
|
|
||||||
private ApiVersion1 api;
|
private ApiVersion1 api;
|
||||||
private string connectionToken;
|
|
||||||
private bool connectedToPowergrid;
|
private bool connectedToPowergrid;
|
||||||
|
private int currentHandId;
|
||||||
private int currentLockScreenId;
|
private int currentLockScreenId;
|
||||||
private int currentPowerSupplyValue;
|
private int currentPowerSupplyValue;
|
||||||
private int currentRaisHandId;
|
|
||||||
private int currentWlanValue;
|
private int currentWlanValue;
|
||||||
private string examId;
|
private string examId;
|
||||||
private HttpClient httpClient;
|
private HttpClient httpClient;
|
||||||
private int notificationId;
|
private int notificationId;
|
||||||
private string oauth2Token;
|
|
||||||
private int pingNumber;
|
private int pingNumber;
|
||||||
private ServerSettings settings;
|
private ServerSettings settings;
|
||||||
|
|
||||||
|
@ -92,15 +86,32 @@ namespace SafeExamBrowser.Server
|
||||||
this.userInfo = userInfo;
|
this.userInfo = userInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public ServerResponse ConfirmLockScreen()
|
||||||
|
{
|
||||||
|
var request = new ConfirmLockScreenRequest(api, httpClient, logger, parser, settings);
|
||||||
|
var success = request.TryExecute(currentLockScreenId, out var message);
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
logger.Info($"Successfully sent notification confirmation for lock screen #{currentLockScreenId}.");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
logger.Error($"Failed to send notification confirmation for lock screen #{currentLockScreenId}!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return new ServerResponse(success, message);
|
||||||
|
}
|
||||||
|
|
||||||
public ServerResponse Connect()
|
public ServerResponse Connect()
|
||||||
{
|
{
|
||||||
var success = TryExecute(HttpMethod.Get, settings.ApiUrl, out var response);
|
var request = new ApiRequest(api, httpClient, logger, parser, settings);
|
||||||
var message = response.ToLogString();
|
var success = request.TryExecute(out api, out var message);
|
||||||
|
|
||||||
if (success && parser.TryParseApi(response.Content, out api))
|
if (success)
|
||||||
{
|
{
|
||||||
logger.Info("Successfully loaded server API.");
|
logger.Info("Successfully loaded server API.");
|
||||||
success = TryRetrieveOAuth2Token(out message);
|
success = new OAuth2TokenRequest(api, httpClient, logger, parser, settings).TryExecute(out message);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -112,13 +123,8 @@ namespace SafeExamBrowser.Server
|
||||||
|
|
||||||
public ServerResponse Disconnect()
|
public ServerResponse Disconnect()
|
||||||
{
|
{
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
var request = new DisconnectionRequest(api, httpClient, logger, parser, settings);
|
||||||
var content = "delete=true";
|
var success = request.TryExecute(out var message);
|
||||||
var contentType = "application/x-www-form-urlencoded";
|
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
|
|
||||||
var success = TryExecute(HttpMethod.Delete, api.HandshakeEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
var message = response.ToLogString();
|
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
|
@ -134,39 +140,16 @@ namespace SafeExamBrowser.Server
|
||||||
|
|
||||||
public ServerResponse<IEnumerable<Exam>> GetAvailableExams(string examId = default)
|
public ServerResponse<IEnumerable<Exam>> GetAvailableExams(string examId = default)
|
||||||
{
|
{
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
var request = new AvailableExamsRequest(api, appConfig, httpClient, logger, parser, settings, systemInfo, userInfo);
|
||||||
var clientInfo = $"client_id={userInfo.GetUserName()}&seb_machine_name={systemInfo.Name}";
|
var success = request.TryExecute(examId, out var exams, out var message);
|
||||||
var versionInfo = $"seb_os_name={systemInfo.OperatingSystemInfo}&seb_version={appConfig.ProgramInformationalVersion}";
|
|
||||||
var content = $"institutionId={settings.Institution}&{clientInfo}&{versionInfo}{(examId == default ? "" : $"&examId={examId}")}";
|
|
||||||
var contentType = "application/x-www-form-urlencoded";
|
|
||||||
var exams = default(IList<Exam>);
|
|
||||||
|
|
||||||
var success = TryExecute(HttpMethod.Post, api.HandshakeEndpoint, out var response, content, contentType, authorization);
|
|
||||||
var message = response.ToLogString();
|
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
var hasExams = parser.TryParseExams(response.Content, out exams);
|
logger.Info("Successfully retrieved available exams.");
|
||||||
var hasToken = parser.TryParseConnectionToken(response, out connectionToken);
|
|
||||||
|
|
||||||
success = hasExams && hasToken;
|
|
||||||
|
|
||||||
if (success)
|
|
||||||
{
|
|
||||||
logger.Info("Successfully retrieved connection token and available exams.");
|
|
||||||
}
|
|
||||||
else if (!hasExams)
|
|
||||||
{
|
|
||||||
logger.Error("Failed to retrieve available exams!");
|
|
||||||
}
|
|
||||||
else if (!hasToken)
|
|
||||||
{
|
|
||||||
logger.Error("Failed to retrieve connection token!");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
logger.Error("Failed to load connection token and available exams!");
|
logger.Error("Failed to retrieve available exams!");
|
||||||
}
|
}
|
||||||
|
|
||||||
return new ServerResponse<IEnumerable<Exam>>(success, exams, message);
|
return new ServerResponse<IEnumerable<Exam>>(success, exams, message);
|
||||||
|
@ -174,18 +157,15 @@ namespace SafeExamBrowser.Server
|
||||||
|
|
||||||
public ServerResponse<Uri> GetConfigurationFor(Exam exam)
|
public ServerResponse<Uri> GetConfigurationFor(Exam exam)
|
||||||
{
|
{
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
var request = new ExamConfigurationRequest(api, httpClient, logger, parser, settings);
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
var success = request.TryExecute(exam, out var content, out var message);
|
||||||
var uri = default(Uri);
|
var uri = default(Uri);
|
||||||
|
|
||||||
var success = TryExecute(HttpMethod.Get, $"{api.ConfigurationEndpoint}?examId={exam.Id}", out var response, default, default, authorization, token);
|
|
||||||
var message = response.ToLogString();
|
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
logger.Info("Successfully retrieved exam configuration.");
|
logger.Info("Successfully retrieved exam configuration.");
|
||||||
|
|
||||||
success = fileSystem.TrySaveFile(response.Content, out uri);
|
success = fileSystem.TrySaveFile(content, out uri);
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
|
@ -209,8 +189,8 @@ namespace SafeExamBrowser.Server
|
||||||
return new ConnectionInfo
|
return new ConnectionInfo
|
||||||
{
|
{
|
||||||
Api = JsonConvert.SerializeObject(api),
|
Api = JsonConvert.SerializeObject(api),
|
||||||
ConnectionToken = connectionToken,
|
ConnectionToken = BaseRequest.ConnectionToken,
|
||||||
Oauth2Token = oauth2Token
|
Oauth2Token = BaseRequest.Oauth2Token
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -230,26 +210,35 @@ namespace SafeExamBrowser.Server
|
||||||
public void Initialize(string api, string connectionToken, string examId, string oauth2Token, ServerSettings settings)
|
public void Initialize(string api, string connectionToken, string examId, string oauth2Token, ServerSettings settings)
|
||||||
{
|
{
|
||||||
this.api = JsonConvert.DeserializeObject<ApiVersion1>(api);
|
this.api = JsonConvert.DeserializeObject<ApiVersion1>(api);
|
||||||
this.connectionToken = connectionToken;
|
|
||||||
this.examId = examId;
|
this.examId = examId;
|
||||||
this.oauth2Token = oauth2Token;
|
|
||||||
|
BaseRequest.ConnectionToken = connectionToken;
|
||||||
|
BaseRequest.Oauth2Token = oauth2Token;
|
||||||
|
|
||||||
Initialize(settings);
|
Initialize(settings);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public ServerResponse LockScreen(string text = null)
|
||||||
|
{
|
||||||
|
var request = new LockScreenRequest(api, httpClient, logger, parser, settings);
|
||||||
|
var success = request.TryExecute(currentLockScreenId = ++notificationId, text, out var message);
|
||||||
|
|
||||||
|
if (success)
|
||||||
|
{
|
||||||
|
logger.Info($"Successfully sent notification for lock screen #{currentLockScreenId}.");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
logger.Error($"Failed to send notification for lock screen #{currentLockScreenId}!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return new ServerResponse(success, message);
|
||||||
|
}
|
||||||
|
|
||||||
public ServerResponse LowerHand()
|
public ServerResponse LowerHand()
|
||||||
{
|
{
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
var request = new LowerHandRequest(api, httpClient, logger, parser, settings);
|
||||||
var contentType = "application/json;charset=UTF-8";
|
var success = request.TryExecute(currentHandId, out var message);
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
var json = new JObject
|
|
||||||
{
|
|
||||||
["type"] = "NOTIFICATION_CONFIRMED",
|
|
||||||
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
|
||||||
["numericValue"] = currentRaisHandId,
|
|
||||||
};
|
|
||||||
var content = json.ToString();
|
|
||||||
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
|
@ -260,33 +249,7 @@ namespace SafeExamBrowser.Server
|
||||||
logger.Error("Failed to send lower hand notification!");
|
logger.Error("Failed to send lower hand notification!");
|
||||||
}
|
}
|
||||||
|
|
||||||
return new ServerResponse(success, response.ToLogString());
|
return new ServerResponse(success, message);
|
||||||
}
|
|
||||||
|
|
||||||
public ServerResponse ConfirmLockScreen()
|
|
||||||
{
|
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
|
||||||
var contentType = "application/json;charset=UTF-8";
|
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
var json = new JObject
|
|
||||||
{
|
|
||||||
["type"] = "NOTIFICATION_CONFIRMED",
|
|
||||||
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
|
||||||
["numericValue"] = currentLockScreenId,
|
|
||||||
};
|
|
||||||
var content = json.ToString();
|
|
||||||
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
|
|
||||||
if (success)
|
|
||||||
{
|
|
||||||
logger.Info("Successfully sent lock screen confirm notification.");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
logger.Error("Failed to send lock screen confirm notification!");
|
|
||||||
}
|
|
||||||
|
|
||||||
return new ServerResponse(success, response.ToLogString());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Notify(ILogContent content)
|
public void Notify(ILogContent content)
|
||||||
|
@ -294,20 +257,10 @@ namespace SafeExamBrowser.Server
|
||||||
logContent.Enqueue(content);
|
logContent.Enqueue(content);
|
||||||
}
|
}
|
||||||
|
|
||||||
public ServerResponse RaiseHand(string message = null)
|
public ServerResponse RaiseHand(string text = null)
|
||||||
{
|
{
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
var request = new RaiseHandRequest(api, httpClient, logger, parser, settings);
|
||||||
var contentType = "application/json;charset=UTF-8";
|
var success = request.TryExecute(currentHandId = ++notificationId, text, out var message);
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
var json = new JObject
|
|
||||||
{
|
|
||||||
["type"] = "NOTIFICATION",
|
|
||||||
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
|
||||||
["numericValue"] = currentRaisHandId = ++notificationId,
|
|
||||||
["text"] = $"<raisehand> {message}"
|
|
||||||
};
|
|
||||||
var content = json.ToString();
|
|
||||||
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
|
@ -318,45 +271,13 @@ namespace SafeExamBrowser.Server
|
||||||
logger.Error("Failed to send raise hand notification!");
|
logger.Error("Failed to send raise hand notification!");
|
||||||
}
|
}
|
||||||
|
|
||||||
return new ServerResponse(success, response.ToLogString());
|
return new ServerResponse(success, message);
|
||||||
}
|
|
||||||
|
|
||||||
public ServerResponse LockScreen(string message = null)
|
|
||||||
{
|
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
|
||||||
var contentType = "application/json;charset=UTF-8";
|
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
var json = new JObject
|
|
||||||
{
|
|
||||||
["type"] = "NOTIFICATION",
|
|
||||||
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
|
||||||
["numericValue"] = currentLockScreenId = ++notificationId,
|
|
||||||
["text"] = $"<lockscreen> {message}"
|
|
||||||
};
|
|
||||||
var content = json.ToString();
|
|
||||||
var success = TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
|
|
||||||
if (success)
|
|
||||||
{
|
|
||||||
logger.Info("Successfully sent lock screen notification.");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
logger.Error("Failed to send lock screen notification!");
|
|
||||||
}
|
|
||||||
|
|
||||||
return new ServerResponse(success, response.ToLogString());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public ServerResponse SendSelectedExam(Exam exam)
|
public ServerResponse SendSelectedExam(Exam exam)
|
||||||
{
|
{
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
var request = new SelectExamRequest(api, httpClient, logger, parser, settings);
|
||||||
var content = $"examId={exam.Id}";
|
var success = request.TryExecute(exam, out var message, out var salt);
|
||||||
var contentType = "application/x-www-form-urlencoded";
|
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
|
|
||||||
var success = TryExecute(new HttpMethod("PATCH"), api.HandshakeEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
var message = response.ToLogString();
|
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
|
@ -367,7 +288,7 @@ namespace SafeExamBrowser.Server
|
||||||
logger.Error("Failed to send selected exam!");
|
logger.Error("Failed to send selected exam!");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (parser.TryParseAppSignatureKeySalt(response, out var salt))
|
if (success && salt != default)
|
||||||
{
|
{
|
||||||
logger.Info("App signature key salt detected, performing key exchange...");
|
logger.Info("App signature key salt detected, performing key exchange...");
|
||||||
success = TrySendAppSignatureKey(out message);
|
success = TrySendAppSignatureKey(out message);
|
||||||
|
@ -382,13 +303,8 @@ namespace SafeExamBrowser.Server
|
||||||
|
|
||||||
public ServerResponse SendSessionIdentifier(string identifier)
|
public ServerResponse SendSessionIdentifier(string identifier)
|
||||||
{
|
{
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
var request = new SessionIdentifierRequest(api, httpClient, logger, parser, settings);
|
||||||
var content = $"examId={examId}&seb_user_session_id={identifier}";
|
var success = request.TryExecute(examId, identifier, out var message);
|
||||||
var contentType = "application/x-www-form-urlencoded";
|
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
|
|
||||||
var success = TryExecute(HttpMethod.Put, api.HandshakeEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
var message = response.ToLogString();
|
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
|
@ -453,24 +369,11 @@ namespace SafeExamBrowser.Server
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var contentType = "application/json;charset=UTF-8";
|
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
|
|
||||||
while (!logContent.IsEmpty)
|
while (!logContent.IsEmpty)
|
||||||
{
|
{
|
||||||
if (logContent.TryDequeue(out var c) && c is ILogMessage message)
|
if (logContent.TryDequeue(out var c) && c is ILogMessage message)
|
||||||
{
|
{
|
||||||
// IMPORTANT: The token needs to be read for every request, as it may get updated by another thread!
|
new LogRequest(api, httpClient, logger, parser, settings).TryExecute(message);
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
|
||||||
var json = new JObject
|
|
||||||
{
|
|
||||||
["type"] = message.Severity.ToLogType(),
|
|
||||||
["timestamp"] = message.DateTime.ToUnixTimestamp(),
|
|
||||||
["text"] = message.Message
|
|
||||||
};
|
|
||||||
var content = json.ToString();
|
|
||||||
|
|
||||||
TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -486,43 +389,16 @@ namespace SafeExamBrowser.Server
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
instructionConfirmations.TryDequeue(out var confirmation);
|
||||||
var content = $"timestamp={DateTime.Now.ToUnixTimestamp()}&ping-number={++pingNumber}";
|
|
||||||
var contentType = "application/x-www-form-urlencoded";
|
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
|
|
||||||
if (instructionConfirmations.TryDequeue(out var confirmation))
|
var request = new PingRequest(api, httpClient, logger, parser, settings);
|
||||||
{
|
var success = request.TryExecute(++pingNumber, out var content, out var message, confirmation);
|
||||||
content = $"{content}&instruction-confirm={confirmation}";
|
|
||||||
}
|
|
||||||
|
|
||||||
var success = TryExecute(HttpMethod.Post, api.PingEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
if (parser.TryParseInstruction(response.Content, out var attributes, out var instruction, out var instructionConfirmation))
|
if (parser.TryParseInstruction(content, out var attributes, out var instruction, out var instructionConfirmation))
|
||||||
{
|
{
|
||||||
switch (instruction)
|
HandleInstruction(attributes, instruction);
|
||||||
{
|
|
||||||
case Instructions.LOCK_SCREEN:
|
|
||||||
Task.Run(() => LockScreenRequested?.Invoke(attributes.Message));
|
|
||||||
break;
|
|
||||||
case Instructions.NOTIFICATION_CONFIRM when attributes.Type == "lockscreen":
|
|
||||||
Task.Run(() => LockScreenConfirmed?.Invoke());
|
|
||||||
break;
|
|
||||||
case Instructions.NOTIFICATION_CONFIRM when attributes.Type == "raisehand":
|
|
||||||
Task.Run(() => HandConfirmed?.Invoke());
|
|
||||||
break;
|
|
||||||
case Instructions.PROCTORING:
|
|
||||||
Task.Run(() => ProctoringInstructionReceived?.Invoke(attributes.Instruction));
|
|
||||||
break;
|
|
||||||
case Instructions.PROCTORING_RECONFIGURATION:
|
|
||||||
Task.Run(() => ProctoringConfigurationReceived?.Invoke(attributes.AllowChat, attributes.ReceiveAudio, attributes.ReceiveVideo));
|
|
||||||
break;
|
|
||||||
case Instructions.QUIT:
|
|
||||||
Task.Run(() => TerminationRequested?.Invoke());
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (instructionConfirmation != default)
|
if (instructionConfirmation != default)
|
||||||
{
|
{
|
||||||
|
@ -532,7 +408,7 @@ namespace SafeExamBrowser.Server
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
logger.Error($"Failed to send ping: {response.ToLogString()}");
|
logger.Error($"Failed to send ping: {message}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
|
@ -547,23 +423,25 @@ namespace SafeExamBrowser.Server
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var value = Convert.ToInt32(status.BatteryCharge * 100);
|
|
||||||
var connected = status.IsOnline;
|
var connected = status.IsOnline;
|
||||||
|
var value = Convert.ToInt32(status.BatteryCharge * 100);
|
||||||
|
var text = default(string);
|
||||||
|
|
||||||
if (value != currentPowerSupplyValue)
|
if (value != currentPowerSupplyValue)
|
||||||
{
|
{
|
||||||
var chargeInfo = $"{status.BatteryChargeStatus} at {value}%";
|
var chargeInfo = $"{status.BatteryChargeStatus} at {value}%";
|
||||||
var gridInfo = $"{(status.IsOnline ? "connected to" : "disconnected from")} the power grid";
|
var gridInfo = $"{(status.IsOnline ? "connected to" : "disconnected from")} the power grid";
|
||||||
var text = $"<battery> {chargeInfo}, {status.BatteryTimeRemaining} remaining, {gridInfo}";
|
|
||||||
SendPowerSupplyStatus(text, value);
|
|
||||||
currentPowerSupplyValue = value;
|
currentPowerSupplyValue = value;
|
||||||
|
text = $"<battery> {chargeInfo}, {status.BatteryTimeRemaining} remaining, {gridInfo}";
|
||||||
}
|
}
|
||||||
else if (connected != connectedToPowergrid)
|
else if (connected != connectedToPowergrid)
|
||||||
{
|
{
|
||||||
var text = $"<battery> Device has been {(connected ? "connected to" : "disconnected from")} power grid";
|
|
||||||
SendPowerSupplyStatus(text, value);
|
|
||||||
connectedToPowergrid = connected;
|
connectedToPowergrid = connected;
|
||||||
|
text = $"<battery> Device has been {(connected ? "connected to" : "disconnected from")} power grid";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
new PowerSupplyRequest(api, httpClient, logger, parser, settings).TryExecute(text, value);
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
|
@ -571,23 +449,6 @@ namespace SafeExamBrowser.Server
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void SendPowerSupplyStatus(string text, int value)
|
|
||||||
{
|
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
|
||||||
var contentType = "application/json;charset=UTF-8";
|
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
var json = new JObject
|
|
||||||
{
|
|
||||||
["type"] = LogLevel.Info.ToLogType(),
|
|
||||||
["timestamp"] = DateTime.Now.ToUnixTimestamp(),
|
|
||||||
["text"] = text,
|
|
||||||
["numericValue"] = value
|
|
||||||
};
|
|
||||||
var content = json.ToString();
|
|
||||||
|
|
||||||
TryExecute(HttpMethod.Post, api.LogEndpoint, out _, content, contentType, authorization, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void NetworkAdapter_Changed()
|
private void NetworkAdapter_Changed()
|
||||||
{
|
{
|
||||||
const int NOT_CONNECTED = -1;
|
const int NOT_CONNECTED = -1;
|
||||||
|
@ -598,23 +459,20 @@ namespace SafeExamBrowser.Server
|
||||||
|
|
||||||
if (network?.SignalStrength != currentWlanValue)
|
if (network?.SignalStrength != currentWlanValue)
|
||||||
{
|
{
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
var text = default(string);
|
||||||
var contentType = "application/json;charset=UTF-8";
|
var value = default(int?);
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
var json = new JObject { ["type"] = LogLevel.Info.ToLogType(), ["timestamp"] = DateTime.Now.ToUnixTimestamp() };
|
|
||||||
|
|
||||||
if (network != default(IWirelessNetwork))
|
if (network != default(IWirelessNetwork))
|
||||||
{
|
{
|
||||||
json["text"] = $"<wlan> {network.Name}: {network.Status}, {network.SignalStrength}%";
|
text = $"<wlan> {network.Name}: {network.Status}, {network.SignalStrength}%";
|
||||||
json["numericValue"] = network.SignalStrength;
|
value = network.SignalStrength;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
json["text"] = "<wlan> not connected";
|
text = "<wlan> not connected";
|
||||||
}
|
}
|
||||||
|
|
||||||
TryExecute(HttpMethod.Post, api.LogEndpoint, out var response, json.ToString(), contentType, authorization, token);
|
new NetworkAdapterRequest(api, httpClient, logger, parser, settings).TryExecute(text, value);
|
||||||
|
|
||||||
currentWlanValue = network?.SignalStrength ?? NOT_CONNECTED;
|
currentWlanValue = network?.SignalStrength ?? NOT_CONNECTED;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -624,26 +482,29 @@ namespace SafeExamBrowser.Server
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool TryRetrieveOAuth2Token(out string message)
|
private void HandleInstruction(Attributes attributes, string instruction)
|
||||||
{
|
{
|
||||||
var secret = Convert.ToBase64String(Encoding.UTF8.GetBytes($"{settings.ClientName}:{settings.ClientSecret}"));
|
switch (instruction)
|
||||||
var authorization = ("Authorization", $"Basic {secret}");
|
|
||||||
var content = "grant_type=client_credentials&scope=read write";
|
|
||||||
var contentType = "application/x-www-form-urlencoded";
|
|
||||||
var success = TryExecute(HttpMethod.Post, api.AccessTokenEndpoint, out var response, content, contentType, authorization);
|
|
||||||
|
|
||||||
message = response.ToLogString();
|
|
||||||
|
|
||||||
if (success && parser.TryParseOauth2Token(response.Content, out oauth2Token))
|
|
||||||
{
|
{
|
||||||
logger.Info("Successfully retrieved OAuth2 token.");
|
case Instructions.LOCK_SCREEN:
|
||||||
|
Task.Run(() => LockScreenRequested?.Invoke(attributes.Message));
|
||||||
|
break;
|
||||||
|
case Instructions.NOTIFICATION_CONFIRM when attributes.Type == AttributeType.LockScreen:
|
||||||
|
Task.Run(() => LockScreenConfirmed?.Invoke());
|
||||||
|
break;
|
||||||
|
case Instructions.NOTIFICATION_CONFIRM when attributes.Type == AttributeType.Hand:
|
||||||
|
Task.Run(() => HandConfirmed?.Invoke());
|
||||||
|
break;
|
||||||
|
case Instructions.PROCTORING:
|
||||||
|
Task.Run(() => ProctoringInstructionReceived?.Invoke(attributes.Instruction));
|
||||||
|
break;
|
||||||
|
case Instructions.PROCTORING_RECONFIGURATION:
|
||||||
|
Task.Run(() => ProctoringConfigurationReceived?.Invoke(attributes.AllowChat, attributes.ReceiveAudio, attributes.ReceiveVideo));
|
||||||
|
break;
|
||||||
|
case Instructions.QUIT:
|
||||||
|
Task.Run(() => TerminationRequested?.Invoke());
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
logger.Error("Failed to retrieve OAuth2 token!");
|
|
||||||
}
|
|
||||||
|
|
||||||
return success;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool TrySendAppSignatureKey(out string message)
|
private bool TrySendAppSignatureKey(out string message)
|
||||||
|
@ -651,13 +512,8 @@ namespace SafeExamBrowser.Server
|
||||||
// TODO:
|
// TODO:
|
||||||
// keyGenerator.CalculateAppSignatureKey(configurationKey, server.AppSignatureKeySalt)
|
// keyGenerator.CalculateAppSignatureKey(configurationKey, server.AppSignatureKeySalt)
|
||||||
|
|
||||||
var authorization = ("Authorization", $"Bearer {oauth2Token}");
|
var request = new AppSignatureKeyRequest(api, httpClient, logger, parser, settings);
|
||||||
var content = $"seb_signature_key={"WINDOWS-TEST-ASK-1234"}";
|
var success = request.TryExecute(out message);
|
||||||
var contentType = "application/x-www-form-urlencoded";
|
|
||||||
var token = ("SEBConnectionToken", connectionToken);
|
|
||||||
var success = TryExecute(new HttpMethod("PATCH"), api.HandshakeEndpoint, out var response, content, contentType, authorization, token);
|
|
||||||
|
|
||||||
message = response.ToLogString();
|
|
||||||
|
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
|
@ -670,100 +526,5 @@ namespace SafeExamBrowser.Server
|
||||||
|
|
||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool TryExecute(
|
|
||||||
HttpMethod method,
|
|
||||||
string url,
|
|
||||||
out HttpResponseMessage response,
|
|
||||||
string content = default,
|
|
||||||
string contentType = default,
|
|
||||||
params (string name, string value)[] headers)
|
|
||||||
{
|
|
||||||
response = default;
|
|
||||||
|
|
||||||
for (var attempt = 0; attempt < settings.RequestAttempts && (response == default || !response.IsSuccessStatusCode); attempt++)
|
|
||||||
{
|
|
||||||
var request = BuildRequest(method, url, content, contentType, headers);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
response = httpClient.SendAsync(request).GetAwaiter().GetResult();
|
|
||||||
|
|
||||||
if (request.RequestUri.AbsolutePath != api.LogEndpoint && request.RequestUri.AbsolutePath != api.PingEndpoint)
|
|
||||||
{
|
|
||||||
logger.Debug($"Completed request: {request.Method} '{request.RequestUri}' -> {response.ToLogString()}");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (response.StatusCode == HttpStatusCode.Unauthorized && parser.IsTokenExpired(response.Content))
|
|
||||||
{
|
|
||||||
logger.Info("OAuth2 token has expired, attempting to retrieve new one...");
|
|
||||||
|
|
||||||
if (TryRetrieveOAuth2Token(out var message))
|
|
||||||
{
|
|
||||||
headers = UpdateOAuth2Token(headers);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (TaskCanceledException)
|
|
||||||
{
|
|
||||||
logger.Debug($"Request {request.Method} '{request.RequestUri}' did not complete within {settings.RequestTimeout}ms!");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
logger.Debug($"Request {request.Method} '{request.RequestUri}' failed due to {e}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return response != default && response.IsSuccessStatusCode;
|
|
||||||
}
|
|
||||||
|
|
||||||
private HttpRequestMessage BuildRequest(
|
|
||||||
HttpMethod method,
|
|
||||||
string url,
|
|
||||||
string content = default,
|
|
||||||
string contentType = default,
|
|
||||||
params (string name, string value)[] headers)
|
|
||||||
{
|
|
||||||
var request = new HttpRequestMessage(method, url);
|
|
||||||
|
|
||||||
if (content != default)
|
|
||||||
{
|
|
||||||
request.Content = new StringContent(content, Encoding.UTF8);
|
|
||||||
|
|
||||||
if (contentType != default)
|
|
||||||
{
|
|
||||||
request.Content.Headers.ContentType = MediaTypeHeaderValue.Parse(contentType);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
request.Headers.Add("Accept", "application/json, */*");
|
|
||||||
|
|
||||||
foreach (var (name, value) in headers)
|
|
||||||
{
|
|
||||||
request.Headers.Add(name, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
return request;
|
|
||||||
}
|
|
||||||
|
|
||||||
private (string name, string value)[] UpdateOAuth2Token((string name, string value)[] headers)
|
|
||||||
{
|
|
||||||
var result = new List<(string name, string value)>();
|
|
||||||
|
|
||||||
foreach (var header in headers)
|
|
||||||
{
|
|
||||||
if (header.name == "Authorization")
|
|
||||||
{
|
|
||||||
result.Add(("Authorization", $"Bearer {oauth2Token}"));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
result.Add(header);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result.ToArray();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue