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/.
|
|
|
|
|
*/
|
|
|
|
|
|
2019-11-21 16:07:13 +01:00
|
|
|
|
using System;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
using SafeExamBrowser.Applications.Contracts;
|
|
|
|
|
using SafeExamBrowser.Applications.Contracts.Events;
|
2021-03-18 23:12:07 +01:00
|
|
|
|
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2019-12-02 15:48:06 +01:00
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.Applications;
|
|
|
|
|
using SafeExamBrowser.Settings.Applications;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Applications
|
|
|
|
|
{
|
2024-02-13 11:04:36 +01:00
|
|
|
|
internal class ExternalApplication : IApplication<IApplicationWindow>
|
2019-11-05 10:08:19 +01:00
|
|
|
|
{
|
2019-12-02 15:48:06 +01:00
|
|
|
|
private readonly object @lock = new object();
|
2019-11-29 14:59:54 +01:00
|
|
|
|
|
2023-06-01 18:18:01 +02:00
|
|
|
|
private readonly IApplicationMonitor applicationMonitor;
|
|
|
|
|
private readonly string executablePath;
|
|
|
|
|
private readonly IList<ExternalApplicationInstance> instances;
|
|
|
|
|
private readonly IModuleLogger logger;
|
|
|
|
|
private readonly INativeMethods nativeMethods;
|
|
|
|
|
private readonly IProcessFactory processFactory;
|
|
|
|
|
private readonly WhitelistApplication settings;
|
|
|
|
|
private readonly int windowMonitoringInterval;
|
2019-11-06 08:45:37 +01:00
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
public bool AutoStart { get; private set; }
|
|
|
|
|
public IconResource Icon { get; private set; }
|
|
|
|
|
public Guid Id { get; private set; }
|
|
|
|
|
public string Name { get; private set; }
|
|
|
|
|
public string Tooltip { get; private set; }
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
public event WindowsChangedEventHandler WindowsChanged;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
|
2019-11-29 14:59:54 +01:00
|
|
|
|
internal ExternalApplication(
|
2019-12-02 15:48:06 +01:00
|
|
|
|
IApplicationMonitor applicationMonitor,
|
2019-11-29 14:59:54 +01:00
|
|
|
|
string executablePath,
|
|
|
|
|
IModuleLogger logger,
|
|
|
|
|
INativeMethods nativeMethods,
|
2019-12-02 15:48:06 +01:00
|
|
|
|
IProcessFactory processFactory,
|
2023-06-01 18:18:01 +02:00
|
|
|
|
WhitelistApplication settings,
|
|
|
|
|
int windowMonitoringInterval_ms)
|
2019-11-06 08:45:37 +01:00
|
|
|
|
{
|
2019-12-02 15:48:06 +01:00
|
|
|
|
this.applicationMonitor = applicationMonitor;
|
2019-11-06 08:45:37 +01:00
|
|
|
|
this.executablePath = executablePath;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
this.logger = logger;
|
2019-11-29 14:59:54 +01:00
|
|
|
|
this.nativeMethods = nativeMethods;
|
2019-11-28 17:22:04 +01:00
|
|
|
|
this.instances = new List<ExternalApplicationInstance>();
|
2019-11-13 10:11:11 +01:00
|
|
|
|
this.processFactory = processFactory;
|
2019-12-02 15:48:06 +01:00
|
|
|
|
this.settings = settings;
|
2023-06-01 18:18:01 +02:00
|
|
|
|
this.windowMonitoringInterval = windowMonitoringInterval_ms;
|
2019-11-06 08:45:37 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-11-28 17:22:04 +01:00
|
|
|
|
public IEnumerable<IApplicationWindow> GetWindows()
|
|
|
|
|
{
|
2019-12-02 15:48:06 +01:00
|
|
|
|
lock (@lock)
|
|
|
|
|
{
|
|
|
|
|
return instances.SelectMany(i => i.GetWindows());
|
|
|
|
|
}
|
2019-11-28 17:22:04 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-11-05 10:08:19 +01:00
|
|
|
|
public void Initialize()
|
|
|
|
|
{
|
2019-12-02 15:48:06 +01:00
|
|
|
|
AutoStart = settings.AutoStart;
|
2019-12-03 15:43:48 +01:00
|
|
|
|
Icon = new EmbeddedIconResource { FilePath = executablePath };
|
2019-12-02 15:48:06 +01:00
|
|
|
|
Id = settings.Id;
|
|
|
|
|
Name = settings.DisplayName;
|
|
|
|
|
Tooltip = settings.Description ?? settings.DisplayName;
|
|
|
|
|
|
|
|
|
|
applicationMonitor.InstanceStarted += ApplicationMonitor_InstanceStarted;
|
2019-11-05 10:08:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Start()
|
|
|
|
|
{
|
2019-11-21 16:07:13 +01:00
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Starting application...");
|
2020-04-29 18:15:50 +02:00
|
|
|
|
InitializeInstance(processFactory.StartNew(executablePath, BuildArguments()));
|
2020-02-26 08:49:16 +01:00
|
|
|
|
logger.Info("Successfully started application.");
|
2019-11-21 16:07:13 +01:00
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to start application!", e);
|
|
|
|
|
}
|
2019-11-05 10:08:19 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
public void Terminate()
|
|
|
|
|
{
|
|
|
|
|
applicationMonitor.InstanceStarted -= ApplicationMonitor_InstanceStarted;
|
|
|
|
|
|
2020-02-26 08:49:16 +01:00
|
|
|
|
try
|
2019-12-02 15:48:06 +01:00
|
|
|
|
{
|
2020-02-26 08:49:16 +01:00
|
|
|
|
lock (@lock)
|
2019-12-02 15:48:06 +01:00
|
|
|
|
{
|
2020-02-26 08:49:16 +01:00
|
|
|
|
if (instances.Any() && !settings.AllowRunning)
|
2019-12-02 15:48:06 +01:00
|
|
|
|
{
|
2020-02-26 08:49:16 +01:00
|
|
|
|
logger.Info($"Terminating application with {instances.Count} instance(s)...");
|
|
|
|
|
|
|
|
|
|
foreach (var instance in instances)
|
|
|
|
|
{
|
|
|
|
|
instance.Terminated -= Instance_Terminated;
|
|
|
|
|
instance.Terminate();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
logger.Info("Successfully terminated application.");
|
2019-12-02 15:48:06 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-02-26 08:49:16 +01:00
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error($"Failed to terminate application!", e);
|
|
|
|
|
}
|
2019-12-02 15:48:06 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ApplicationMonitor_InstanceStarted(Guid applicationId, IProcess process)
|
|
|
|
|
{
|
2019-12-02 17:00:55 +01:00
|
|
|
|
lock (@lock)
|
2019-12-02 15:48:06 +01:00
|
|
|
|
{
|
2019-12-02 17:00:55 +01:00
|
|
|
|
var isNewInstance = instances.All(i => i.Id != process.Id);
|
|
|
|
|
|
|
|
|
|
if (applicationId == Id && isNewInstance)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("New application instance was started.");
|
|
|
|
|
InitializeInstance(process);
|
|
|
|
|
}
|
2019-12-02 15:48:06 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-29 14:59:54 +01:00
|
|
|
|
private void Instance_Terminated(int id)
|
|
|
|
|
{
|
2019-12-02 15:48:06 +01:00
|
|
|
|
lock (@lock)
|
|
|
|
|
{
|
|
|
|
|
instances.Remove(instances.First(i => i.Id == id));
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-29 14:59:54 +01:00
|
|
|
|
WindowsChanged?.Invoke();
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-01 18:18:01 +02:00
|
|
|
|
private string[] BuildArguments()
|
|
|
|
|
{
|
|
|
|
|
var arguments = new List<string>();
|
|
|
|
|
|
|
|
|
|
foreach (var argument in settings.Arguments)
|
|
|
|
|
{
|
|
|
|
|
arguments.Add(Environment.ExpandEnvironmentVariables(argument));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return arguments.ToArray();
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
private void InitializeInstance(IProcess process)
|
2019-11-05 10:08:19 +01:00
|
|
|
|
{
|
2019-12-02 15:48:06 +01:00
|
|
|
|
lock (@lock)
|
|
|
|
|
{
|
2019-12-02 17:00:55 +01:00
|
|
|
|
var instanceLogger = logger.CloneFor($"{Name} ({process.Id})");
|
2023-06-01 18:18:01 +02:00
|
|
|
|
var instance = new ExternalApplicationInstance(Icon, instanceLogger, nativeMethods, process, windowMonitoringInterval);
|
2019-12-02 17:00:55 +01:00
|
|
|
|
|
|
|
|
|
instance.Terminated += Instance_Terminated;
|
|
|
|
|
instance.WindowsChanged += () => WindowsChanged?.Invoke();
|
|
|
|
|
instance.Initialize();
|
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
instances.Add(instance);
|
2019-11-13 10:11:11 +01:00
|
|
|
|
}
|
2019-11-05 10:08:19 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|