2018-02-06 15:12:11 +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/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
using System;
|
2018-11-08 09:39:52 +01:00
|
|
|
|
using System.Collections.Generic;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
using System.IO;
|
2018-11-08 09:39:52 +01:00
|
|
|
|
using System.IO.Compression;
|
|
|
|
|
using System.Linq;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Configuration;
|
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.Settings;
|
2018-08-31 07:49:41 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.Logging;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Configuration
|
|
|
|
|
{
|
|
|
|
|
public class ConfigurationRepository : IConfigurationRepository
|
|
|
|
|
{
|
2018-03-14 11:04:28 +01:00
|
|
|
|
private const string BASE_ADDRESS = "net.pipe://localhost/safeexambrowser";
|
|
|
|
|
|
2018-09-04 10:58:56 +02:00
|
|
|
|
private readonly string executablePath;
|
|
|
|
|
private readonly string programCopyright;
|
|
|
|
|
private readonly string programTitle;
|
|
|
|
|
private readonly string programVersion;
|
|
|
|
|
|
2018-06-29 09:50:20 +02:00
|
|
|
|
private AppConfig appConfig;
|
2018-11-08 09:39:52 +01:00
|
|
|
|
private IList<IDataFormat> dataFormats;
|
|
|
|
|
private ILogger logger;
|
|
|
|
|
private IList<IResourceLoader> resourceLoaders;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
|
2018-11-08 09:39:52 +01:00
|
|
|
|
public ConfigurationRepository(ILogger logger, string executablePath, string programCopyright, string programTitle, string programVersion)
|
2018-09-04 10:58:56 +02:00
|
|
|
|
{
|
2018-11-08 09:39:52 +01:00
|
|
|
|
dataFormats = new List<IDataFormat>();
|
|
|
|
|
resourceLoaders = new List<IResourceLoader>();
|
|
|
|
|
|
|
|
|
|
this.logger = logger;
|
2018-10-12 11:16:59 +02:00
|
|
|
|
this.executablePath = executablePath ?? string.Empty;
|
|
|
|
|
this.programCopyright = programCopyright ?? string.Empty;
|
|
|
|
|
this.programTitle = programTitle ?? string.Empty;
|
|
|
|
|
this.programVersion = programVersion ?? string.Empty;
|
2018-02-06 15:12:11 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-12 11:16:59 +02:00
|
|
|
|
public AppConfig InitializeAppConfig()
|
2018-02-06 15:12:11 +01:00
|
|
|
|
{
|
|
|
|
|
var appDataFolder = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), nameof(SafeExamBrowser));
|
|
|
|
|
var startTime = DateTime.Now;
|
|
|
|
|
var logFolder = Path.Combine(appDataFolder, "Logs");
|
|
|
|
|
var logFilePrefix = startTime.ToString("yyyy-MM-dd\\_HH\\hmm\\mss\\s");
|
|
|
|
|
|
2018-06-29 09:50:20 +02:00
|
|
|
|
appConfig = new AppConfig();
|
|
|
|
|
appConfig.ApplicationStartTime = startTime;
|
|
|
|
|
appConfig.AppDataFolder = appDataFolder;
|
|
|
|
|
appConfig.BrowserCachePath = Path.Combine(appDataFolder, "Cache");
|
2018-08-16 11:23:37 +02:00
|
|
|
|
appConfig.BrowserLogFile = Path.Combine(logFolder, $"{logFilePrefix}_Browser.log");
|
2018-06-29 09:50:20 +02:00
|
|
|
|
appConfig.ClientId = Guid.NewGuid();
|
|
|
|
|
appConfig.ClientAddress = $"{BASE_ADDRESS}/client/{Guid.NewGuid()}";
|
2018-09-04 10:58:56 +02:00
|
|
|
|
appConfig.ClientExecutablePath = Path.Combine(Path.GetDirectoryName(executablePath), $"{nameof(SafeExamBrowser)}.Client.exe");
|
2018-08-16 11:23:37 +02:00
|
|
|
|
appConfig.ClientLogFile = Path.Combine(logFolder, $"{logFilePrefix}_Client.log");
|
2018-06-29 09:50:20 +02:00
|
|
|
|
appConfig.ConfigurationFileExtension = ".seb";
|
|
|
|
|
appConfig.DefaultSettingsFileName = "SebClientSettings.seb";
|
|
|
|
|
appConfig.DownloadDirectory = Path.Combine(appDataFolder, "Downloads");
|
2018-08-31 07:49:41 +02:00
|
|
|
|
appConfig.LogLevel = LogLevel.Debug;
|
2018-09-04 10:58:56 +02:00
|
|
|
|
appConfig.ProgramCopyright = programCopyright;
|
2018-06-29 09:50:20 +02:00
|
|
|
|
appConfig.ProgramDataFolder = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.CommonApplicationData), nameof(SafeExamBrowser));
|
2018-09-04 10:58:56 +02:00
|
|
|
|
appConfig.ProgramTitle = programTitle;
|
|
|
|
|
appConfig.ProgramVersion = programVersion;
|
2018-06-29 09:50:20 +02:00
|
|
|
|
appConfig.RuntimeId = Guid.NewGuid();
|
|
|
|
|
appConfig.RuntimeAddress = $"{BASE_ADDRESS}/runtime/{Guid.NewGuid()}";
|
2018-08-16 11:23:37 +02:00
|
|
|
|
appConfig.RuntimeLogFile = Path.Combine(logFolder, $"{logFilePrefix}_Runtime.log");
|
2018-08-14 09:06:35 +02:00
|
|
|
|
appConfig.SebUriScheme = "seb";
|
|
|
|
|
appConfig.SebUriSchemeSecure = "sebs";
|
2018-06-29 09:50:20 +02:00
|
|
|
|
appConfig.ServiceAddress = $"{BASE_ADDRESS}/service";
|
2018-10-12 11:16:59 +02:00
|
|
|
|
|
|
|
|
|
return appConfig;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public ISessionConfiguration InitializeSessionConfiguration()
|
|
|
|
|
{
|
|
|
|
|
var configuration = new SessionConfiguration();
|
|
|
|
|
|
|
|
|
|
UpdateAppConfig();
|
|
|
|
|
|
2018-11-08 09:39:52 +01:00
|
|
|
|
configuration.AppConfig = appConfig.Clone();
|
2018-10-12 11:16:59 +02:00
|
|
|
|
configuration.Id = Guid.NewGuid();
|
|
|
|
|
configuration.StartupToken = Guid.NewGuid();
|
|
|
|
|
|
|
|
|
|
return configuration;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Settings LoadDefaultSettings()
|
|
|
|
|
{
|
|
|
|
|
var settings = new Settings();
|
|
|
|
|
|
2018-11-08 09:39:52 +01:00
|
|
|
|
settings.KioskMode = KioskMode.None;
|
2018-10-12 11:16:59 +02:00
|
|
|
|
settings.ServicePolicy = ServicePolicy.Optional;
|
|
|
|
|
|
|
|
|
|
settings.Browser.StartUrl = "https://www.safeexambrowser.org/testing";
|
|
|
|
|
settings.Browser.AllowAddressBar = true;
|
|
|
|
|
settings.Browser.AllowBackwardNavigation = true;
|
2018-11-08 09:39:52 +01:00
|
|
|
|
settings.Browser.AllowConfigurationDownloads = true;
|
2018-10-12 11:16:59 +02:00
|
|
|
|
settings.Browser.AllowDeveloperConsole = true;
|
2018-11-08 09:39:52 +01:00
|
|
|
|
settings.Browser.AllowDownloads = true;
|
2018-10-12 11:16:59 +02:00
|
|
|
|
settings.Browser.AllowForwardNavigation = true;
|
|
|
|
|
settings.Browser.AllowReloading = true;
|
|
|
|
|
|
|
|
|
|
settings.Taskbar.AllowApplicationLog = true;
|
|
|
|
|
settings.Taskbar.AllowKeyboardLayout = true;
|
|
|
|
|
settings.Taskbar.AllowWirelessNetwork = true;
|
|
|
|
|
|
|
|
|
|
return settings;
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-08 09:39:52 +01:00
|
|
|
|
public void Register(IDataFormat dataFormat)
|
|
|
|
|
{
|
|
|
|
|
dataFormats.Add(dataFormat);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Register(IResourceLoader resourceLoader)
|
|
|
|
|
{
|
|
|
|
|
resourceLoaders.Add(resourceLoader);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public LoadStatus TryLoadSettings(Uri resource, out Settings settings, string adminPassword = null, string settingsPassword = null)
|
2018-10-12 11:16:59 +02:00
|
|
|
|
{
|
2018-11-08 09:39:52 +01:00
|
|
|
|
settings = default(Settings);
|
|
|
|
|
|
|
|
|
|
logger.Info($"Attempting to load '{resource}'...");
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
2018-11-09 14:15:56 +01:00
|
|
|
|
var status = TryLoadData(resource, out Stream data);
|
2018-11-08 09:39:52 +01:00
|
|
|
|
|
2018-11-09 14:15:56 +01:00
|
|
|
|
switch (status)
|
2018-11-08 09:39:52 +01:00
|
|
|
|
{
|
2018-11-09 14:15:56 +01:00
|
|
|
|
case LoadStatus.LoadWithBrowser:
|
|
|
|
|
return HandleBrowserResource(resource, out settings);
|
|
|
|
|
case LoadStatus.Success:
|
|
|
|
|
return TryParseData(data, out settings, adminPassword, settingsPassword);
|
2018-11-08 09:39:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-11-09 14:15:56 +01:00
|
|
|
|
return status;
|
2018-11-08 09:39:52 +01:00
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error($"Unexpected error while trying to load '{resource}'!", e);
|
|
|
|
|
|
|
|
|
|
return LoadStatus.UnexpectedError;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-09 14:15:56 +01:00
|
|
|
|
private LoadStatus TryLoadData(Uri resource, out Stream data)
|
2018-11-08 09:39:52 +01:00
|
|
|
|
{
|
2018-11-09 14:15:56 +01:00
|
|
|
|
var status = LoadStatus.NotSupported;
|
|
|
|
|
var resourceLoader = resourceLoaders.FirstOrDefault(l => l.CanLoad(resource));
|
|
|
|
|
|
|
|
|
|
data = default(Stream);
|
|
|
|
|
|
|
|
|
|
if (resourceLoader != null)
|
|
|
|
|
{
|
|
|
|
|
status = resourceLoader.TryLoad(resource, out data);
|
|
|
|
|
logger.Info($"Tried to load data from '{resource}' using {resourceLoader.GetType().Name} -> Result: {status}.");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Warn($"No resource loader found for '{resource}'!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return status;
|
|
|
|
|
}
|
2018-11-08 09:39:52 +01:00
|
|
|
|
|
2018-11-09 14:15:56 +01:00
|
|
|
|
private LoadStatus TryParseData(Stream data, out Settings settings, string adminPassword, string settingsPassword)
|
|
|
|
|
{
|
|
|
|
|
var status = LoadStatus.NotSupported;
|
|
|
|
|
var dataFormat = dataFormats.FirstOrDefault(f => f.CanParse(data));
|
|
|
|
|
|
|
|
|
|
settings = default(Settings);
|
|
|
|
|
|
|
|
|
|
if (dataFormat != null)
|
|
|
|
|
{
|
|
|
|
|
status = dataFormat.TryParse(data, out settings, adminPassword, settingsPassword);
|
|
|
|
|
logger.Info($"Tried to parse data from '{data}' using {dataFormat.GetType().Name} -> Result: {status}.");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Warn($"No data format found for '{data}'!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private LoadStatus HandleBrowserResource(Uri resource, out Settings settings)
|
|
|
|
|
{
|
2018-11-08 09:39:52 +01:00
|
|
|
|
settings = LoadDefaultSettings();
|
|
|
|
|
settings.Browser.StartUrl = resource.AbsoluteUri;
|
|
|
|
|
|
2018-11-09 14:15:56 +01:00
|
|
|
|
logger.Info($"The resource needs authentication or is HTML data, loaded default settings with '{resource}' as startup URL.");
|
|
|
|
|
|
2018-11-08 09:39:52 +01:00
|
|
|
|
return LoadStatus.Success;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private byte[] Decompress(byte[] bytes)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var buffer = new byte[4096];
|
|
|
|
|
|
|
|
|
|
using (var stream = new GZipStream(new MemoryStream(bytes), CompressionMode.Decompress))
|
|
|
|
|
using (var decompressed = new MemoryStream())
|
|
|
|
|
{
|
|
|
|
|
var bytesRead = 0;
|
|
|
|
|
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
bytesRead = stream.Read(buffer, 0, buffer.Length);
|
|
|
|
|
decompressed.Write(buffer, 0, bytesRead);
|
|
|
|
|
} while (bytesRead > 0);
|
|
|
|
|
|
|
|
|
|
return decompressed.ToArray();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (InvalidDataException)
|
2018-10-12 11:16:59 +02:00
|
|
|
|
{
|
2018-11-08 09:39:52 +01:00
|
|
|
|
return bytes;
|
|
|
|
|
}
|
2018-02-06 15:12:11 +01:00
|
|
|
|
}
|
2018-03-14 11:04:28 +01:00
|
|
|
|
|
2018-06-29 09:50:20 +02:00
|
|
|
|
private void UpdateAppConfig()
|
2018-03-14 11:04:28 +01:00
|
|
|
|
{
|
2018-10-12 11:16:59 +02:00
|
|
|
|
appConfig.ClientId = Guid.NewGuid();
|
|
|
|
|
appConfig.ClientAddress = $"{BASE_ADDRESS}/client/{Guid.NewGuid()}";
|
2018-03-14 11:04:28 +01:00
|
|
|
|
}
|
2018-02-06 15:12:11 +01:00
|
|
|
|
}
|
|
|
|
|
}
|