/* * 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; using System.Security.AccessControl; using System.Threading; using SafeExamBrowser.Contracts.Communication.Hosts; using SafeExamBrowser.Contracts.Communication.Proxies; using SafeExamBrowser.Contracts.Configuration; using SafeExamBrowser.Contracts.Configuration.Settings; using SafeExamBrowser.Contracts.Core.OperationModel; using SafeExamBrowser.Contracts.Core.OperationModel.Events; using SafeExamBrowser.Contracts.I18n; using SafeExamBrowser.Contracts.Logging; using SafeExamBrowser.Contracts.UserInterface.MessageBox; using SafeExamBrowser.Runtime.Operations.Events; namespace SafeExamBrowser.Runtime.Operations { internal class ServiceOperation : SessionOperation { private ILogger logger; private IRuntimeHost runtimeHost; private IServiceProxy service; private int timeout_ms; public override event ActionRequiredEventHandler ActionRequired; public override event StatusChangedEventHandler StatusChanged; public ServiceOperation( ILogger logger, IRuntimeHost runtimeHost, IServiceProxy service, SessionContext sessionContext, int timeout_ms) : base(sessionContext) { this.logger = logger; this.runtimeHost = runtimeHost; this.service = service; this.timeout_ms = timeout_ms; } public override OperationResult Perform() { logger.Info($"Initializing service..."); StatusChanged?.Invoke(TextKey.OperationStatus_InitializeServiceSession); var success = TryEstablishConnection(); if (service.IsConnected) { success = TryStartSession(); } return success ? OperationResult.Success : OperationResult.Failed; } public override OperationResult Repeat() { logger.Info($"Initializing service..."); StatusChanged?.Invoke(TextKey.OperationStatus_InitializeServiceSession); var success = false; if (service.IsConnected) { success = TryStopSession(); } else { success = TryEstablishConnection(); } if (success && service.IsConnected) { success = TryStartSession(); } return success ? OperationResult.Success : OperationResult.Failed; } public override OperationResult Revert() { logger.Info("Finalizing service..."); StatusChanged?.Invoke(TextKey.OperationStatus_FinalizeServiceSession); var success = true; if (service.IsConnected) { if (Context.Current != null) { success &= TryStopSession(); } success &= TryTerminateConnection(); } return success ? OperationResult.Success : OperationResult.Failed; } private bool TryEstablishConnection() { var mandatory = Context.Next.Settings.ServicePolicy == ServicePolicy.Mandatory; var warn = Context.Next.Settings.ServicePolicy == ServicePolicy.Warn; var connected = service.Connect(); var success = connected || !mandatory; if (success) { service.Ignore = !connected; logger.Info($"The service is {(mandatory ? "mandatory" : "optional")} and {(connected ? "connected." : "not connected. All service-related operations will be ignored!")}"); if (!connected && warn) { ActionRequired?.Invoke(new MessageEventArgs { Icon = MessageBoxIcon.Warning, Message = TextKey.MessageBox_ServiceUnavailableWarning, Title = TextKey.MessageBox_ServiceUnavailableWarningTitle }); } } else { logger.Error("The service is mandatory but no connection could be established!"); ActionRequired?.Invoke(new MessageEventArgs { Icon = MessageBoxIcon.Error, Message = TextKey.MessageBox_ServiceUnavailableError, Title = TextKey.MessageBox_ServiceUnavailableErrorTitle }); } return success; } private bool TryTerminateConnection() { var disconnected = service.Disconnect(); if (disconnected) { logger.Info("Successfully disconnected from service."); } else { logger.Error("Failed to disconnect from service!"); } return disconnected; } private bool TryStartSession() { var configuration = new ServiceConfiguration { AppConfig = Context.Next.AppConfig, SessionId = Context.Next.SessionId, Settings = Context.Next.Settings }; var started = false; logger.Info("Starting new service session..."); var communication = service.StartSession(configuration); if (communication.Success) { started = TryWaitForServiceEvent(Context.Next.AppConfig.ServiceEventName); if (started) { logger.Info("Successfully started new service session."); } else { logger.Error($"Failed to start new service session within {timeout_ms / 1000} seconds!"); } } else { logger.Error("Failed to communicate session start to service!"); } return started; } private bool TryStopSession() { var stopped = false; logger.Info("Stopping current service session..."); var communication = service.StopSession(Context.Current.SessionId); if (communication.Success) { stopped = TryWaitForServiceEvent(Context.Current.AppConfig.ServiceEventName); if (stopped) { logger.Info("Successfully stopped service session."); } else { logger.Error($"Failed to stop service session within {timeout_ms / 1000} seconds!"); } } else { logger.Error("Failed to communicate session stop to service!"); } return stopped; } private bool TryWaitForServiceEvent(string eventName) { var serviceEvent = default(EventWaitHandle); var startTime = DateTime.Now; do { if (EventWaitHandle.TryOpenExisting(eventName, EventWaitHandleRights.Synchronize, out serviceEvent)) { break; } } while (startTime.AddMilliseconds(timeout_ms) > DateTime.Now); if (serviceEvent != default(EventWaitHandle)) { using (serviceEvent) { return serviceEvent.WaitOne(timeout_ms); } } return false; } } }