using System; using System.Collections.Generic; using System.Diagnostics; using System.Linq; using Microsoft.Extensions.Logging; using Ombi.Updater; namespace Ombi.Helpers { public class ProcessProvider : IProcessProvider { public ProcessProvider(ILogger log) { _log = log; } private readonly ILogger _log; public int GetCurrentProcessId() { return Process.GetCurrentProcess().Id; } public ProcessInfo GetCurrentProcess() { return ConvertToProcessInfo(Process.GetCurrentProcess()); } public bool Exists(int processId) { return GetProcessById(processId) != null; } public bool Exists(string processName) { return GetProcessesByName(processName).Any(); } public ProcessInfo GetProcessById(int id) { _log.LogInformation("Finding process with Id:{0}", id); var processInfo = ConvertToProcessInfo(Process.GetProcesses().FirstOrDefault(p => p.Id == id)); if (processInfo == null) { _log.LogInformation("Unable to find process with ID {0}", id); } else { _log.LogInformation("Found process {0}", processInfo.ToString()); } return processInfo; } public List FindProcessByName(string name) { return GetProcessesByName(name).Select(ConvertToProcessInfo).Where(c => c != null).ToList(); } public void WaitForExit(Process process) { _log.LogInformation("Waiting for process {0} to exit.", process.ProcessName); process.WaitForExit(); } public void SetPriority(int processId, ProcessPriorityClass priority) { var process = Process.GetProcessById(processId); _log.LogInformation("Updating [{0}] process priority from {1} to {2}", process.ProcessName, process.PriorityClass, priority); process.PriorityClass = priority; } public void Kill(int processId) { var process = Process.GetProcesses().FirstOrDefault(p => p.Id == processId); if (process == null) { _log.LogInformation("Cannot find process with id: {0}", processId); return; } process.Refresh(); if (process.Id != Process.GetCurrentProcess().Id && process.HasExited) { _log.LogInformation("Process has already exited"); return; } _log.LogInformation("[{0}]: Killing process", process.Id); process.Kill(); _log.LogInformation("[{0}]: Waiting for exit", process.Id); process.WaitForExit(); _log.LogInformation("[{0}]: Process terminated successfully", process.Id); } public void KillAll(string processName) { var processes = GetProcessesByName(processName); _log.LogInformation("Found {0} processes to kill", processes.Count); foreach (var processInfo in processes) { if (processInfo.Id == Process.GetCurrentProcess().Id) { _log.LogInformation("Tried killing own process, skipping: {0} [{1}]", processInfo.Id, processInfo.ProcessName); continue; } _log.LogInformation("Killing process: {0} [{1}]", processInfo.Id, processInfo.ProcessName); Kill(processInfo.Id); } } private ProcessInfo ConvertToProcessInfo(Process process) { if (process == null) return null; process.Refresh(); ProcessInfo processInfo = null; try { if (process.Id <= 0) return null; processInfo = new ProcessInfo { Id = process.Id, Name = process.ProcessName, StartPath = GetExeFileName(process) }; if (process.Id != Process.GetCurrentProcess().Id && process.HasExited) { processInfo = null; } } catch (Exception e) { _log.LogInformation(e.Message); } return processInfo; } public Process Start(string path, string args = null) { var startInfo = new ProcessStartInfo(path, args) { CreateNoWindow = true, UseShellExecute = false, RedirectStandardError = true, RedirectStandardOutput = true, RedirectStandardInput = true }; _log.LogDebug("Starting {0} {1}", path, args); var process = new Process { StartInfo = startInfo }; process.OutputDataReceived += (sender, eventArgs) => { if (string.IsNullOrWhiteSpace(eventArgs.Data)) return; _log.LogDebug(eventArgs.Data); }; process.ErrorDataReceived += (sender, eventArgs) => { if (string.IsNullOrWhiteSpace(eventArgs.Data)) return; _log.LogDebug(eventArgs.Data); }; process.Start(); process.BeginErrorReadLine(); process.BeginOutputReadLine(); return process; } private static string GetExeFileName(Process process) { return process.MainModule.FileName; } private List GetProcessesByName(string name) { var processes = Process.GetProcessesByName(name).ToList(); _log.LogInformation("Found {0} processes with the name: {1}", processes.Count, name); try { foreach (var process in processes) { _log.LogInformation(" - [{0}] {1}", process.Id, process.ProcessName); } } catch { // Don't crash on gettings some log data. } return processes; } } public class ProcessInfo { public int Id { get; set; } public string Name { get; set; } public string StartPath { get; set; } public override string ToString() { return string.Format("{0}:{1} [{2}]", Id, Name ?? "Unknown", StartPath ?? "Unknown"); } } }