2021-04-12 10:59:31 +02:00
|
|
|
|
/*
|
2022-01-21 16:33:52 +01:00
|
|
|
|
* Copyright (c) 2022 ETH Zürich, Educational Development and Technology (LET)
|
2021-04-12 10:59:31 +02:00
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Net.Http;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Newtonsoft.Json;
|
|
|
|
|
using Newtonsoft.Json.Linq;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
|
|
|
|
using SafeExamBrowser.Server.Contracts.Data;
|
|
|
|
|
using SafeExamBrowser.Server.Data;
|
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Server
|
|
|
|
|
{
|
|
|
|
|
internal class Parser
|
|
|
|
|
{
|
|
|
|
|
private readonly ILogger logger;
|
|
|
|
|
|
|
|
|
|
internal Parser(ILogger logger)
|
|
|
|
|
{
|
|
|
|
|
this.logger = logger;
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-27 15:26:42 +02:00
|
|
|
|
internal bool IsTokenExpired(HttpContent content)
|
|
|
|
|
{
|
|
|
|
|
var isExpired = false;
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var json = JsonConvert.DeserializeObject(Extract(content)) as JObject;
|
|
|
|
|
var error = json["error"].Value<string>();
|
|
|
|
|
|
|
|
|
|
isExpired = error?.Equals("invalid_token", StringComparison.OrdinalIgnoreCase) == true;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to parse token expiration content!", e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return isExpired;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
internal bool TryParseApi(HttpContent content, out ApiVersion1 api)
|
|
|
|
|
{
|
|
|
|
|
var success = false;
|
|
|
|
|
|
|
|
|
|
api = new ApiVersion1();
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var json = JsonConvert.DeserializeObject(Extract(content)) as JObject;
|
|
|
|
|
var apisJson = json["api-versions"];
|
|
|
|
|
|
|
|
|
|
foreach (var apiJson in apisJson.AsJEnumerable())
|
|
|
|
|
{
|
|
|
|
|
if (apiJson["name"].Value<string>().Equals("v1"))
|
|
|
|
|
{
|
|
|
|
|
foreach (var endpoint in apiJson["endpoints"].AsJEnumerable())
|
|
|
|
|
{
|
|
|
|
|
var name = endpoint["name"].Value<string>();
|
|
|
|
|
var location = endpoint["location"].Value<string>();
|
|
|
|
|
|
|
|
|
|
switch (name)
|
|
|
|
|
{
|
|
|
|
|
case "access-token-endpoint":
|
|
|
|
|
api.AccessTokenEndpoint = location;
|
|
|
|
|
break;
|
|
|
|
|
case "seb-configuration-endpoint":
|
|
|
|
|
api.ConfigurationEndpoint = location;
|
|
|
|
|
break;
|
|
|
|
|
case "seb-handshake-endpoint":
|
|
|
|
|
api.HandshakeEndpoint = location;
|
|
|
|
|
break;
|
|
|
|
|
case "seb-log-endpoint":
|
|
|
|
|
api.LogEndpoint = location;
|
|
|
|
|
break;
|
|
|
|
|
case "seb-ping-endpoint":
|
|
|
|
|
api.PingEndpoint = location;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
success = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("The selected SEB server instance does not support the required API version!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to parse server API!", e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return success;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal bool TryParseConnectionToken(HttpResponseMessage response, out string connectionToken)
|
|
|
|
|
{
|
2022-07-27 15:26:42 +02:00
|
|
|
|
connectionToken = default;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var hasHeader = response.Headers.TryGetValues("SEBConnectionToken", out var values);
|
|
|
|
|
|
|
|
|
|
if (hasHeader)
|
|
|
|
|
{
|
|
|
|
|
connectionToken = values.First();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to retrieve connection token!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to parse connection token!", e);
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-27 15:26:42 +02:00
|
|
|
|
return connectionToken != default;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal bool TryParseExams(HttpContent content, out IList<Exam> exams)
|
|
|
|
|
{
|
|
|
|
|
exams = new List<Exam>();
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var json = JsonConvert.DeserializeObject(Extract(content)) as JArray;
|
|
|
|
|
|
|
|
|
|
foreach (var exam in json.AsJEnumerable())
|
|
|
|
|
{
|
|
|
|
|
exams.Add(new Exam
|
|
|
|
|
{
|
|
|
|
|
Id = exam["examId"].Value<string>(),
|
|
|
|
|
LmsName = exam["lmsType"].Value<string>(),
|
|
|
|
|
Name = exam["name"].Value<string>(),
|
|
|
|
|
Url = exam["url"].Value<string>()
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to parse exams!", e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return exams.Any();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal bool TryParseInstruction(HttpContent content, out Attributes attributes, out string instruction, out string instructionConfirmation)
|
|
|
|
|
{
|
|
|
|
|
attributes = new Attributes();
|
2022-07-27 15:26:42 +02:00
|
|
|
|
instruction = default;
|
|
|
|
|
instructionConfirmation = default;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var json = JsonConvert.DeserializeObject(Extract(content)) as JObject;
|
|
|
|
|
|
|
|
|
|
if (json != default(JObject))
|
|
|
|
|
{
|
|
|
|
|
instruction = json["instruction"].Value<string>();
|
|
|
|
|
|
|
|
|
|
if (json.ContainsKey("attributes"))
|
|
|
|
|
{
|
|
|
|
|
var attributesJson = json["attributes"] as JObject;
|
|
|
|
|
|
|
|
|
|
if (attributesJson.ContainsKey("instruction-confirm"))
|
|
|
|
|
{
|
|
|
|
|
instructionConfirmation = attributesJson["instruction-confirm"].Value<string>();
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-31 14:11:19 +02:00
|
|
|
|
attributes = ParseAttributes(attributesJson, instruction);
|
2021-04-12 10:59:31 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to parse instruction!", e);
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-27 15:26:42 +02:00
|
|
|
|
return instruction != default;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal bool TryParseOauth2Token(HttpContent content, out string oauth2Token)
|
|
|
|
|
{
|
|
|
|
|
oauth2Token = default;
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var json = JsonConvert.DeserializeObject(Extract(content)) as JObject;
|
|
|
|
|
|
|
|
|
|
oauth2Token = json["access_token"].Value<string>();
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to parse Oauth2 token!", e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return oauth2Token != default;
|
2021-04-12 10:59:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-31 14:11:19 +02:00
|
|
|
|
private Attributes ParseAttributes(JObject attributesJson, string instruction)
|
2021-06-16 15:38:55 +02:00
|
|
|
|
{
|
|
|
|
|
var attributes = new Attributes();
|
2022-07-27 15:26:42 +02:00
|
|
|
|
|
2021-06-16 15:38:55 +02:00
|
|
|
|
switch (instruction)
|
|
|
|
|
{
|
2022-09-02 14:56:49 +02:00
|
|
|
|
case Instructions.LOCK_SCREEN:
|
|
|
|
|
ParseLockScreenInstruction(attributes, attributesJson);
|
|
|
|
|
break;
|
2021-09-17 10:47:02 +02:00
|
|
|
|
case Instructions.NOTIFICATION_CONFIRM:
|
|
|
|
|
ParseNotificationConfirmation(attributes, attributesJson);
|
|
|
|
|
break;
|
2021-06-16 15:38:55 +02:00
|
|
|
|
case Instructions.PROCTORING:
|
|
|
|
|
ParseProctoringInstruction(attributes, attributesJson);
|
|
|
|
|
break;
|
|
|
|
|
case Instructions.PROCTORING_RECONFIGURATION:
|
|
|
|
|
ParseReconfigurationInstruction(attributes, attributesJson);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return attributes;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-31 14:11:19 +02:00
|
|
|
|
private void ParseLockScreenInstruction(Attributes attributes, JObject attributesJson)
|
|
|
|
|
{
|
|
|
|
|
if (attributesJson.ContainsKey("message"))
|
|
|
|
|
{
|
|
|
|
|
attributes.Message = attributesJson["message"].Value<string>();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-17 10:47:02 +02:00
|
|
|
|
private void ParseNotificationConfirmation(Attributes attributes, JObject attributesJson)
|
|
|
|
|
{
|
|
|
|
|
if (attributesJson.ContainsKey("id"))
|
|
|
|
|
{
|
|
|
|
|
attributes.Id = attributesJson["id"].Value<int>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (attributesJson.ContainsKey("type"))
|
|
|
|
|
{
|
|
|
|
|
attributes.Type = attributesJson["type"].Value<string>();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-16 15:38:55 +02:00
|
|
|
|
private void ParseProctoringInstruction(Attributes attributes, JObject attributesJson)
|
|
|
|
|
{
|
|
|
|
|
var provider = attributesJson["service-type"].Value<string>();
|
|
|
|
|
|
|
|
|
|
switch (provider)
|
|
|
|
|
{
|
|
|
|
|
case "JITSI_MEET":
|
|
|
|
|
attributes.Instruction.JitsiMeetRoomName = attributesJson["jitsiMeetRoom"].Value<string>();
|
|
|
|
|
attributes.Instruction.JitsiMeetServerUrl = attributesJson["jitsiMeetServerURL"].Value<string>();
|
|
|
|
|
attributes.Instruction.JitsiMeetToken = attributesJson["jitsiMeetToken"].Value<string>();
|
|
|
|
|
break;
|
|
|
|
|
case "ZOOM":
|
|
|
|
|
attributes.Instruction.ZoomApiKey = attributesJson["zoomAPIKey"].Value<string>();
|
|
|
|
|
attributes.Instruction.ZoomMeetingNumber = attributesJson["zoomRoom"].Value<string>();
|
|
|
|
|
attributes.Instruction.ZoomPassword = attributesJson["zoomMeetingKey"].Value<string>();
|
|
|
|
|
attributes.Instruction.ZoomSignature = attributesJson["zoomToken"].Value<string>();
|
2021-06-19 21:04:13 +02:00
|
|
|
|
attributes.Instruction.ZoomSubject = attributesJson["zoomSubject"].Value<string>();
|
2021-06-16 15:38:55 +02:00
|
|
|
|
attributes.Instruction.ZoomUserName = attributesJson["zoomUserName"].Value<string>();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ParseReconfigurationInstruction(Attributes attributes, JObject attributesJson)
|
|
|
|
|
{
|
|
|
|
|
if (attributesJson.ContainsKey("jitsiMeetFeatureFlagChat"))
|
|
|
|
|
{
|
|
|
|
|
attributes.AllowChat = attributesJson["jitsiMeetFeatureFlagChat"].Value<bool>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (attributesJson.ContainsKey("zoomFeatureFlagChat"))
|
|
|
|
|
{
|
|
|
|
|
attributes.AllowChat = attributesJson["zoomFeatureFlagChat"].Value<bool>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (attributesJson.ContainsKey("jitsiMeetReceiveAudio"))
|
|
|
|
|
{
|
|
|
|
|
attributes.ReceiveAudio = attributesJson["jitsiMeetReceiveAudio"].Value<bool>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (attributesJson.ContainsKey("zoomReceiveAudio"))
|
|
|
|
|
{
|
|
|
|
|
attributes.ReceiveAudio = attributesJson["zoomReceiveAudio"].Value<bool>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (attributesJson.ContainsKey("jitsiMeetReceiveVideo"))
|
|
|
|
|
{
|
|
|
|
|
attributes.ReceiveVideo = attributesJson["jitsiMeetReceiveVideo"].Value<bool>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (attributesJson.ContainsKey("zoomReceiveVideo"))
|
|
|
|
|
{
|
|
|
|
|
attributes.ReceiveVideo = attributesJson["zoomReceiveVideo"].Value<bool>();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-12 10:59:31 +02:00
|
|
|
|
private string Extract(HttpContent content)
|
|
|
|
|
{
|
|
|
|
|
var task = Task.Run(async () =>
|
|
|
|
|
{
|
|
|
|
|
return await content.ReadAsStreamAsync();
|
|
|
|
|
});
|
|
|
|
|
var stream = task.GetAwaiter().GetResult();
|
|
|
|
|
var reader = new StreamReader(stream);
|
|
|
|
|
|
|
|
|
|
return reader.ReadToEnd();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|