2018-01-17 08:26:44 +01:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2018 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/.
|
|
|
|
|
*/
|
|
|
|
|
|
2018-01-30 14:41:36 +01:00
|
|
|
|
using System;
|
2018-01-24 07:46:22 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Behaviour;
|
2018-02-01 08:37:12 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Behaviour.Operations;
|
2018-01-24 07:46:22 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Communication;
|
2018-01-30 14:41:36 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Configuration;
|
2018-01-18 15:14:05 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.Settings;
|
2018-01-26 12:33:36 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.I18n;
|
2018-01-17 08:26:44 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Logging;
|
2018-01-26 12:33:36 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.UserInterface;
|
2018-01-17 08:26:44 +01:00
|
|
|
|
|
2018-01-18 08:16:20 +01:00
|
|
|
|
namespace SafeExamBrowser.Runtime.Behaviour
|
2018-01-17 08:26:44 +01:00
|
|
|
|
{
|
2018-01-18 08:16:20 +01:00
|
|
|
|
internal class RuntimeController : IRuntimeController
|
2018-01-17 08:26:44 +01:00
|
|
|
|
{
|
2018-02-08 13:32:48 +01:00
|
|
|
|
private bool sessionRunning;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
|
2018-02-14 15:26:05 +01:00
|
|
|
|
private IClientProxy client;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
private IConfigurationRepository configuration;
|
2018-01-17 08:26:44 +01:00
|
|
|
|
private ILogger logger;
|
2018-02-02 09:18:35 +01:00
|
|
|
|
private IOperationSequence bootstrapSequence;
|
|
|
|
|
private IOperationSequence sessionSequence;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
private IRuntimeHost runtimeHost;
|
2018-02-15 15:42:54 +01:00
|
|
|
|
private RuntimeInfo runtimeInfo;
|
2018-01-26 12:33:36 +01:00
|
|
|
|
private IRuntimeWindow runtimeWindow;
|
2018-02-14 15:26:05 +01:00
|
|
|
|
private IServiceProxy service;
|
2018-02-02 09:18:35 +01:00
|
|
|
|
private ISplashScreen splashScreen;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
private Action shutdown;
|
2018-01-30 14:41:36 +01:00
|
|
|
|
private IUserInterfaceFactory uiFactory;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
|
2018-01-24 07:46:22 +01:00
|
|
|
|
public RuntimeController(
|
2018-02-14 15:26:05 +01:00
|
|
|
|
IClientProxy client,
|
2018-02-06 15:12:11 +01:00
|
|
|
|
IConfigurationRepository configuration,
|
2018-01-24 07:46:22 +01:00
|
|
|
|
ILogger logger,
|
2018-02-02 09:18:35 +01:00
|
|
|
|
IOperationSequence bootstrapSequence,
|
|
|
|
|
IOperationSequence sessionSequence,
|
2018-02-06 15:12:11 +01:00
|
|
|
|
IRuntimeHost runtimeHost,
|
2018-02-15 15:42:54 +01:00
|
|
|
|
RuntimeInfo runtimeInfo,
|
2018-02-14 15:26:05 +01:00
|
|
|
|
IServiceProxy service,
|
2018-02-06 15:12:11 +01:00
|
|
|
|
Action shutdown,
|
2018-01-30 14:41:36 +01:00
|
|
|
|
IUserInterfaceFactory uiFactory)
|
2018-01-17 08:26:44 +01:00
|
|
|
|
{
|
2018-02-14 15:26:05 +01:00
|
|
|
|
this.client = client;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
this.configuration = configuration;
|
2018-01-17 08:26:44 +01:00
|
|
|
|
this.logger = logger;
|
2018-02-02 09:18:35 +01:00
|
|
|
|
this.bootstrapSequence = bootstrapSequence;
|
|
|
|
|
this.sessionSequence = sessionSequence;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
this.runtimeHost = runtimeHost;
|
2018-01-30 14:41:36 +01:00
|
|
|
|
this.runtimeInfo = runtimeInfo;
|
2018-02-14 15:26:05 +01:00
|
|
|
|
this.service = service;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
this.shutdown = shutdown;
|
2018-01-30 14:41:36 +01:00
|
|
|
|
this.uiFactory = uiFactory;
|
2018-01-24 07:46:22 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-02-02 09:18:35 +01:00
|
|
|
|
public bool TryStart()
|
2018-01-24 07:46:22 +01:00
|
|
|
|
{
|
2018-02-01 08:37:12 +01:00
|
|
|
|
logger.Info("--- Initiating startup procedure ---");
|
|
|
|
|
|
2018-02-07 13:25:49 +01:00
|
|
|
|
runtimeWindow = uiFactory.CreateRuntimeWindow(runtimeInfo);
|
|
|
|
|
splashScreen = uiFactory.CreateSplashScreen(runtimeInfo);
|
2018-01-24 07:46:22 +01:00
|
|
|
|
|
2018-02-02 09:18:35 +01:00
|
|
|
|
bootstrapSequence.ProgressIndicator = splashScreen;
|
2018-02-07 13:25:49 +01:00
|
|
|
|
sessionSequence.ProgressIndicator = runtimeWindow;
|
|
|
|
|
|
|
|
|
|
splashScreen.Show();
|
2018-02-02 09:18:35 +01:00
|
|
|
|
|
2018-02-08 13:32:48 +01:00
|
|
|
|
var initialized = bootstrapSequence.TryPerform();
|
2018-01-30 14:41:36 +01:00
|
|
|
|
|
2018-02-06 15:12:11 +01:00
|
|
|
|
if (initialized)
|
2018-01-24 07:46:22 +01:00
|
|
|
|
{
|
2018-02-27 15:28:54 +01:00
|
|
|
|
RegisterEvents();
|
|
|
|
|
|
2018-02-22 10:00:18 +01:00
|
|
|
|
logger.Info("--- Application successfully initialized ---");
|
2018-02-01 08:37:12 +01:00
|
|
|
|
logger.Log(string.Empty);
|
2018-01-30 14:41:36 +01:00
|
|
|
|
logger.Subscribe(runtimeWindow);
|
2018-02-02 09:18:35 +01:00
|
|
|
|
splashScreen.Hide();
|
|
|
|
|
|
|
|
|
|
StartSession(true);
|
2018-02-01 08:37:12 +01:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Info("--- Application startup aborted! ---");
|
|
|
|
|
logger.Log(string.Empty);
|
2018-02-27 15:28:54 +01:00
|
|
|
|
|
|
|
|
|
uiFactory.Show(TextKey.MessageBox_StartupError, TextKey.MessageBox_StartupErrorTitle, icon: MessageBoxIcon.Error);
|
2018-01-24 07:46:22 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-02-08 13:32:48 +01:00
|
|
|
|
return initialized && sessionRunning;
|
2018-01-24 07:46:22 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-02-01 08:37:12 +01:00
|
|
|
|
public void Terminate()
|
2018-01-17 08:26:44 +01:00
|
|
|
|
{
|
2018-02-27 15:28:54 +01:00
|
|
|
|
DeregisterEvents();
|
|
|
|
|
|
2018-02-08 13:32:48 +01:00
|
|
|
|
if (sessionRunning)
|
2018-02-06 15:12:11 +01:00
|
|
|
|
{
|
2018-02-27 15:28:54 +01:00
|
|
|
|
DeregisterSessionEvents();
|
2018-02-08 13:32:48 +01:00
|
|
|
|
StopSession();
|
2018-02-06 15:12:11 +01:00
|
|
|
|
}
|
2018-01-26 12:33:36 +01:00
|
|
|
|
|
2018-01-30 14:41:36 +01:00
|
|
|
|
logger.Unsubscribe(runtimeWindow);
|
2018-02-02 09:18:35 +01:00
|
|
|
|
runtimeWindow?.Close();
|
|
|
|
|
splashScreen?.Show();
|
2018-02-21 14:01:21 +01:00
|
|
|
|
splashScreen?.BringToForeground();
|
2018-02-01 08:37:12 +01:00
|
|
|
|
|
|
|
|
|
logger.Log(string.Empty);
|
|
|
|
|
logger.Info("--- Initiating shutdown procedure ---");
|
|
|
|
|
|
2018-02-02 09:18:35 +01:00
|
|
|
|
var success = bootstrapSequence.TryRevert();
|
2018-02-01 08:37:12 +01:00
|
|
|
|
|
|
|
|
|
if (success)
|
|
|
|
|
{
|
2018-02-22 10:00:18 +01:00
|
|
|
|
logger.Info("--- Application successfully finalized ---");
|
2018-02-07 13:25:49 +01:00
|
|
|
|
logger.Log(string.Empty);
|
2018-02-01 08:37:12 +01:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Info("--- Shutdown procedure failed! ---");
|
2018-02-07 13:25:49 +01:00
|
|
|
|
logger.Log(string.Empty);
|
2018-02-27 15:28:54 +01:00
|
|
|
|
|
|
|
|
|
uiFactory.Show(TextKey.MessageBox_ShutdownError, TextKey.MessageBox_ShutdownErrorTitle, icon: MessageBoxIcon.Error);
|
2018-02-01 08:37:12 +01:00
|
|
|
|
}
|
2018-02-02 09:18:35 +01:00
|
|
|
|
|
|
|
|
|
splashScreen?.Close();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void StartSession(bool initial = false)
|
|
|
|
|
{
|
|
|
|
|
runtimeWindow.Show();
|
2018-02-27 15:28:54 +01:00
|
|
|
|
logger.Info(">>>--- Initiating session procedure ---<<<");
|
|
|
|
|
|
|
|
|
|
if (sessionRunning)
|
|
|
|
|
{
|
|
|
|
|
DeregisterSessionEvents();
|
|
|
|
|
}
|
2018-02-02 09:18:35 +01:00
|
|
|
|
|
2018-02-08 13:32:48 +01:00
|
|
|
|
sessionRunning = initial ? sessionSequence.TryPerform() : sessionSequence.TryRepeat();
|
2018-02-02 09:18:35 +01:00
|
|
|
|
|
2018-02-08 13:32:48 +01:00
|
|
|
|
if (sessionRunning)
|
2018-02-02 09:18:35 +01:00
|
|
|
|
{
|
2018-02-27 15:28:54 +01:00
|
|
|
|
RegisterSessionEvents();
|
|
|
|
|
|
|
|
|
|
logger.Info(">>>--- Session is running ---<<<");
|
2018-02-07 13:25:49 +01:00
|
|
|
|
runtimeWindow.HideProgressBar();
|
|
|
|
|
runtimeWindow.UpdateText(TextKey.RuntimeWindow_ApplicationRunning);
|
2018-02-02 09:18:35 +01:00
|
|
|
|
|
2018-02-07 13:25:49 +01:00
|
|
|
|
if (configuration.CurrentSettings.KioskMode == KioskMode.DisableExplorerShell)
|
|
|
|
|
{
|
|
|
|
|
runtimeWindow.Hide();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
2018-02-02 09:18:35 +01:00
|
|
|
|
{
|
2018-02-27 15:28:54 +01:00
|
|
|
|
logger.Info(">>>--- Session procedure was aborted! ---<<<");
|
2018-02-15 15:42:54 +01:00
|
|
|
|
// TODO: Not when user chose to terminate after reconfiguration! Probably needs IOperationSequenceResult or alike...
|
2018-02-07 13:25:49 +01:00
|
|
|
|
uiFactory.Show(TextKey.MessageBox_SessionStartError, TextKey.MessageBox_SessionStartErrorTitle, icon: MessageBoxIcon.Error);
|
|
|
|
|
|
2018-02-08 13:32:48 +01:00
|
|
|
|
if (!initial)
|
2018-02-07 13:25:49 +01:00
|
|
|
|
{
|
2018-02-08 13:32:48 +01:00
|
|
|
|
shutdown.Invoke();
|
2018-02-07 13:25:49 +01:00
|
|
|
|
}
|
2018-02-02 09:18:35 +01:00
|
|
|
|
}
|
2018-01-17 08:26:44 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-01-30 14:41:36 +01:00
|
|
|
|
private void StopSession()
|
2018-01-17 08:26:44 +01:00
|
|
|
|
{
|
2018-01-30 14:41:36 +01:00
|
|
|
|
runtimeWindow.Show();
|
|
|
|
|
runtimeWindow.BringToForeground();
|
2018-02-06 15:12:11 +01:00
|
|
|
|
runtimeWindow.ShowProgressBar();
|
2018-02-27 15:28:54 +01:00
|
|
|
|
logger.Info(">>>--- Reverting session operations ---<<<");
|
2018-01-30 14:41:36 +01:00
|
|
|
|
|
2018-02-06 15:12:11 +01:00
|
|
|
|
var success = sessionSequence.TryRevert();
|
|
|
|
|
|
|
|
|
|
if (success)
|
|
|
|
|
{
|
2018-02-27 15:28:54 +01:00
|
|
|
|
logger.Info(">>>--- Session is terminated ---<<<");
|
2018-02-08 13:32:48 +01:00
|
|
|
|
sessionRunning = false;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2018-02-27 15:28:54 +01:00
|
|
|
|
logger.Info(">>>--- Session reversion was erroneous! ---<<<");
|
|
|
|
|
uiFactory.Show(TextKey.MessageBox_SessionStopError, TextKey.MessageBox_SessionStopErrorTitle, icon: MessageBoxIcon.Error);
|
2018-02-06 15:12:11 +01:00
|
|
|
|
}
|
2018-01-17 08:26:44 +01:00
|
|
|
|
}
|
2018-02-20 15:15:26 +01:00
|
|
|
|
|
2018-02-27 15:28:54 +01:00
|
|
|
|
private void RegisterEvents()
|
|
|
|
|
{
|
|
|
|
|
client.ConnectionLost += Client_ConnectionLost;
|
|
|
|
|
runtimeHost.ShutdownRequested += RuntimeHost_ShutdownRequested;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void RegisterSessionEvents()
|
|
|
|
|
{
|
|
|
|
|
configuration.CurrentSession.ClientProcess.Terminated += ClientProcess_Terminated;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void DeregisterEvents()
|
|
|
|
|
{
|
|
|
|
|
client.ConnectionLost -= Client_ConnectionLost;
|
|
|
|
|
runtimeHost.ShutdownRequested -= RuntimeHost_ShutdownRequested;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void DeregisterSessionEvents()
|
|
|
|
|
{
|
|
|
|
|
configuration.CurrentSession.ClientProcess.Terminated -= ClientProcess_Terminated;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ClientProcess_Terminated(int exitCode)
|
|
|
|
|
{
|
|
|
|
|
logger.Error($"Client application has unexpectedly terminated with exit code {exitCode}!");
|
|
|
|
|
// TODO: Check if message box is rendered on new desktop as well -> otherwise shutdown is blocked!
|
|
|
|
|
uiFactory.Show(TextKey.MessageBox_ApplicationError, TextKey.MessageBox_ApplicationErrorTitle, icon: MessageBoxIcon.Error);
|
|
|
|
|
|
|
|
|
|
shutdown.Invoke();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Client_ConnectionLost()
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Lost connection to the client application!");
|
|
|
|
|
// TODO: Check if message box is rendered on new desktop as well -> otherwise shutdown is blocked!
|
|
|
|
|
uiFactory.Show(TextKey.MessageBox_ApplicationError, TextKey.MessageBox_ApplicationErrorTitle, icon: MessageBoxIcon.Error);
|
|
|
|
|
|
|
|
|
|
shutdown.Invoke();
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-20 15:15:26 +01:00
|
|
|
|
private void RuntimeHost_ShutdownRequested()
|
|
|
|
|
{
|
2018-02-27 15:28:54 +01:00
|
|
|
|
logger.Info("Received shutdown request from the client application.");
|
2018-02-20 15:15:26 +01:00
|
|
|
|
shutdown.Invoke();
|
|
|
|
|
}
|
2018-01-17 08:26:44 +01:00
|
|
|
|
}
|
|
|
|
|
}
|