/* * Copyright (c) 2022 ETH Zürich, Educational Development and Technology (LET) * * 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.ComponentModel; using System.Diagnostics; using System.IO; using System.Linq; using System.Management; using System.Runtime.InteropServices; using SafeExamBrowser.Logging.Contracts; using SafeExamBrowser.WindowsApi.Constants; using SafeExamBrowser.WindowsApi.Contracts; using SafeExamBrowser.WindowsApi.Types; namespace SafeExamBrowser.WindowsApi { public class ProcessFactory : IProcessFactory { private IModuleLogger logger; public IDesktop StartupDesktop { private get; set; } public ProcessFactory(IModuleLogger logger) { this.logger = logger; } public IList GetAllRunning() { var processes = new List(); var running = System.Diagnostics.Process.GetProcesses(); var names = LoadAllProcessNames(); foreach (var process in running) { if (names.Any(n => n.processId == process.Id)) { var (_, name, originalName) = names.First(n => n.processId == process.Id); processes.Add(new Process(process, name, originalName, LoggerFor(process, name))); } } return processes; } public IProcess StartNew(string path, params string[] args) { var raw = default(System.Diagnostics.Process); logger.Info($"Attempting to start process '{path}'..."); if (StartupDesktop != default(IDesktop)) { raw = StartOnDesktop(path, args); } else { raw = StartNormal(path, args); } var (name, originalName) = LoadProcessNamesFor(raw); var process = new Process(raw, name, originalName, LoggerFor(raw, name)); logger.Info($"Successfully started process '{path}' with ID = {process.Id}."); return process; } public bool TryGetById(int id, out IProcess process) { process = default(IProcess); try { var raw = System.Diagnostics.Process.GetProcessById(id); var (name, originalName) = LoadProcessNamesFor(raw); process = new Process(raw, name, originalName, LoggerFor(raw, name)); } catch (Exception e) { logger.Error($"Failed to get process with ID = {id}!", e); } return process != default(IProcess); } private IEnumerable<(int processId, string name, string originalName)> LoadAllProcessNames() { var names = new List<(int, string, string)>(); try { using (var searcher = new ManagementObjectSearcher($"SELECT ProcessId, Name, ExecutablePath FROM Win32_Process")) using (var results = searcher.Get()) { var processData = results.Cast().ToList(); foreach (var process in processData) { using (process) { var name = Convert.ToString(process["Name"]); var processId = Convert.ToInt32(process["ProcessId"]); var executablePath = Convert.ToString(process["ExecutablePath"]); if (File.Exists(executablePath)) { names.Add((processId, name, FileVersionInfo.GetVersionInfo(executablePath).OriginalFilename)); } else { names.Add((processId, name, default(string))); } } } } } catch (Exception e) { logger.Error("Failed to load process names!", e); } return names; } private (string name, string originalName) LoadProcessNamesFor(System.Diagnostics.Process process) { var name = process.ProcessName; var originalName = default(string); try { using (var searcher = new ManagementObjectSearcher($"SELECT Name, ExecutablePath FROM Win32_Process WHERE ProcessId = {process.Id}")) using (var results = searcher.Get()) using (var processData = results.Cast().First()) { var executablePath = Convert.ToString(processData["ExecutablePath"]); name = Convert.ToString(processData["Name"]); if (File.Exists(executablePath)) { originalName = FileVersionInfo.GetVersionInfo(executablePath).OriginalFilename; } } } catch (Exception e) { logger.Error($"Failed to load process names for {process.ProcessName}!", e); } return (name, originalName); } private ILogger LoggerFor(System.Diagnostics.Process process, string name) { return logger.CloneFor($"{nameof(Process)} '{name}' ({process.Id})"); } private System.Diagnostics.Process StartNormal(string path, params string[] args) { var process = new System.Diagnostics.Process(); process.StartInfo.Arguments = string.Join(" ", args); process.StartInfo.FileName = path; process.StartInfo.UseShellExecute = false; process.StartInfo.WorkingDirectory = Environment.GetFolderPath(Environment.SpecialFolder.Windows); process.Start(); return process; } private System.Diagnostics.Process StartOnDesktop(string path, params string[] args) { var commandLine = $"{'"' + path + '"'} {string.Join(" ", args)}"; var processInfo = new PROCESS_INFORMATION(); var startupInfo = new STARTUPINFO(); startupInfo.cb = Marshal.SizeOf(startupInfo); startupInfo.lpDesktop = StartupDesktop?.Name; var success = Kernel32.CreateProcess(null, commandLine, IntPtr.Zero, IntPtr.Zero, true, Constant.NORMAL_PRIORITY_CLASS, IntPtr.Zero, null, ref startupInfo, ref processInfo); if (success) { return System.Diagnostics.Process.GetProcessById(processInfo.dwProcessId); } var errorCode = Marshal.GetLastWin32Error(); logger.Error($"Failed to start process '{path}' on desktop '{StartupDesktop}'! Error code: {errorCode}."); throw new Win32Exception(errorCode); } } }