SEBWIN-608: Refactored server proxy by extracting request implementations.

This commit is contained in:
Damian Büchel 2023-02-24 21:33:26 +01:00
parent ae3755df84
commit da458bcfb0
26 changed files with 1085 additions and 364 deletions

View file

@ -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)
{ {

View file

@ -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);
} }
} }

View 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
}
}

View file

@ -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()
{ {

View file

@ -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;
}
} }
} }

View 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;
}
}
}

View 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;
}
}
}

View 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;
}
}
}

View 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();
}
}
}

View 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;
}
}
}

View 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";
}
}

View 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;
}
}
}

View 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;
}
}
}

View 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";
}
}

View 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;
}
}
}

View 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;
}
}
}

View 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;
}
}
}

View 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;
}
}
}

View 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);
}
}
}

View 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;
}
}
}

View 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;
}
}
}

View 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;
}
}
}

View 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;
}
}
}

View 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;
}
}
}

View file

@ -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>

View file

@ -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();
}
} }
} }