/* * 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.IO; using SafeExamBrowser.Contracts.Communication.Data; using SafeExamBrowser.Contracts.Configuration; using SafeExamBrowser.Contracts.Configuration.Cryptography; 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.Runtime.Operations.Events; namespace SafeExamBrowser.Runtime.Operations { internal class ConfigurationOperation : SessionOperation { private string[] commandLineArgs; private IConfigurationRepository configuration; private IHashAlgorithm hashAlgorithm; private ILogger logger; private string AppDataFile { get { return Path.Combine(Context.Next.AppConfig.AppDataFolder, Context.Next.AppConfig.DefaultSettingsFileName); } } private string ProgramDataFile { get { return Path.Combine(Context.Next.AppConfig.ProgramDataFolder, Context.Next.AppConfig.DefaultSettingsFileName); } } public override event ActionRequiredEventHandler ActionRequired; public override event StatusChangedEventHandler StatusChanged; public ConfigurationOperation( string[] commandLineArgs, IConfigurationRepository configuration, IHashAlgorithm hashAlgorithm, ILogger logger, SessionContext sessionContext) : base(sessionContext) { this.commandLineArgs = commandLineArgs; this.configuration = configuration; this.hashAlgorithm = hashAlgorithm; this.logger = logger; } public override OperationResult Perform() { logger.Info("Initializing application configuration..."); StatusChanged?.Invoke(TextKey.OperationStatus_InitializeConfiguration); var result = OperationResult.Failed; var isValidUri = TryInitializeSettingsUri(out var uri, out var source); if (isValidUri) { result = LoadSettingsForStartup(uri, source); } else { result = LoadDefaultSettings(); } LogOperationResult(result); return result; } public override OperationResult Repeat() { logger.Info("Initializing new application configuration..."); StatusChanged?.Invoke(TextKey.OperationStatus_InitializeConfiguration); var result = OperationResult.Failed; var isValidUri = TryValidateSettingsUri(Context.ReconfigurationFilePath, out var uri); if (isValidUri) { result = LoadSettingsForReconfiguration(uri); } else { logger.Warn($"The resource specified for reconfiguration does not exist or is not valid!"); } LogOperationResult(result); return result; } public override OperationResult Revert() { return OperationResult.Success; } private OperationResult LoadDefaultSettings() { logger.Info("No valid configuration resource specified and no local client configuration found - loading default settings..."); Context.Next.Settings = configuration.LoadDefaultSettings(); return OperationResult.Success; } private OperationResult LoadSettingsForStartup(Uri uri, UriSource source) { var currentPassword = default(string); var passwordParams = default(PasswordParameters); var settings = default(Settings); var status = default(LoadStatus?); if (source == UriSource.CommandLine) { var hasAppDataFile = File.Exists(AppDataFile); var hasProgramDataFile = File.Exists(ProgramDataFile); if (hasProgramDataFile) { status = TryLoadSettings(new Uri(ProgramDataFile, UriKind.Absolute), UriSource.ProgramData, out _, out settings); } else if (hasAppDataFile) { status = TryLoadSettings(new Uri(AppDataFile, UriKind.Absolute), UriSource.AppData, out _, out settings); } if ((!hasProgramDataFile && !hasAppDataFile) || status == LoadStatus.Success) { currentPassword = settings?.AdminPasswordHash; status = TryLoadSettings(uri, source, out passwordParams, out settings, currentPassword); } } else { status = TryLoadSettings(uri, source, out passwordParams, out settings); } if (status.HasValue) { return DetermineLoadResult(uri, source, settings, status.Value, passwordParams, currentPassword); } else { return OperationResult.Aborted; } } private OperationResult LoadSettingsForReconfiguration(Uri uri) { var currentPassword = Context.Current.Settings.AdminPasswordHash; var source = UriSource.Reconfiguration; var status = TryLoadSettings(uri, source, out var passwordParams, out var settings, currentPassword); if (status.HasValue) { return DetermineLoadResult(uri, source, settings, status.Value, passwordParams, currentPassword); } else { return OperationResult.Aborted; } } private OperationResult DetermineLoadResult(Uri uri, UriSource source, Settings settings, LoadStatus status, PasswordParameters passwordParams, string currentPassword = default(string)) { if (status == LoadStatus.LoadWithBrowser || status == LoadStatus.Success) { var isNewConfiguration = source == UriSource.CommandLine || source == UriSource.Reconfiguration; Context.Next.Settings = settings; if (status == LoadStatus.LoadWithBrowser) { return HandleBrowserResource(uri); } if (isNewConfiguration && settings.ConfigurationMode == ConfigurationMode.ConfigureClient) { return HandleClientConfiguration(uri, passwordParams, currentPassword); } return OperationResult.Success; } ShowFailureMessage(status, uri); return OperationResult.Failed; } private OperationResult HandleBrowserResource(Uri uri) { Context.Next.Settings.Browser.StartUrl = uri.AbsoluteUri; logger.Info($"The configuration resource needs authentication or is a webpage, using '{uri}' as startup URL for the browser."); return OperationResult.Success; } private OperationResult HandleClientConfiguration(Uri uri, PasswordParameters passwordParams, string currentPassword = default(string)) { var isFirstSession = Context.Current == null; var success = TryConfigureClient(uri, passwordParams, currentPassword); if (success == true) { if (isFirstSession && AbortAfterClientConfiguration()) { return OperationResult.Aborted; } return OperationResult.Success; } if (!success.HasValue) { return OperationResult.Aborted; } return OperationResult.Failed; } private LoadStatus? TryLoadSettings(Uri uri, UriSource source, out PasswordParameters passwordParams, out Settings settings, string currentPassword = default(string)) { passwordParams = new PasswordParameters { Password = string.Empty, IsHash = true }; var status = configuration.TryLoadSettings(uri, out settings, passwordParams); if (status == LoadStatus.PasswordNeeded && currentPassword != default(string)) { passwordParams.Password = currentPassword; passwordParams.IsHash = true; status = configuration.TryLoadSettings(uri, out settings, passwordParams); } for (int attempts = 0; attempts < 5 && status == LoadStatus.PasswordNeeded; attempts++) { var isLocalConfig = source == UriSource.AppData || source == UriSource.ProgramData; var purpose = isLocalConfig ? PasswordRequestPurpose.LocalSettings : PasswordRequestPurpose.Settings; var success = TryGetPassword(purpose, out var password); if (success) { passwordParams.Password = password; passwordParams.IsHash = false; } else { return null; } status = configuration.TryLoadSettings(uri, out settings, passwordParams); } return status; } private bool? TryConfigureClient(Uri uri, PasswordParameters passwordParams, string currentPassword = default(string)) { var mustAuthenticate = IsRequiredToAuthenticateForClientConfiguration(passwordParams, currentPassword); logger.Info("Starting client configuration..."); if (mustAuthenticate) { var authenticated = AuthenticateForClientConfiguration(currentPassword); if (authenticated == true) { logger.Info("Authentication was successful."); } if (authenticated == false) { logger.Info("Authentication has failed!"); ActionRequired?.Invoke(new InvalidPasswordMessageArgs()); return false; } if (!authenticated.HasValue) { logger.Info("Authentication was aborted."); return null; } } else { logger.Info("Authentication is not required."); } var status = configuration.ConfigureClientWith(uri, passwordParams); var success = status == SaveStatus.Success; if (success) { logger.Info("Client configuration was successful."); } else { logger.Error($"Client configuration failed with status '{status}'!"); ActionRequired?.Invoke(new ClientConfigurationErrorMessageArgs()); } return success; } private bool IsRequiredToAuthenticateForClientConfiguration(PasswordParameters passwordParams, string currentPassword = default(string)) { var mustAuthenticate = currentPassword != default(string); if (mustAuthenticate) { var nextPassword = Context.Next.Settings.AdminPasswordHash; var hasSettingsPassword = passwordParams.Password != null; var sameAdminPassword = currentPassword.Equals(nextPassword, StringComparison.OrdinalIgnoreCase); if (sameAdminPassword) { mustAuthenticate = false; } else if (hasSettingsPassword) { var settingsPassword = passwordParams.IsHash ? passwordParams.Password : hashAlgorithm.GenerateHashFor(passwordParams.Password); var knowsAdminPassword = currentPassword.Equals(settingsPassword, StringComparison.OrdinalIgnoreCase); mustAuthenticate = !knowsAdminPassword; } } return mustAuthenticate; } private bool? AuthenticateForClientConfiguration(string currentPassword) { var authenticated = false; for (int attempts = 0; attempts < 5 && !authenticated; attempts++) { var success = TryGetPassword(PasswordRequestPurpose.LocalAdministrator, out var password); if (success) { authenticated = currentPassword.Equals(hashAlgorithm.GenerateHashFor(password), StringComparison.OrdinalIgnoreCase); } else { return null; } } return authenticated; } private bool AbortAfterClientConfiguration() { var args = new ConfigurationCompletedEventArgs(); ActionRequired?.Invoke(args); logger.Info($"The user chose to {(args.AbortStartup ? "abort" : "continue")} startup after successful client configuration."); return args.AbortStartup; } private void ShowFailureMessage(LoadStatus status, Uri uri) { switch (status) { case LoadStatus.PasswordNeeded: ActionRequired?.Invoke(new InvalidPasswordMessageArgs()); break; case LoadStatus.InvalidData: ActionRequired?.Invoke(new InvalidDataMessageArgs(uri.ToString())); break; case LoadStatus.NotSupported: ActionRequired?.Invoke(new NotSupportedMessageArgs(uri.ToString())); break; case LoadStatus.UnexpectedError: ActionRequired?.Invoke(new UnexpectedErrorMessageArgs(uri.ToString())); break; } } private bool TryGetPassword(PasswordRequestPurpose purpose, out string password) { var args = new PasswordRequiredEventArgs { Purpose = purpose }; ActionRequired?.Invoke(args); password = args.Password; return args.Success; } private bool TryInitializeSettingsUri(out Uri uri, out UriSource source) { var isValidUri = false; uri = null; source = default(UriSource); if (commandLineArgs?.Length > 1) { isValidUri = Uri.TryCreate(commandLineArgs[1], UriKind.Absolute, out uri); source = UriSource.CommandLine; logger.Info($"Found command-line argument for configuration resource: '{uri}', the URI is {(isValidUri ? "valid" : "invalid")}."); } if (!isValidUri && File.Exists(ProgramDataFile)) { isValidUri = Uri.TryCreate(ProgramDataFile, UriKind.Absolute, out uri); source = UriSource.ProgramData; logger.Info($"Found configuration file in PROGRAMDATA directory: '{uri}'."); } if (!isValidUri && File.Exists(AppDataFile)) { isValidUri = Uri.TryCreate(AppDataFile, UriKind.Absolute, out uri); source = UriSource.AppData; logger.Info($"Found configuration file in APPDATA directory: '{uri}'."); } return isValidUri; } private bool TryValidateSettingsUri(string path, out Uri uri) { var isValidUri = Uri.TryCreate(path, UriKind.Absolute, out uri); isValidUri &= uri != null && uri.IsFile; isValidUri &= File.Exists(path); return isValidUri; } private void LogOperationResult(OperationResult result) { switch (result) { case OperationResult.Aborted: logger.Info("The configuration was aborted by the user."); break; case OperationResult.Failed: logger.Warn("The configuration has failed!"); break; case OperationResult.Success: logger.Info("The configuration was successful."); break; } } private enum UriSource { Undefined, AppData, CommandLine, ProgramData, Reconfiguration } } }