/* * Copyright (c) 2019 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.Linq; using SafeExamBrowser.Client.Operations.Events; using SafeExamBrowser.Core.Contracts.OperationModel; using SafeExamBrowser.Core.Contracts.OperationModel.Events; using SafeExamBrowser.I18n.Contracts; using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.Monitoring.Contracts.Applications; using SafeExamBrowser.Settings; using SafeExamBrowser.Settings.Applications; namespace SafeExamBrowser.Client.Operations { internal class ApplicationOperation : ClientOperation { private ILogger logger; private IApplicationMonitor applicationMonitor; public override event ActionRequiredEventHandler ActionRequired; public override event StatusChangedEventHandler StatusChanged; public ApplicationOperation(IApplicationMonitor applicationMonitor, ClientContext context, ILogger logger) : base(context) { this.applicationMonitor = applicationMonitor; this.logger = logger; } public override OperationResult Perform() { logger.Info("Initializing applications..."); StatusChanged?.Invoke(TextKey.OperationStatus_InitializeApplications); var result = InitializeApplications(); if (result == OperationResult.Success) { StartMonitor(); } return result; } public override OperationResult Revert() { logger.Info("Finalizing applications..."); StatusChanged?.Invoke(TextKey.OperationStatus_FinalizeApplications); FinalizeApplications(); StopMonitor(); return OperationResult.Success; } private OperationResult InitializeApplications() { var initialization = applicationMonitor.Initialize(Context.Settings.Applications); var result = OperationResult.Success; if (initialization.FailedAutoTerminations.Any()) { result = HandleAutoTerminationFailure(initialization.FailedAutoTerminations); } else if (initialization.RunningApplications.Any()) { result = TryTerminate(initialization.RunningApplications); } if (result == OperationResult.Success) { CreateApplications(); } return result; } private void CreateApplications() { foreach (var application in Context.Settings.Applications.Whitelist) { Create(application); } } private void Create(WhitelistApplication application) { // TODO: Use IApplicationFactory to create new application according to configuration, load into Context.Applications // StatusChanged?.Invoke(); } private void FinalizeApplications() { } private OperationResult HandleAutoTerminationFailure(IList applications) { logger.Error($"{applications.Count} application(s) could not be automatically terminated: {string.Join(", ", applications.Select(a => a.Name))}"); ActionRequired?.Invoke(new ApplicationTerminationFailedEventArgs(applications)); return OperationResult.Failed; } private void StartMonitor() { if (Context.Settings.KioskMode != KioskMode.None) { applicationMonitor.Start(); } } private void StopMonitor() { if (Context.Settings.KioskMode != KioskMode.None) { applicationMonitor.Stop(); } } private OperationResult TryTerminate(IEnumerable runningApplications) { var args = new ApplicationTerminationEventArgs(runningApplications); var failed = new List(); var result = OperationResult.Success; logger.Info($"The following applications need to be terminated: {string.Join(", ", runningApplications.Select(a => a.Name))}."); ActionRequired?.Invoke(args); if (args.TerminateProcesses) { logger.Info($"The user chose to automatically terminate all running applications."); foreach (var application in runningApplications) { var success = applicationMonitor.TryTerminate(application); if (success) { logger.Info($"Successfully terminated application '{application.Name}'."); } else { result = OperationResult.Failed; failed.Add(application); logger.Error($"Failed to automatically terminate application '{application.Name}'!"); } } } else { logger.Info("The user chose not to automatically terminate all running applications. Aborting..."); result = OperationResult.Aborted; } if (failed.Any()) { ActionRequired?.Invoke(new ApplicationTerminationFailedEventArgs(failed)); } return result; } } }