2019-11-13 10:11:11 +01:00
|
|
|
|
/*
|
2024-03-05 18:37:42 +01:00
|
|
|
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
2019-11-13 10:11:11 +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-29 14:59:54 +01:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Timers;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
using SafeExamBrowser.Applications.Contracts;
|
2019-11-29 14:59:54 +01:00
|
|
|
|
using SafeExamBrowser.Applications.Contracts.Events;
|
|
|
|
|
using SafeExamBrowser.Applications.Events;
|
2023-06-01 18:18:01 +02:00
|
|
|
|
using SafeExamBrowser.Core.Contracts.Resources.Icons;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts;
|
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Applications
|
|
|
|
|
{
|
2019-11-28 17:22:04 +01:00
|
|
|
|
internal class ExternalApplicationInstance
|
2019-11-13 10:11:11 +01:00
|
|
|
|
{
|
2019-11-29 14:59:54 +01:00
|
|
|
|
private readonly object @lock = new object();
|
|
|
|
|
|
2023-06-01 18:18:01 +02:00
|
|
|
|
private readonly IconResource icon;
|
|
|
|
|
private readonly ILogger logger;
|
|
|
|
|
private readonly INativeMethods nativeMethods;
|
|
|
|
|
private readonly IProcess process;
|
|
|
|
|
private readonly int windowMonitoringInterval;
|
|
|
|
|
private readonly IList<ExternalApplicationWindow> windows;
|
|
|
|
|
|
2019-11-29 14:59:54 +01:00
|
|
|
|
private Timer timer;
|
|
|
|
|
|
2019-12-02 15:48:06 +01:00
|
|
|
|
internal int Id { get; private set; }
|
2019-11-29 14:59:54 +01:00
|
|
|
|
|
|
|
|
|
internal event InstanceTerminatedEventHandler Terminated;
|
|
|
|
|
internal event WindowsChangedEventHandler WindowsChanged;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
|
2023-06-01 18:18:01 +02:00
|
|
|
|
internal ExternalApplicationInstance(
|
|
|
|
|
IconResource icon,
|
|
|
|
|
ILogger logger,
|
|
|
|
|
INativeMethods nativeMethods,
|
|
|
|
|
IProcess process,
|
|
|
|
|
int windowMonitoringInterval_ms)
|
2019-11-13 10:11:11 +01:00
|
|
|
|
{
|
2019-11-28 17:22:04 +01:00
|
|
|
|
this.icon = icon;
|
2019-11-13 10:11:11 +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.process = process;
|
2023-06-01 18:18:01 +02:00
|
|
|
|
this.windowMonitoringInterval = windowMonitoringInterval_ms;
|
2019-11-29 14:59:54 +01:00
|
|
|
|
this.windows = new List<ExternalApplicationWindow>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal IEnumerable<IApplicationWindow> GetWindows()
|
|
|
|
|
{
|
|
|
|
|
lock (@lock)
|
|
|
|
|
{
|
|
|
|
|
return new List<IApplicationWindow>(windows);
|
|
|
|
|
}
|
2019-11-13 10:11:11 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-11-28 17:22:04 +01:00
|
|
|
|
internal void Initialize()
|
2019-11-13 10:11:11 +01:00
|
|
|
|
{
|
2019-12-02 15:48:06 +01:00
|
|
|
|
Id = process.Id;
|
2019-11-29 14:59:54 +01:00
|
|
|
|
InitializeEvents();
|
|
|
|
|
logger.Info("Initialized application instance.");
|
2019-11-13 10:11:11 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-11-28 17:22:04 +01:00
|
|
|
|
internal void Terminate()
|
2019-11-13 10:11:11 +01:00
|
|
|
|
{
|
|
|
|
|
const int MAX_ATTEMPTS = 5;
|
|
|
|
|
const int TIMEOUT_MS = 500;
|
|
|
|
|
|
|
|
|
|
var terminated = process.HasTerminated;
|
|
|
|
|
|
2020-02-26 08:49:16 +01:00
|
|
|
|
if (terminated)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Application instance is already terminated.");
|
|
|
|
|
}
|
|
|
|
|
else
|
2019-11-13 10:11:11 +01:00
|
|
|
|
{
|
2019-11-29 14:59:54 +01:00
|
|
|
|
FinalizeEvents();
|
2019-11-13 11:43:34 +01:00
|
|
|
|
|
|
|
|
|
for (var attempt = 0; attempt < MAX_ATTEMPTS && !terminated; attempt++)
|
|
|
|
|
{
|
|
|
|
|
terminated = process.TryClose(TIMEOUT_MS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (var attempt = 0; attempt < MAX_ATTEMPTS && !terminated; attempt++)
|
|
|
|
|
{
|
|
|
|
|
terminated = process.TryKill(TIMEOUT_MS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (terminated)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Successfully terminated application instance.");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Warn("Failed to terminate application instance!");
|
|
|
|
|
}
|
2019-11-13 10:11:11 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Process_Terminated(int exitCode)
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"Application instance has terminated with exit code {exitCode}.");
|
2019-11-29 14:59:54 +01:00
|
|
|
|
FinalizeEvents();
|
|
|
|
|
Terminated?.Invoke(Id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Timer_Elapsed(object sender, ElapsedEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
var changed = false;
|
|
|
|
|
var openWindows = nativeMethods.GetOpenWindows();
|
|
|
|
|
|
|
|
|
|
lock (@lock)
|
|
|
|
|
{
|
|
|
|
|
var closedWindows = windows.Where(w => openWindows.All(ow => ow != w.Handle)).ToList();
|
2019-12-02 15:48:06 +01:00
|
|
|
|
var openedWindows = openWindows.Where(ow => windows.All(w => w.Handle != ow) && BelongsToInstance(ow)).ToList();
|
2019-11-29 14:59:54 +01:00
|
|
|
|
|
|
|
|
|
foreach (var window in closedWindows)
|
|
|
|
|
{
|
|
|
|
|
changed = true;
|
|
|
|
|
windows.Remove(window);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach (var window in openedWindows)
|
|
|
|
|
{
|
|
|
|
|
changed = true;
|
|
|
|
|
windows.Add(new ExternalApplicationWindow(icon, nativeMethods, window));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach (var window in windows)
|
|
|
|
|
{
|
|
|
|
|
window.Update();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (changed)
|
|
|
|
|
{
|
|
|
|
|
WindowsChanged?.Invoke();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
timer.Start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool BelongsToInstance(IntPtr window)
|
|
|
|
|
{
|
|
|
|
|
return nativeMethods.GetProcessIdFor(window) == process.Id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void InitializeEvents()
|
|
|
|
|
{
|
|
|
|
|
process.Terminated += Process_Terminated;
|
|
|
|
|
|
2023-06-01 18:18:01 +02:00
|
|
|
|
timer = new Timer(windowMonitoringInterval);
|
2019-11-29 14:59:54 +01:00
|
|
|
|
timer.Elapsed += Timer_Elapsed;
|
|
|
|
|
timer.Start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void FinalizeEvents()
|
|
|
|
|
{
|
2023-06-02 15:51:45 +02:00
|
|
|
|
if (timer != default)
|
|
|
|
|
{
|
|
|
|
|
timer.Elapsed -= Timer_Elapsed;
|
|
|
|
|
timer.Stop();
|
|
|
|
|
}
|
2019-11-29 14:59:54 +01:00
|
|
|
|
|
|
|
|
|
process.Terminated -= Process_Terminated;
|
2019-11-13 10:11:11 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|