2018-02-14 15:26:05 +01:00
|
|
|
|
/*
|
2019-01-09 11:25:21 +01:00
|
|
|
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
2018-02-14 15:26:05 +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-10-04 16:36:12 +02:00
|
|
|
|
using System;
|
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Management;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts;
|
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts.Events;
|
2018-02-14 15:26:05 +01:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.WindowsApi
|
|
|
|
|
{
|
|
|
|
|
internal class Process : IProcess
|
|
|
|
|
{
|
2019-10-04 16:36:12 +02:00
|
|
|
|
private bool eventInitialized, originalNameInitialized;
|
|
|
|
|
private ILogger logger;
|
|
|
|
|
private string originalName;
|
2018-02-14 15:26:05 +01:00
|
|
|
|
private System.Diagnostics.Process process;
|
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
private event ProcessTerminatedEventHandler TerminatedEvent;
|
2018-02-20 15:15:26 +01:00
|
|
|
|
|
2018-02-14 15:26:05 +01:00
|
|
|
|
public int Id
|
|
|
|
|
{
|
|
|
|
|
get { return process.Id; }
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-20 15:15:26 +01:00
|
|
|
|
public bool HasTerminated
|
|
|
|
|
{
|
2019-10-04 16:36:12 +02:00
|
|
|
|
get { return IsTerminated(); }
|
2018-02-20 15:15:26 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
public string Name { get; }
|
|
|
|
|
|
|
|
|
|
public string OriginalName
|
|
|
|
|
{
|
|
|
|
|
get { return originalNameInitialized ? originalName : InitializeOriginalName(); }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public event ProcessTerminatedEventHandler Terminated
|
|
|
|
|
{
|
|
|
|
|
add { TerminatedEvent += value; InitializeEvent(); }
|
|
|
|
|
remove { TerminatedEvent -= value; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal Process(System.Diagnostics.Process process, ILogger logger)
|
|
|
|
|
{
|
|
|
|
|
this.Name = process.ProcessName;
|
|
|
|
|
this.process = process;
|
|
|
|
|
this.logger = logger;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal Process(System.Diagnostics.Process process, string originalName, ILogger logger) : this(process, logger)
|
|
|
|
|
{
|
|
|
|
|
this.originalName = originalName;
|
|
|
|
|
this.originalNameInitialized = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool TryClose()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
process.Refresh();
|
|
|
|
|
|
|
|
|
|
if (!process.HasExited)
|
|
|
|
|
{
|
|
|
|
|
process.CloseMainWindow();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return process.HasExited;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to close main window!", e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool TryKill()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
process.Refresh();
|
|
|
|
|
|
|
|
|
|
if (!process.HasExited)
|
|
|
|
|
{
|
|
|
|
|
process.Kill();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return process.HasExited;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to kill process!", e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool IsTerminated()
|
2018-02-14 15:26:05 +01:00
|
|
|
|
{
|
2019-10-04 16:36:12 +02:00
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
process.Refresh();
|
|
|
|
|
|
|
|
|
|
return process.HasExited;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Failed to check whether process is terminated!", e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void InitializeEvent()
|
|
|
|
|
{
|
|
|
|
|
if (!eventInitialized)
|
|
|
|
|
{
|
|
|
|
|
eventInitialized = true;
|
|
|
|
|
process.Exited += Process_Exited;
|
|
|
|
|
process.EnableRaisingEvents = true;
|
|
|
|
|
}
|
2018-02-20 15:15:26 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
private string InitializeOriginalName()
|
2018-02-20 15:15:26 +01:00
|
|
|
|
{
|
2019-10-04 16:36:12 +02:00
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
using (var searcher = new ManagementObjectSearcher($"SELECT ExecutablePath FROM Win32_Process WHERE ProcessId = {process.Id}"))
|
|
|
|
|
using (var results = searcher.Get())
|
|
|
|
|
using (var processData = results.Cast<ManagementObject>().First())
|
|
|
|
|
{
|
|
|
|
|
var executablePath = Convert.ToString(processData["ExecutablePath"]);
|
|
|
|
|
|
|
|
|
|
if (File.Exists(executablePath))
|
|
|
|
|
{
|
|
|
|
|
var executableInfo = FileVersionInfo.GetVersionInfo(executablePath);
|
|
|
|
|
var originalName = Path.GetFileNameWithoutExtension(executableInfo.OriginalFilename);
|
2018-09-25 12:10:34 +02:00
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
this.originalName = originalName;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Warn("Could not find original name!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
2018-09-25 12:10:34 +02:00
|
|
|
|
{
|
2019-10-04 16:36:12 +02:00
|
|
|
|
logger.Error("Failed to initialize original name!", e);
|
2018-09-25 12:10:34 +02:00
|
|
|
|
}
|
2019-10-04 16:36:12 +02:00
|
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
originalNameInitialized = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return originalName;
|
2018-02-20 15:15:26 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-10-04 16:36:12 +02:00
|
|
|
|
private void Process_Exited(object sender, EventArgs e)
|
2018-02-20 15:15:26 +01:00
|
|
|
|
{
|
2019-10-04 16:36:12 +02:00
|
|
|
|
TerminatedEvent?.Invoke(process.ExitCode);
|
2018-02-14 15:26:05 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|