/* * Copyright (c) 2019 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 = LoadProcessNames(); foreach (var process in running) { var name = names.FirstOrDefault(n => n.processId == process.Id).name; var originalName = names.FirstOrDefault(n => n.processId == process.Id).originalName; processes.Add(new Process(process, name, originalName, LoggerFor(process))); } 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 = System.Diagnostics.Process.Start(path, string.Join(" ", args)); } var process = new Process(raw, LoggerFor(raw)); logger.Info($"Successfully started process '{path}' with ID = {process.Id}."); 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); } public bool TryGetById(int id, out IProcess process) { var raw = System.Diagnostics.Process.GetProcesses().FirstOrDefault(p => p.Id == id); process = default(IProcess); if (raw != default(System.Diagnostics.Process)) { process = new Process(raw, LoggerFor(raw)); } return process != default(IProcess); } private IEnumerable<(int processId, string name, string originalName)> LoadProcessNames() { 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 ILogger LoggerFor(System.Diagnostics.Process process) { return logger.CloneFor($"{nameof(Process)} '{process.ProcessName}' ({process.Id})"); } } }