/* * Copyright (c) 2024 ETH Zürich, IT Services * * 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.Text; using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.WindowsApi.Contracts; using SafeExamBrowser.WindowsApi.Contracts.Events; namespace SafeExamBrowser.WindowsApi.Processes { internal class Process : IProcess { private readonly ILogger logger; private readonly System.Diagnostics.Process process; private bool eventInitialized; public bool HasTerminated { get { return IsTerminated(); } } public int Id { get { return process.Id; } } public string Name { get; } public string OriginalName { get; } public string Path { get; } public string Signature { get; } private event ProcessTerminatedEventHandler TerminatedEvent; public event ProcessTerminatedEventHandler Terminated { add { TerminatedEvent += value; InitializeEvent(); } remove { TerminatedEvent -= value; } } internal Process(System.Diagnostics.Process process, string name, string originalName, ILogger logger, string path, string signature) { this.logger = logger; this.process = process; this.Name = name; this.OriginalName = originalName; this.Path = path; this.Signature = signature?.ToLower(); } public string GetAdditionalInfo() { var info = new StringBuilder(); info.Append($"Original Name: {(string.IsNullOrWhiteSpace(OriginalName) ? "n/a" : $"'{OriginalName}'")}, "); info.Append($"Path: {(string.IsNullOrWhiteSpace(Path) ? "n/a" : $"'{Path}'")}, "); info.Append($"Signature: {(string.IsNullOrWhiteSpace(Signature) ? "n/a" : Signature)}"); return info.ToString(); } public bool TryClose(int timeout_ms = 0) { try { logger.Debug("Attempting to close process..."); process.Refresh(); var success = process.CloseMainWindow(); if (success) { logger.Debug("Successfully sent close message to main window."); } else { logger.Warn("Failed to send close message to main window!"); } return success && WaitForTermination(timeout_ms); } catch (Exception e) { logger.Error("Failed to close main window!", e); } return false; } public bool TryKill(int timeout_ms = 0) { try { logger.Debug("Attempting to kill process..."); process.Refresh(); process.Kill(); return WaitForTermination(timeout_ms); } catch (Exception e) { logger.Error("Failed to kill process!", e); } return false; } public override string ToString() { return $"'{Name}' ({Id})"; } private bool IsTerminated() { 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; logger.Debug("Initialized termination event."); } } private bool WaitForTermination(int timeout_ms) { var terminated = process.WaitForExit(timeout_ms); if (terminated) { logger.Debug($"Process has terminated within {timeout_ms}ms."); } else { logger.Warn($"Process failed to terminate within {timeout_ms}ms!"); } return terminated; } private void Process_Exited(object sender, EventArgs e) { TerminatedEvent?.Invoke(process.ExitCode); logger.Debug("Process has terminated."); } } }