2019-10-01 11:30:53 +02:00
|
|
|
|
/*
|
|
|
|
|
* 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;
|
2019-10-09 14:04:27 +02:00
|
|
|
|
using SafeExamBrowser.Settings;
|
2019-10-01 11:30:53 +02:00
|
|
|
|
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...");
|
2019-10-04 16:36:12 +02:00
|
|
|
|
StatusChanged?.Invoke(TextKey.OperationStatus_InitializeApplications);
|
2019-10-01 11:30:53 +02:00
|
|
|
|
|
|
|
|
|
var result = InitializeApplications();
|
|
|
|
|
|
|
|
|
|
if (result == OperationResult.Success)
|
|
|
|
|
{
|
|
|
|
|
StartMonitor();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override OperationResult Revert()
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Finalizing applications...");
|
2019-10-04 16:36:12 +02:00
|
|
|
|
StatusChanged?.Invoke(TextKey.OperationStatus_FinalizeApplications);
|
2019-10-01 11:30:53 +02:00
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
FinalizeApplications();
|
2019-10-01 11:30:53 +02:00
|
|
|
|
StopMonitor();
|
|
|
|
|
|
|
|
|
|
return OperationResult.Success;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private OperationResult InitializeApplications()
|
|
|
|
|
{
|
|
|
|
|
var initialization = applicationMonitor.Initialize(Context.Settings.Applications);
|
|
|
|
|
var result = OperationResult.Success;
|
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
if (initialization.FailedAutoTerminations.Any())
|
|
|
|
|
{
|
|
|
|
|
result = HandleAutoTerminationFailure(initialization.FailedAutoTerminations);
|
|
|
|
|
}
|
|
|
|
|
else if (initialization.RunningApplications.Any())
|
2019-10-01 11:30:53 +02:00
|
|
|
|
{
|
|
|
|
|
result = TryTerminate(initialization.RunningApplications);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (result == OperationResult.Success)
|
|
|
|
|
{
|
2019-10-04 16:36:12 +02:00
|
|
|
|
CreateApplications();
|
2019-10-01 11:30:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
private void CreateApplications()
|
|
|
|
|
{
|
|
|
|
|
foreach (var application in Context.Settings.Applications.Whitelist)
|
|
|
|
|
{
|
|
|
|
|
Create(application);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-01 11:30:53 +02:00
|
|
|
|
private void Create(WhitelistApplication application)
|
|
|
|
|
{
|
|
|
|
|
// TODO: Use IApplicationFactory to create new application according to configuration, load into Context.Applications
|
2019-10-04 16:36:12 +02:00
|
|
|
|
// StatusChanged?.Invoke();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void FinalizeApplications()
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private OperationResult HandleAutoTerminationFailure(IList<RunningApplication> 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;
|
2019-10-01 11:30:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void StartMonitor()
|
|
|
|
|
{
|
2019-10-09 14:04:27 +02:00
|
|
|
|
if (Context.Settings.KioskMode != KioskMode.None)
|
|
|
|
|
{
|
2019-10-01 11:30:53 +02:00
|
|
|
|
applicationMonitor.Start();
|
2019-10-09 14:04:27 +02:00
|
|
|
|
}
|
2019-10-01 11:30:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void StopMonitor()
|
|
|
|
|
{
|
2019-10-09 14:04:27 +02:00
|
|
|
|
if (Context.Settings.KioskMode != KioskMode.None)
|
|
|
|
|
{
|
2019-10-01 11:30:53 +02:00
|
|
|
|
applicationMonitor.Stop();
|
2019-10-09 14:04:27 +02:00
|
|
|
|
}
|
2019-10-01 11:30:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
private OperationResult TryTerminate(IEnumerable<RunningApplication> runningApplications)
|
2019-10-01 11:30:53 +02:00
|
|
|
|
{
|
2019-10-04 16:36:12 +02:00
|
|
|
|
var args = new ApplicationTerminationEventArgs(runningApplications);
|
|
|
|
|
var failed = new List<RunningApplication>();
|
2019-10-01 11:30:53 +02:00
|
|
|
|
var result = OperationResult.Success;
|
|
|
|
|
|
2019-10-08 10:03:58 +02:00
|
|
|
|
logger.Info($"The following applications need to be terminated: {string.Join(", ", runningApplications.Select(a => a.Name))}.");
|
2019-10-01 11:30:53 +02:00
|
|
|
|
ActionRequired?.Invoke(args);
|
|
|
|
|
|
|
|
|
|
if (args.TerminateProcesses)
|
|
|
|
|
{
|
2019-10-08 10:03:58 +02:00
|
|
|
|
logger.Info($"The user chose to automatically terminate all running applications.");
|
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
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}'!");
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-10-01 11:30:53 +02:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2019-10-08 10:03:58 +02:00
|
|
|
|
logger.Info("The user chose not to automatically terminate all running applications. Aborting...");
|
2019-10-01 11:30:53 +02:00
|
|
|
|
result = OperationResult.Aborted;
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
if (failed.Any())
|
|
|
|
|
{
|
|
|
|
|
ActionRequired?.Invoke(new ApplicationTerminationFailedEventArgs(failed));
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-01 11:30:53 +02:00
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|