Update Safe Exam Browser Patch to 3.10.0.826
This commit is contained in:
@@ -6,98 +6,48 @@
|
||||
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
||||
*/
|
||||
|
||||
using System;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Threading;
|
||||
using SafeExamBrowser.Communication.Contracts.Data;
|
||||
using SafeExamBrowser.Communication.Contracts.Events;
|
||||
using SafeExamBrowser.Communication.Contracts.Hosts;
|
||||
using SafeExamBrowser.Communication.Contracts.Proxies;
|
||||
using SafeExamBrowser.Configuration.Contracts;
|
||||
using SafeExamBrowser.Core.Contracts.OperationModel;
|
||||
using SafeExamBrowser.Core.Contracts.OperationModel.Events;
|
||||
using SafeExamBrowser.I18n.Contracts;
|
||||
using SafeExamBrowser.Core.Contracts.ResponsibilityModel;
|
||||
using SafeExamBrowser.Logging.Contracts;
|
||||
using SafeExamBrowser.Runtime.Operations.Events;
|
||||
using SafeExamBrowser.Server.Contracts.Data;
|
||||
using SafeExamBrowser.Settings.Security;
|
||||
using SafeExamBrowser.Settings.Service;
|
||||
using SafeExamBrowser.UserInterface.Contracts;
|
||||
using SafeExamBrowser.UserInterface.Contracts.MessageBox;
|
||||
using SafeExamBrowser.Runtime.Responsibilities;
|
||||
using SafeExamBrowser.UserInterface.Contracts.Windows;
|
||||
using System.Deployment.Internal;
|
||||
|
||||
namespace SafeExamBrowser.Runtime
|
||||
{
|
||||
internal class RuntimeController
|
||||
{
|
||||
private readonly AppConfig appConfig;
|
||||
private readonly ILogger logger;
|
||||
private readonly IMessageBox messageBox;
|
||||
private readonly IOperationSequence bootstrapSequence;
|
||||
private readonly IRepeatableOperationSequence sessionSequence;
|
||||
private readonly IRuntimeHost runtimeHost;
|
||||
private readonly IResponsibilityCollection<RuntimeTask> responsibilities;
|
||||
private readonly RuntimeContext runtimeContext;
|
||||
private readonly IRuntimeWindow runtimeWindow;
|
||||
private readonly IServiceProxy service;
|
||||
private readonly SessionContext sessionContext;
|
||||
private readonly ISplashScreen splashScreen;
|
||||
private readonly Action shutdown;
|
||||
private readonly IText text;
|
||||
private readonly IUserInterfaceFactory uiFactory;
|
||||
|
||||
private SessionConfiguration Session
|
||||
{
|
||||
get { return sessionContext.Current; }
|
||||
}
|
||||
|
||||
private bool SessionIsRunning
|
||||
{
|
||||
get { return Session != null; }
|
||||
}
|
||||
private bool SessionIsRunning => runtimeContext.Current != default;
|
||||
|
||||
internal RuntimeController(
|
||||
AppConfig appConfig,
|
||||
ILogger logger,
|
||||
IMessageBox messageBox,
|
||||
IOperationSequence bootstrapSequence,
|
||||
IRepeatableOperationSequence sessionSequence,
|
||||
IRuntimeHost runtimeHost,
|
||||
IResponsibilityCollection<RuntimeTask> responsibilities,
|
||||
RuntimeContext runtimeContext,
|
||||
IRuntimeWindow runtimeWindow,
|
||||
IServiceProxy service,
|
||||
SessionContext sessionContext,
|
||||
Action shutdown,
|
||||
ISplashScreen splashScreen,
|
||||
IText text,
|
||||
IUserInterfaceFactory uiFactory)
|
||||
ISplashScreen splashScreen)
|
||||
{
|
||||
this.appConfig = appConfig;
|
||||
this.bootstrapSequence = bootstrapSequence;
|
||||
this.responsibilities = responsibilities;
|
||||
this.logger = logger;
|
||||
this.messageBox = messageBox;
|
||||
this.runtimeHost = runtimeHost;
|
||||
this.runtimeWindow = runtimeWindow;
|
||||
this.sessionSequence = sessionSequence;
|
||||
this.service = service;
|
||||
this.sessionContext = sessionContext;
|
||||
this.shutdown = shutdown;
|
||||
this.runtimeContext = runtimeContext;
|
||||
this.splashScreen = splashScreen;
|
||||
this.text = text;
|
||||
this.uiFactory = uiFactory;
|
||||
}
|
||||
|
||||
internal bool TryStart()
|
||||
{
|
||||
logger.Info("Initiating startup procedure...");
|
||||
|
||||
bootstrapSequence.ProgressChanged += BootstrapSequence_ProgressChanged;
|
||||
bootstrapSequence.StatusChanged += BootstrapSequence_StatusChanged;
|
||||
sessionSequence.ActionRequired += SessionSequence_ActionRequired;
|
||||
sessionSequence.ProgressChanged += SessionSequence_ProgressChanged;
|
||||
sessionSequence.StatusChanged += SessionSequence_StatusChanged;
|
||||
|
||||
// We need to show the runtime window here already, this way implicitly setting it as the runtime application's main window.
|
||||
// Otherwise, the splash screen is considered as the main window and thus the operating system and/or WPF does not correctly
|
||||
// activate the runtime window once bootstrapping has finished, which in turn leads to undesired UI behavior.
|
||||
// activate the runtime window once bootstrapping has finished, which in turn leads to undesired user interface behavior.
|
||||
runtimeWindow.Show();
|
||||
runtimeWindow.BringToForeground();
|
||||
runtimeWindow.SetIndeterminate();
|
||||
@@ -109,21 +59,21 @@ namespace SafeExamBrowser.Runtime
|
||||
|
||||
if (initialized)
|
||||
{
|
||||
RegisterEvents();
|
||||
responsibilities.Delegate(RuntimeTask.RegisterEvents);
|
||||
splashScreen.Hide();
|
||||
|
||||
logger.Info("Application successfully initialized.");
|
||||
logger.Log(string.Empty);
|
||||
logger.Subscribe(runtimeWindow);
|
||||
splashScreen.Hide();
|
||||
|
||||
StartSession();
|
||||
responsibilities.Delegate(RuntimeTask.StartSession);
|
||||
}
|
||||
else
|
||||
{
|
||||
logger.Info("Application startup aborted!");
|
||||
logger.Log(string.Empty);
|
||||
|
||||
messageBox.Show(AppendLogFilePaths(TextKey.MessageBox_StartupError), text.Get(TextKey.MessageBox_StartupErrorTitle), icon: MessageBoxIcon.Error, parent: splashScreen);
|
||||
responsibilities.Delegate(RuntimeTask.ShowStartupError);
|
||||
}
|
||||
|
||||
return initialized && SessionIsRunning;
|
||||
@@ -131,11 +81,11 @@ namespace SafeExamBrowser.Runtime
|
||||
|
||||
internal void Terminate()
|
||||
{
|
||||
DeregisterEvents();
|
||||
responsibilities.Delegate(RuntimeTask.DeregisterEvents);
|
||||
|
||||
if (SessionIsRunning)
|
||||
{
|
||||
StopSession();
|
||||
responsibilities.Delegate(RuntimeTask.StopSession);
|
||||
}
|
||||
|
||||
logger.Unsubscribe(runtimeWindow);
|
||||
@@ -159,586 +109,10 @@ namespace SafeExamBrowser.Runtime
|
||||
logger.Info("Shutdown procedure failed!");
|
||||
logger.Log(string.Empty);
|
||||
|
||||
messageBox.Show(AppendLogFilePaths(TextKey.MessageBox_ShutdownError), text.Get(TextKey.MessageBox_ShutdownErrorTitle), icon: MessageBoxIcon.Error, parent: splashScreen);
|
||||
responsibilities.Delegate(RuntimeTask.ShowShutdownError);
|
||||
}
|
||||
|
||||
splashScreen.Close();
|
||||
}
|
||||
|
||||
private void StartSession()
|
||||
{
|
||||
runtimeWindow.Show();
|
||||
runtimeWindow.BringToForeground();
|
||||
runtimeWindow.ShowProgressBar = true;
|
||||
logger.Info(AppendDivider("Session Start Procedure"));
|
||||
|
||||
if (SessionIsRunning)
|
||||
{
|
||||
DeregisterSessionEvents();
|
||||
}
|
||||
|
||||
var result = SessionIsRunning ? sessionSequence.TryRepeat() : sessionSequence.TryPerform();
|
||||
|
||||
if (result == OperationResult.Success)
|
||||
{
|
||||
logger.Info(AppendDivider("Session Running"));
|
||||
|
||||
HandleSessionStartSuccess();
|
||||
}
|
||||
else if (result == OperationResult.Failed)
|
||||
{
|
||||
logger.Info(AppendDivider("Session Start Failed"));
|
||||
|
||||
HandleSessionStartFailure();
|
||||
}
|
||||
else if (result == OperationResult.Aborted)
|
||||
{
|
||||
logger.Info(AppendDivider("Session Start Aborted"));
|
||||
|
||||
HandleSessionStartAbortion();
|
||||
}
|
||||
}
|
||||
|
||||
private void HandleSessionStartSuccess()
|
||||
{
|
||||
RegisterSessionEvents();
|
||||
|
||||
runtimeWindow.ShowProgressBar = false;
|
||||
runtimeWindow.ShowLog = Session.Settings.Security.AllowApplicationLogAccess;
|
||||
runtimeWindow.TopMost = Session.Settings.Security.KioskMode != KioskMode.None;
|
||||
runtimeWindow.UpdateStatus(TextKey.RuntimeWindow_ApplicationRunning);
|
||||
|
||||
//if (Session.Settings.Security.KioskMode == KioskMode.DisableExplorerShell)
|
||||
//{
|
||||
// runtimeWindow.Hide();
|
||||
//}
|
||||
runtimeWindow.Hide();
|
||||
}
|
||||
|
||||
private void HandleSessionStartFailure()
|
||||
{
|
||||
if (SessionIsRunning)
|
||||
{
|
||||
StopSession();
|
||||
|
||||
messageBox.Show(AppendLogFilePaths(TextKey.MessageBox_SessionStartError), text.Get(TextKey.MessageBox_SessionStartErrorTitle), icon: MessageBoxIcon.Error, parent: runtimeWindow);
|
||||
|
||||
logger.Info("Terminating application...");
|
||||
shutdown.Invoke();
|
||||
}
|
||||
else
|
||||
{
|
||||
messageBox.Show(AppendLogFilePaths(TextKey.MessageBox_SessionStartError), text.Get(TextKey.MessageBox_SessionStartErrorTitle), icon: MessageBoxIcon.Error, parent: runtimeWindow);
|
||||
}
|
||||
}
|
||||
|
||||
private void HandleSessionStartAbortion()
|
||||
{
|
||||
if (SessionIsRunning)
|
||||
{
|
||||
RegisterSessionEvents();
|
||||
|
||||
runtimeWindow.ShowProgressBar = false;
|
||||
runtimeWindow.UpdateStatus(TextKey.RuntimeWindow_ApplicationRunning);
|
||||
runtimeWindow.TopMost = Session.Settings.Security.KioskMode != KioskMode.None;
|
||||
|
||||
//if (Session.Settings.Security.KioskMode == KioskMode.DisableExplorerShell)
|
||||
//{
|
||||
// runtimeWindow.Hide();
|
||||
//}
|
||||
runtimeWindow.Hide();
|
||||
|
||||
sessionContext.ClientProxy.InformReconfigurationAborted();
|
||||
}
|
||||
}
|
||||
|
||||
private void StopSession()
|
||||
{
|
||||
runtimeWindow.Show();
|
||||
runtimeWindow.BringToForeground();
|
||||
runtimeWindow.ShowProgressBar = true;
|
||||
logger.Info(AppendDivider("Session Stop Procedure"));
|
||||
|
||||
DeregisterSessionEvents();
|
||||
|
||||
var success = sessionSequence.TryRevert() == OperationResult.Success;
|
||||
|
||||
if (success)
|
||||
{
|
||||
logger.Info(AppendDivider("Session Terminated"));
|
||||
}
|
||||
else
|
||||
{
|
||||
logger.Info(AppendDivider("Session Stop Failed"));
|
||||
}
|
||||
}
|
||||
|
||||
private void RegisterEvents()
|
||||
{
|
||||
runtimeHost.ClientConfigurationNeeded += RuntimeHost_ClientConfigurationNeeded;
|
||||
runtimeHost.ReconfigurationRequested += RuntimeHost_ReconfigurationRequested;
|
||||
runtimeHost.ShutdownRequested += RuntimeHost_ShutdownRequested;
|
||||
}
|
||||
|
||||
private void DeregisterEvents()
|
||||
{
|
||||
runtimeHost.ClientConfigurationNeeded -= RuntimeHost_ClientConfigurationNeeded;
|
||||
runtimeHost.ReconfigurationRequested -= RuntimeHost_ReconfigurationRequested;
|
||||
runtimeHost.ShutdownRequested -= RuntimeHost_ShutdownRequested;
|
||||
}
|
||||
|
||||
private void RegisterSessionEvents()
|
||||
{
|
||||
service.ConnectionLost += ServiceProxy_ConnectionLost;
|
||||
sessionContext.ClientProcess.Terminated += ClientProcess_Terminated;
|
||||
sessionContext.ClientProxy.ConnectionLost += ClientProxy_ConnectionLost;
|
||||
}
|
||||
|
||||
private void DeregisterSessionEvents()
|
||||
{
|
||||
service.ConnectionLost -= ServiceProxy_ConnectionLost;
|
||||
|
||||
if (sessionContext.ClientProcess != null)
|
||||
{
|
||||
sessionContext.ClientProcess.Terminated -= ClientProcess_Terminated;
|
||||
}
|
||||
|
||||
if (sessionContext.ClientProxy != null)
|
||||
{
|
||||
sessionContext.ClientProxy.ConnectionLost -= ClientProxy_ConnectionLost;
|
||||
}
|
||||
}
|
||||
|
||||
private void BootstrapSequence_ProgressChanged(ProgressChangedEventArgs args)
|
||||
{
|
||||
MapProgress(splashScreen, args);
|
||||
}
|
||||
|
||||
private void BootstrapSequence_StatusChanged(TextKey status)
|
||||
{
|
||||
splashScreen.UpdateStatus(status, true);
|
||||
}
|
||||
|
||||
private void ClientProcess_Terminated(int exitCode)
|
||||
{
|
||||
logger.Error($"Client application has unexpectedly terminated with exit code {exitCode}!");
|
||||
|
||||
if (SessionIsRunning)
|
||||
{
|
||||
StopSession();
|
||||
}
|
||||
|
||||
//messageBox.Show(TextKey.MessageBox_ApplicationError, TextKey.MessageBox_ApplicationErrorTitle, icon: MessageBoxIcon.Error, parent: runtimeWindow);
|
||||
shutdown.Invoke();
|
||||
}
|
||||
|
||||
private void ClientProxy_ConnectionLost()
|
||||
{
|
||||
logger.Error("Lost connection to the client application!");
|
||||
|
||||
if (SessionIsRunning)
|
||||
{
|
||||
StopSession();
|
||||
}
|
||||
|
||||
messageBox.Show(TextKey.MessageBox_ApplicationError, TextKey.MessageBox_ApplicationErrorTitle, icon: MessageBoxIcon.Error, parent: runtimeWindow);
|
||||
shutdown.Invoke();
|
||||
}
|
||||
|
||||
private void RuntimeHost_ClientConfigurationNeeded(ClientConfigurationEventArgs args)
|
||||
{
|
||||
args.ClientConfiguration = new ClientConfiguration
|
||||
{
|
||||
AppConfig = sessionContext.Next.AppConfig,
|
||||
SessionId = sessionContext.Next.SessionId,
|
||||
Settings = sessionContext.Next.Settings
|
||||
};
|
||||
}
|
||||
|
||||
private void RuntimeHost_ReconfigurationRequested(ReconfigurationEventArgs args)
|
||||
{
|
||||
logger.Info($"Accepted request for reconfiguration with '{args.ConfigurationPath}'.");
|
||||
|
||||
sessionContext.ReconfigurationFilePath = args.ConfigurationPath;
|
||||
sessionContext.ReconfigurationUrl = args.ResourceUrl;
|
||||
|
||||
StartSession();
|
||||
}
|
||||
|
||||
private void RuntimeHost_ShutdownRequested()
|
||||
{
|
||||
logger.Info("Received shutdown request from the client application.");
|
||||
shutdown.Invoke();
|
||||
}
|
||||
|
||||
private void ServiceProxy_ConnectionLost()
|
||||
{
|
||||
if (SessionIsRunning && Session.Settings.Service.Policy == ServicePolicy.Mandatory)
|
||||
{
|
||||
logger.Error("Lost connection to the service component!");
|
||||
StopSession();
|
||||
messageBox.Show(TextKey.MessageBox_ApplicationError, TextKey.MessageBox_ApplicationErrorTitle, icon: MessageBoxIcon.Error, parent: runtimeWindow);
|
||||
shutdown.Invoke();
|
||||
}
|
||||
else
|
||||
{
|
||||
logger.Warn("Lost connection to the service component!");
|
||||
}
|
||||
}
|
||||
|
||||
private void SessionSequence_ActionRequired(ActionRequiredEventArgs args)
|
||||
{
|
||||
switch (args)
|
||||
{
|
||||
case ConfigurationCompletedEventArgs a:
|
||||
AskIfConfigurationSufficient(a);
|
||||
break;
|
||||
case ExamSelectionEventArgs a:
|
||||
AskForExamSelection(a);
|
||||
break;
|
||||
case MessageEventArgs m:
|
||||
ShowMessageBox(m);
|
||||
break;
|
||||
case PasswordRequiredEventArgs p:
|
||||
AskForPassword(p);
|
||||
break;
|
||||
case ServerFailureEventArgs a:
|
||||
AskForServerFailureAction(a);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
private void AskForExamSelection(ExamSelectionEventArgs args)
|
||||
{
|
||||
var isStartup = !SessionIsRunning;
|
||||
var isRunningOnDefaultDesktop = SessionIsRunning && Session.Settings.Security.KioskMode == KioskMode.DisableExplorerShell;
|
||||
|
||||
if (isStartup || isRunningOnDefaultDesktop)
|
||||
{
|
||||
TryAskForExamSelectionViaDialog(args);
|
||||
}
|
||||
else
|
||||
{
|
||||
TryAskForExamSelectionViaClient(args);
|
||||
}
|
||||
}
|
||||
|
||||
private void AskForServerFailureAction(ServerFailureEventArgs args)
|
||||
{
|
||||
var isStartup = !SessionIsRunning;
|
||||
var isRunningOnDefaultDesktop = SessionIsRunning && Session.Settings.Security.KioskMode == KioskMode.DisableExplorerShell;
|
||||
|
||||
if (isStartup || isRunningOnDefaultDesktop)
|
||||
{
|
||||
TryAskForServerFailureActionViaDialog(args);
|
||||
}
|
||||
else
|
||||
{
|
||||
TryAskForServerFailureActionViaClient(args);
|
||||
}
|
||||
}
|
||||
|
||||
private void AskIfConfigurationSufficient(ConfigurationCompletedEventArgs args)
|
||||
{
|
||||
var message = TextKey.MessageBox_ClientConfigurationQuestion;
|
||||
var title = TextKey.MessageBox_ClientConfigurationQuestionTitle;
|
||||
var result = messageBox.Show(message, title, MessageBoxAction.YesNo, MessageBoxIcon.Question, runtimeWindow);
|
||||
|
||||
args.AbortStartup = result == MessageBoxResult.Yes;
|
||||
}
|
||||
|
||||
private void AskForPassword(PasswordRequiredEventArgs args)
|
||||
{
|
||||
var isStartup = !SessionIsRunning;
|
||||
var isRunningOnDefaultDesktop = SessionIsRunning && Session.Settings.Security.KioskMode == KioskMode.DisableExplorerShell;
|
||||
|
||||
if (isStartup || isRunningOnDefaultDesktop)
|
||||
{
|
||||
TryGetPasswordViaDialog(args);
|
||||
}
|
||||
else
|
||||
{
|
||||
TryGetPasswordViaClient(args);
|
||||
}
|
||||
}
|
||||
|
||||
private void ShowMessageBox(MessageEventArgs args)
|
||||
{
|
||||
var isStartup = !SessionIsRunning;
|
||||
var isRunningOnDefaultDesktop = SessionIsRunning && Session.Settings.Security.KioskMode == KioskMode.DisableExplorerShell;
|
||||
var message = text.Get(args.Message);
|
||||
var title = text.Get(args.Title);
|
||||
|
||||
foreach (var placeholder in args.MessagePlaceholders)
|
||||
{
|
||||
message = message.Replace(placeholder.Key, placeholder.Value);
|
||||
}
|
||||
|
||||
foreach (var placeholder in args.TitlePlaceholders)
|
||||
{
|
||||
title = title.Replace(placeholder.Key, placeholder.Value);
|
||||
}
|
||||
|
||||
if (isStartup || isRunningOnDefaultDesktop)
|
||||
{
|
||||
args.Result = messageBox.Show(message, title, args.Action, args.Icon, runtimeWindow);
|
||||
}
|
||||
else
|
||||
{
|
||||
args.Result = ShowMessageBoxViaClient(message, title, args.Action, args.Icon);
|
||||
}
|
||||
}
|
||||
|
||||
private MessageBoxResult ShowMessageBoxViaClient(string message, string title, MessageBoxAction action, MessageBoxIcon icon)
|
||||
{
|
||||
var requestId = Guid.NewGuid();
|
||||
var result = MessageBoxResult.None;
|
||||
var response = default(MessageBoxReplyEventArgs);
|
||||
var responseEvent = new AutoResetEvent(false);
|
||||
var responseEventHandler = new CommunicationEventHandler<MessageBoxReplyEventArgs>((args) =>
|
||||
{
|
||||
if (args.RequestId == requestId)
|
||||
{
|
||||
response = args;
|
||||
responseEvent.Set();
|
||||
}
|
||||
});
|
||||
|
||||
runtimeHost.MessageBoxReplyReceived += responseEventHandler;
|
||||
|
||||
var communication = sessionContext.ClientProxy.ShowMessage(message, title, (int) action, (int) icon, requestId);
|
||||
|
||||
if (communication.Success)
|
||||
{
|
||||
responseEvent.WaitOne();
|
||||
result = (MessageBoxResult) response.Result;
|
||||
}
|
||||
|
||||
runtimeHost.MessageBoxReplyReceived -= responseEventHandler;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
private void TryAskForExamSelectionViaDialog(ExamSelectionEventArgs args)
|
||||
{
|
||||
var dialog = uiFactory.CreateExamSelectionDialog(args.Exams);
|
||||
var result = dialog.Show(runtimeWindow);
|
||||
|
||||
args.SelectedExam = result.SelectedExam;
|
||||
args.Success = result.Success;
|
||||
}
|
||||
|
||||
private void TryAskForExamSelectionViaClient(ExamSelectionEventArgs args)
|
||||
{
|
||||
var exams = args.Exams.Select(e => (e.Id, e.LmsName, e.Name, e.Url));
|
||||
var requestId = Guid.NewGuid();
|
||||
var response = default(ExamSelectionReplyEventArgs);
|
||||
var responseEvent = new AutoResetEvent(false);
|
||||
var responseEventHandler = new CommunicationEventHandler<ExamSelectionReplyEventArgs>((a) =>
|
||||
{
|
||||
if (a.RequestId == requestId)
|
||||
{
|
||||
response = a;
|
||||
responseEvent.Set();
|
||||
}
|
||||
});
|
||||
|
||||
runtimeHost.ExamSelectionReceived += responseEventHandler;
|
||||
|
||||
var communication = sessionContext.ClientProxy.RequestExamSelection(exams, requestId);
|
||||
|
||||
if (communication.Success)
|
||||
{
|
||||
responseEvent.WaitOne();
|
||||
args.SelectedExam = args.Exams.First(e => e.Id == response.SelectedExamId);
|
||||
args.Success = response.Success;
|
||||
}
|
||||
else
|
||||
{
|
||||
args.SelectedExam = default(Exam);
|
||||
args.Success = false;
|
||||
}
|
||||
|
||||
runtimeHost.ExamSelectionReceived -= responseEventHandler;
|
||||
}
|
||||
|
||||
private void TryAskForServerFailureActionViaDialog(ServerFailureEventArgs args)
|
||||
{
|
||||
var dialog = uiFactory.CreateServerFailureDialog(args.Message, args.ShowFallback);
|
||||
var result = dialog.Show(runtimeWindow);
|
||||
|
||||
args.Abort = result.Abort;
|
||||
args.Fallback = result.Fallback;
|
||||
args.Retry = result.Retry;
|
||||
}
|
||||
|
||||
private void TryAskForServerFailureActionViaClient(ServerFailureEventArgs args)
|
||||
{
|
||||
var requestId = Guid.NewGuid();
|
||||
var response = default(ServerFailureActionReplyEventArgs);
|
||||
var responseEvent = new AutoResetEvent(false);
|
||||
var responseEventHandler = new CommunicationEventHandler<ServerFailureActionReplyEventArgs>((a) =>
|
||||
{
|
||||
if (a.RequestId == requestId)
|
||||
{
|
||||
response = a;
|
||||
responseEvent.Set();
|
||||
}
|
||||
});
|
||||
|
||||
runtimeHost.ServerFailureActionReceived += responseEventHandler;
|
||||
|
||||
var communication = sessionContext.ClientProxy.RequestServerFailureAction(args.Message, args.ShowFallback, requestId);
|
||||
|
||||
if (communication.Success)
|
||||
{
|
||||
responseEvent.WaitOne();
|
||||
args.Abort = response.Abort;
|
||||
args.Fallback = response.Fallback;
|
||||
args.Retry = response.Retry;
|
||||
}
|
||||
else
|
||||
{
|
||||
args.Abort = true;
|
||||
args.Fallback = false;
|
||||
args.Retry = false;
|
||||
}
|
||||
|
||||
runtimeHost.ServerFailureActionReceived -= responseEventHandler;
|
||||
}
|
||||
|
||||
private void TryGetPasswordViaDialog(PasswordRequiredEventArgs args)
|
||||
{
|
||||
var message = default(TextKey);
|
||||
var title = default(TextKey);
|
||||
|
||||
switch (args.Purpose)
|
||||
{
|
||||
case PasswordRequestPurpose.LocalAdministrator:
|
||||
message = TextKey.PasswordDialog_LocalAdminPasswordRequired;
|
||||
title = TextKey.PasswordDialog_LocalAdminPasswordRequiredTitle;
|
||||
break;
|
||||
case PasswordRequestPurpose.LocalSettings:
|
||||
message = TextKey.PasswordDialog_LocalSettingsPasswordRequired;
|
||||
title = TextKey.PasswordDialog_LocalSettingsPasswordRequiredTitle;
|
||||
break;
|
||||
case PasswordRequestPurpose.Settings:
|
||||
message = TextKey.PasswordDialog_SettingsPasswordRequired;
|
||||
title = TextKey.PasswordDialog_SettingsPasswordRequiredTitle;
|
||||
break;
|
||||
}
|
||||
|
||||
var dialog = uiFactory.CreatePasswordDialog(text.Get(message), text.Get(title));
|
||||
var result = dialog.Show(runtimeWindow);
|
||||
|
||||
args.Password = result.Password;
|
||||
args.Success = result.Success;
|
||||
}
|
||||
|
||||
private void TryGetPasswordViaClient(PasswordRequiredEventArgs args)
|
||||
{
|
||||
var requestId = Guid.NewGuid();
|
||||
var response = default(PasswordReplyEventArgs);
|
||||
var responseEvent = new AutoResetEvent(false);
|
||||
var responseEventHandler = new CommunicationEventHandler<PasswordReplyEventArgs>((a) =>
|
||||
{
|
||||
if (a.RequestId == requestId)
|
||||
{
|
||||
response = a;
|
||||
responseEvent.Set();
|
||||
}
|
||||
});
|
||||
|
||||
runtimeHost.PasswordReceived += responseEventHandler;
|
||||
|
||||
var communication = sessionContext.ClientProxy.RequestPassword(args.Purpose, requestId);
|
||||
|
||||
if (communication.Success)
|
||||
{
|
||||
responseEvent.WaitOne();
|
||||
args.Password = response.Password;
|
||||
args.Success = response.Success;
|
||||
}
|
||||
else
|
||||
{
|
||||
args.Password = default(string);
|
||||
args.Success = false;
|
||||
}
|
||||
|
||||
runtimeHost.PasswordReceived -= responseEventHandler;
|
||||
}
|
||||
|
||||
private void SessionSequence_ProgressChanged(ProgressChangedEventArgs args)
|
||||
{
|
||||
MapProgress(runtimeWindow, args);
|
||||
}
|
||||
|
||||
private void SessionSequence_StatusChanged(TextKey status)
|
||||
{
|
||||
runtimeWindow?.UpdateStatus(status, true);
|
||||
}
|
||||
|
||||
private void MapProgress(IProgressIndicator progressIndicator, ProgressChangedEventArgs args)
|
||||
{
|
||||
if (args.CurrentValue.HasValue)
|
||||
{
|
||||
progressIndicator?.SetValue(args.CurrentValue.Value);
|
||||
}
|
||||
|
||||
if (args.IsIndeterminate == true)
|
||||
{
|
||||
progressIndicator?.SetIndeterminate();
|
||||
}
|
||||
|
||||
if (args.MaxValue.HasValue)
|
||||
{
|
||||
progressIndicator?.SetMaxValue(args.MaxValue.Value);
|
||||
}
|
||||
|
||||
if (args.Progress == true)
|
||||
{
|
||||
progressIndicator?.Progress();
|
||||
}
|
||||
|
||||
if (args.Regress == true)
|
||||
{
|
||||
progressIndicator?.Regress();
|
||||
}
|
||||
}
|
||||
|
||||
private string AppendDivider(string message)
|
||||
{
|
||||
var dashesLeft = new String('-', 48 - message.Length / 2 - message.Length % 2);
|
||||
var dashesRight = new String('-', 48 - message.Length / 2);
|
||||
|
||||
return $"### {dashesLeft} {message} {dashesRight} ###";
|
||||
}
|
||||
|
||||
private string AppendLogFilePaths(TextKey key)
|
||||
{
|
||||
var message = text.Get(key);
|
||||
|
||||
if (File.Exists(appConfig.BrowserLogFilePath))
|
||||
{
|
||||
message += $"{Environment.NewLine}{Environment.NewLine}{appConfig.BrowserLogFilePath}";
|
||||
}
|
||||
|
||||
if (File.Exists(appConfig.ClientLogFilePath))
|
||||
{
|
||||
message += $"{Environment.NewLine}{Environment.NewLine}{appConfig.ClientLogFilePath}";
|
||||
}
|
||||
|
||||
if (File.Exists(appConfig.RuntimeLogFilePath))
|
||||
{
|
||||
message += $"{Environment.NewLine}{Environment.NewLine}{appConfig.RuntimeLogFilePath}";
|
||||
}
|
||||
|
||||
if (File.Exists(appConfig.ServiceLogFilePath))
|
||||
{
|
||||
message += $"{Environment.NewLine}{Environment.NewLine}{appConfig.ServiceLogFilePath}";
|
||||
}
|
||||
|
||||
return message;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user