2019-11-05 10:08:19 +01:00
|
|
|
|
/*
|
2023-03-08 00:30:20 +01:00
|
|
|
|
* Copyright (c) 2023 ETH Zürich, Educational Development and Technology (LET)
|
2019-11-05 10:08:19 +01:00
|
|
|
|
*
|
|
|
|
|
* 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.Collections.Generic;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using SafeExamBrowser.Applications.Contracts;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2019-12-02 15:48:06 +01:00
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
using SafeExamBrowser.Settings.Applications;
|
2023-06-01 18:18:01 +02:00
|
|
|
|
using SafeExamBrowser.SystemComponents.Contracts.Registry;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Applications
|
|
|
|
|
{
|
|
|
|
|
public class ApplicationFactory : IApplicationFactory
|
|
|
|
|
{
|
2023-05-30 18:08:50 +02:00
|
|
|
|
private readonly IApplicationMonitor applicationMonitor;
|
|
|
|
|
private readonly IModuleLogger logger;
|
|
|
|
|
private readonly INativeMethods nativeMethods;
|
|
|
|
|
private readonly IProcessFactory processFactory;
|
2023-06-01 18:18:01 +02:00
|
|
|
|
private readonly IRegistry registry;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
public ApplicationFactory(
|
|
|
|
|
IApplicationMonitor applicationMonitor,
|
|
|
|
|
IModuleLogger logger,
|
|
|
|
|
INativeMethods nativeMethods,
|
2023-06-01 18:18:01 +02:00
|
|
|
|
IProcessFactory processFactory,
|
|
|
|
|
IRegistry registry)
|
2019-11-05 10:08:19 +01:00
|
|
|
|
{
|
2019-12-02 15:48:06 +01:00
|
|
|
|
this.applicationMonitor = applicationMonitor;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
this.logger = logger;
|
2019-11-29 14:59:54 +01:00
|
|
|
|
this.nativeMethods = nativeMethods;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
this.processFactory = processFactory;
|
2023-06-01 18:18:01 +02:00
|
|
|
|
this.registry = registry;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
2024-02-13 11:04:36 +01:00
|
|
|
|
public FactoryResult TryCreate(WhitelistApplication settings, out IApplication<IApplicationWindow> application)
|
2019-11-05 10:08:19 +01:00
|
|
|
|
{
|
2023-05-30 18:08:50 +02:00
|
|
|
|
var name = $"'{settings.DisplayName}' ({settings.ExecutableName})";
|
2019-11-06 08:45:37 +01:00
|
|
|
|
|
2023-05-30 18:08:50 +02:00
|
|
|
|
application = default;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
2019-11-06 08:45:37 +01:00
|
|
|
|
var success = TryFindApplication(settings, out var executablePath);
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
|
|
|
|
if (success)
|
|
|
|
|
{
|
2019-11-06 08:45:37 +01:00
|
|
|
|
application = BuildApplication(executablePath, settings);
|
|
|
|
|
application.Initialize();
|
|
|
|
|
|
|
|
|
|
logger.Debug($"Successfully initialized application {name}.");
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
|
|
|
|
return FactoryResult.Success;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-06 08:45:37 +01:00
|
|
|
|
logger.Error($"Could not find application {name}!");
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
|
|
|
|
return FactoryResult.NotFound;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
2019-11-06 08:45:37 +01:00
|
|
|
|
logger.Error($"Unexpected error while trying to initialize application {name}!", e);
|
2019-11-05 10:08:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return FactoryResult.Error;
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-13 11:04:36 +01:00
|
|
|
|
private IApplication<IApplicationWindow> BuildApplication(string executablePath, WhitelistApplication settings)
|
2019-11-06 08:45:37 +01:00
|
|
|
|
{
|
2023-06-01 18:18:01 +02:00
|
|
|
|
const int ONE_SECOND = 1000;
|
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
var applicationLogger = logger.CloneFor(settings.DisplayName);
|
2023-06-01 18:18:01 +02:00
|
|
|
|
var application = new ExternalApplication(applicationMonitor, executablePath, applicationLogger, nativeMethods, processFactory, settings, ONE_SECOND);
|
2019-11-06 08:45:37 +01:00
|
|
|
|
|
|
|
|
|
return application;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool TryFindApplication(WhitelistApplication settings, out string mainExecutable)
|
2019-11-05 10:08:19 +01:00
|
|
|
|
{
|
|
|
|
|
var paths = new List<string[]>();
|
|
|
|
|
var registryPath = QueryPathFromRegistry(settings);
|
|
|
|
|
|
2023-05-30 18:08:50 +02:00
|
|
|
|
mainExecutable = default;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
2019-11-06 15:45:17 +01:00
|
|
|
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFiles), settings.ExecutableName });
|
2019-11-05 10:08:19 +01:00
|
|
|
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFilesX86), settings.ExecutableName });
|
|
|
|
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.System), settings.ExecutableName });
|
|
|
|
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.SystemX86), settings.ExecutableName });
|
|
|
|
|
|
2023-05-30 18:08:50 +02:00
|
|
|
|
if (settings.ExecutablePath != default)
|
2019-11-05 10:08:19 +01:00
|
|
|
|
{
|
|
|
|
|
paths.Add(new[] { settings.ExecutablePath, settings.ExecutableName });
|
2019-11-06 15:45:17 +01:00
|
|
|
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFiles), settings.ExecutablePath, settings.ExecutableName });
|
2019-11-05 10:08:19 +01:00
|
|
|
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.ProgramFilesX86), settings.ExecutablePath, settings.ExecutableName });
|
|
|
|
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.System), settings.ExecutablePath, settings.ExecutableName });
|
|
|
|
|
paths.Add(new[] { Environment.GetFolderPath(Environment.SpecialFolder.SystemX86), settings.ExecutablePath, settings.ExecutableName });
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-30 18:08:50 +02:00
|
|
|
|
if (registryPath != default)
|
2019-11-05 10:08:19 +01:00
|
|
|
|
{
|
|
|
|
|
paths.Add(new[] { registryPath, settings.ExecutableName });
|
|
|
|
|
|
2023-05-30 18:08:50 +02:00
|
|
|
|
if (settings.ExecutablePath != default)
|
2019-11-05 10:08:19 +01:00
|
|
|
|
{
|
|
|
|
|
paths.Add(new[] { registryPath, settings.ExecutablePath, settings.ExecutableName });
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach (var path in paths)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
mainExecutable = Path.Combine(path);
|
|
|
|
|
mainExecutable = Environment.ExpandEnvironmentVariables(mainExecutable);
|
|
|
|
|
|
|
|
|
|
if (File.Exists(mainExecutable))
|
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error($"Failed to test path {string.Join(@"\", path)}!", e);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string QueryPathFromRegistry(WhitelistApplication settings)
|
|
|
|
|
{
|
2023-06-01 18:18:01 +02:00
|
|
|
|
if (registry.TryRead($@"{RegistryValue.MachineHive.AppPaths_Key}\{settings.ExecutableName}", "Path", out var value))
|
2019-11-05 10:08:19 +01:00
|
|
|
|
{
|
2023-06-01 18:18:01 +02:00
|
|
|
|
return value as string;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
2023-05-30 18:08:50 +02:00
|
|
|
|
return default;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|