/* * Copyright (c) 2021 ETH Zürich, Educational Development and Technology (LET) * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ using System; using System.Collections.Generic; using System.Linq; using SafeExamBrowser.Configuration.Contracts; using SafeExamBrowser.Core.Contracts.OperationModel; using SafeExamBrowser.Core.Contracts.OperationModel.Events; using SafeExamBrowser.I18n.Contracts; using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Runtime.Operations.Events; using SafeExamBrowser.Server.Contracts; using SafeExamBrowser.Server.Contracts.Data; using SafeExamBrowser.Settings; using SafeExamBrowser.SystemComponents.Contracts; namespace SafeExamBrowser.Runtime.Operations { internal class ServerOperation : ConfigurationBaseOperation { private readonly IFileSystem fileSystem; private readonly ILogger logger; private readonly IServerProxy server; public override event ActionRequiredEventHandler ActionRequired; public override event StatusChangedEventHandler StatusChanged; public ServerOperation( string[] commandLineArgs, IConfigurationRepository configuration, IFileSystem fileSystem, ILogger logger, SessionContext context, IServerProxy server) : base(commandLineArgs, configuration, context) { this.fileSystem = fileSystem; this.logger = logger; this.server = server; } public override OperationResult Perform() { var result = OperationResult.Success; if (Context.Next.Settings.SessionMode == SessionMode.Server) { logger.Info("Initializing server..."); StatusChanged?.Invoke(TextKey.OperationStatus_InitializeServer); server.Initialize(Context.Next.Settings.Server); var (abort, fallback, success) = TryPerformWithFallback(() => server.Connect()); if (success) { (abort, fallback, success) = TryPerformWithFallback(() => server.GetAvailableExams(Context.Next.Settings.Server.ExamId), out var exams); if (success) { var exam = default(Exam); if (!string.IsNullOrWhiteSpace(Context.Next.Settings.Server.ExamId)) { exam = exams.First(); logger.Info("Automatically selected exam as defined in configuration."); } else { success = TrySelectExam(exams, out exam); } if (success) { (abort, fallback, success) = TryPerformWithFallback(() => server.GetConfigurationFor(exam), out var uri); if (success) { var info = server.GetConnectionInfo(); var status = TryLoadSettings(uri, UriSource.Server, out _, out var settings); fileSystem.Delete(uri.LocalPath); if (status == LoadStatus.Success) { var serverSettings = Context.Next.Settings.Server; Context.Next.AppConfig.ServerApi = info.Api; Context.Next.AppConfig.ServerConnectionToken = info.ConnectionToken; Context.Next.AppConfig.ServerExamId = exam.Id; Context.Next.AppConfig.ServerOauth2Token = info.Oauth2Token; Context.Next.Settings = settings; Context.Next.Settings.Browser.StartUrl = exam.Url; Context.Next.Settings.Server = serverSettings; Context.Next.Settings.SessionMode = SessionMode.Server; result = OperationResult.Success; } else { result = OperationResult.Failed; } } } else { logger.Info("The user aborted the exam selection."); result = OperationResult.Aborted; } } } if (abort) { result = OperationResult.Aborted; } if (fallback) { Context.Next.Settings.SessionMode = SessionMode.Normal; result = OperationResult.Success; } } return result; } public override OperationResult Repeat() { if (Context.Current.Settings.SessionMode == SessionMode.Server) { logger.Info("Initializing server configuration for next session..."); Context.Next.AppConfig.ServerApi = Context.Current.AppConfig.ServerApi; Context.Next.AppConfig.ServerConnectionToken = Context.Current.AppConfig.ServerConnectionToken; Context.Next.AppConfig.ServerExamId = Context.Current.AppConfig.ServerExamId; Context.Next.AppConfig.ServerOauth2Token = Context.Current.AppConfig.ServerOauth2Token; Context.Next.Settings.Server = Context.Current.Settings.Server; Context.Next.Settings.SessionMode = SessionMode.Server; } else if (Context.Next.Settings.SessionMode == SessionMode.Server) { return Perform(); } return OperationResult.Success; } public override OperationResult Revert() { var result = OperationResult.Success; if (Context.Current?.Settings.SessionMode == SessionMode.Server) { logger.Info("Finalizing server..."); StatusChanged?.Invoke(TextKey.OperationStatus_FinalizeServer); var disconnect = server.Disconnect(); if (disconnect.Success) { result = OperationResult.Success; } else { result = OperationResult.Failed; } } return result; } protected override void InvokeActionRequired(ActionRequiredEventArgs args) { ActionRequired?.Invoke(args); } private (bool abort, bool fallback, bool success) TryPerformWithFallback(Func request) { var abort = false; var fallback = false; var success = false; while (!success) { var response = request(); success = response.Success; if (!success && !Retry(response.Message, out abort, out fallback)) { break; } } return (abort, fallback, success); } private (bool abort, bool fallback, bool success) TryPerformWithFallback(Func> request, out T value) { var abort = false; var fallback = false; var success = false; value = default(T); while (!success) { var response = request(); success = response.Success; value = response.Value; if (!success && !Retry(response.Message, out abort, out fallback)) { break; } } return (abort, fallback, success); } private bool Retry(string message, out bool abort, out bool fallback) { var args = new ServerFailureEventArgs(message, Context.Next.Settings.Server.PerformFallback); ActionRequired?.Invoke(args); abort = args.Abort; fallback = args.Fallback; return args.Retry; } private bool TrySelectExam(IEnumerable exams, out Exam exam) { var args = new ExamSelectionEventArgs(exams); ActionRequired?.Invoke(args); exam = args.SelectedExam; return args.Success; } } }