Program.cs 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. using Newtonsoft.Json;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Diagnostics;
  5. using System.IO;
  6. using System.Linq;
  7. using System.Net;
  8. using System.Net.NetworkInformation;
  9. using System.Runtime.InteropServices;
  10. using System.Threading;
  11. using Vinno.IUS.Common.Log;
  12. namespace ProcessDaemonTool
  13. {
  14. class Program
  15. {
  16. private static List<ProtectProgress> _needProtectProcessList = new List<ProtectProgress>();
  17. private static SpinThread _spinThread;
  18. static void Main(string[] args)
  19. {
  20. WriteInfo("Start....ProcessDaemonTool ");
  21. var successed= DisableShowErrorUIHelper.SetDontShowUI();
  22. WriteInfo($"{successed}, set disable show error UI for regedit");
  23. Process curProcess = System.Diagnostics.Process.GetCurrentProcess();
  24. //only one cmd on the same time
  25. if (Process.GetProcesses().Any(x => x.ProcessName == curProcess.ProcessName && x.Id != curProcess.Id))
  26. {
  27. return;
  28. }
  29. AppManager.Instance.Initialize();
  30. if (!initNeedProtectProcess())
  31. {
  32. return;
  33. }
  34. _spinThread = new SpinThread("StartProtect", TimeSpan.FromSeconds(AppManager.Instance.Config.Interval), StartProtect);
  35. _spinThread.Start();
  36. WriteInfo("Successed ProcessDaemonTool ");
  37. Console.ReadLine();
  38. }
  39. private static bool initNeedProtectProcess()
  40. {
  41. var appSettings = AppManager.Instance.Config.Process;
  42. if (appSettings.Count == 0)
  43. {
  44. WriteTitle("No Found Check Process For Setting");
  45. return false;
  46. }
  47. foreach (var setting in appSettings)
  48. {
  49. string name = setting.Key;
  50. var protectProgress = new ProtectProgress()
  51. {
  52. Description = name,
  53. Command = setting.Value,
  54. ProtectCount = 0,
  55. StartupMode= setting.StartupMode,
  56. IsCheckProcessName= setting.IsCheckProcessName,
  57. CheckPort= setting.CheckPort,
  58. EnableCheckProcess= setting.EnableCheckProcess
  59. };
  60. var existProcessArray = Process.GetProcessesByName(setting.ProcessName);
  61. if(existProcessArray != null&& existProcessArray.Length>0)
  62. {
  63. protectProgress.Process = existProcessArray.FirstOrDefault();
  64. }
  65. _needProtectProcessList.Add(protectProgress);
  66. }
  67. var descriptions = _needProtectProcessList.Where(v => v.Process != null)?.Select(v => "Attach To Process:" + v.Description + " "+v.Command).ToList();
  68. if(descriptions.Count>0)
  69. {
  70. WriteInfo(descriptions);
  71. }
  72. else
  73. {
  74. WriteSucceed(_needProtectProcessList.Select(x => $" Load setting Process: {x.Description} {x.Command}\n").ToList());
  75. }
  76. return true;
  77. }
  78. private static void StartProtect(object o)
  79. {
  80. try
  81. {
  82. //一些没有初始化的可以初始化了
  83. Process[] processIdAry = Process.GetProcesses();
  84. foreach (ProtectProgress protectProgress in _needProtectProcessList)
  85. {
  86. if (!string.IsNullOrEmpty(protectProgress.CheckPort))
  87. {
  88. var isUsed = PortInUse(int.Parse(protectProgress.CheckPort));
  89. if (isUsed)
  90. {
  91. continue;
  92. }
  93. }
  94. if (protectProgress.Process != null&&!protectProgress.EnableCheckProcess)
  95. {
  96. continue;
  97. }
  98. //说明已经启动了
  99. if (protectProgress.Process != null && processIdAry.Any(x => x.Id == protectProgress.Process.Id))
  100. {
  101. var exist = processIdAry.FirstOrDefault(v => v.ProcessName.Contains(protectProgress.Description));
  102. continue;
  103. }
  104. if (protectProgress.IsCheckProcessName)
  105. {
  106. var existProcessArray = Process.GetProcessesByName(protectProgress.Description);
  107. if (existProcessArray != null && existProcessArray.Length > 0)
  108. {
  109. protectProgress.Process = existProcessArray.FirstOrDefault();
  110. continue;
  111. }
  112. }
  113. try
  114. {
  115. string log1 = $"Check Process Exited: {protectProgress.Description}";
  116. Logger.WriteLineInfo(log1);
  117. protectProgress.Process = startCmd(protectProgress.Command, protectProgress.StartupMode== StartupMode.cmd);
  118. if (protectProgress.Process != null)
  119. {
  120. string log = $"Start Process successed: {protectProgress.Description}, Id:{protectProgress.Process.Id}, Name:{protectProgress.Process.ProcessName} ,RetryNum:{protectProgress.ProtectCount} ";
  121. Logger.WriteLineInfo(log);
  122. WriteSucceed(log);
  123. protectProgress.ProtectCount++;
  124. protectProgress.Name = protectProgress.Process.ProcessName;
  125. }
  126. }
  127. catch (Exception ex)
  128. {
  129. string log = $"Start Process Failed {protectProgress.Description},Command:{protectProgress.Command} ,ex:{ex}";
  130. Logger.WriteLineError(log);
  131. WriteError(log);
  132. }
  133. Thread.Sleep(3000);
  134. }
  135. }
  136. catch (Exception exception)
  137. {
  138. Console.WriteLine(exception);
  139. throw;
  140. }
  141. finally
  142. {
  143. GC.Collect();
  144. }
  145. }
  146. private static Process startCmd(string cmd,bool useShellExecute=true)
  147. {
  148. cmd = cmd.Replace("|", "&&");
  149. var processStartInfo = new ProcessStartInfo("cmd.exe", "/c " + cmd);
  150. processStartInfo.Verb = "runas";
  151. processStartInfo.CreateNoWindow = false;
  152. processStartInfo.UseShellExecute = useShellExecute;
  153. // processStartInfo.RedirectStandardOutput = true;
  154. Process p = Process.Start(processStartInfo);
  155. return p;
  156. }
  157. private static bool PortInUse(int port)
  158. {
  159. bool inUse = false;
  160. IPGlobalProperties ipProperties = IPGlobalProperties.GetIPGlobalProperties();
  161. IPEndPoint[] ipEndPoints = ipProperties.GetActiveTcpListeners();
  162. foreach (IPEndPoint endPoint in ipEndPoints)
  163. {
  164. if (endPoint.Port == port)
  165. {
  166. inUse = true;
  167. break;
  168. }
  169. }
  170. return inUse;
  171. }
  172. [DllImport("user32.dll", CharSet = CharSet.Auto)]
  173. public static extern IntPtr SendMessageTimeout(
  174. HandleRef hWnd,
  175. int msg,
  176. IntPtr wParam,
  177. IntPtr lParam,
  178. int flags,
  179. int timeout,
  180. out IntPtr pdwResult);
  181. const int SMTO_ABORTIFHUNG = 2;
  182. public static bool RespondingWithinMs(Process process, int timeoutMs)
  183. {
  184. IntPtr ptr2;
  185. return SendMessageTimeout(
  186. new HandleRef(process, process.MainWindowHandle),
  187. 0,
  188. IntPtr.Zero,
  189. IntPtr.Zero,
  190. SMTO_ABORTIFHUNG,
  191. timeoutMs,
  192. out ptr2) != IntPtr.Zero;
  193. }
  194. #region write
  195. private static void WriteTitle(string text)
  196. {
  197. WriteTitle(new List<string>() { text });
  198. }
  199. private static void WriteSucceed(List<string> textList)
  200. {
  201. textList.ForEach(Console.WriteLine);
  202. }
  203. private static void WriteSucceed(string text)
  204. {
  205. text = $"[{DateTime.Now:yyyyMMddHHmmss},{DateTime.Now.Millisecond:d3}]-" + "[Thread:" + Thread.CurrentThread.ManagedThreadId.ToString() + "]" + text;
  206. Console.WriteLine(text);
  207. }
  208. private static void WriteInfo(string text)
  209. {
  210. text = $"[{DateTime.Now:yyyyMMddHHmmss},{DateTime.Now.Millisecond:d3}]-" + "[Thread:" + Thread.CurrentThread.ManagedThreadId.ToString() + "]" + text;
  211. Console.WriteLine(text);
  212. }
  213. private static void WriteInfo(List<string> textList)
  214. {
  215. textList.ForEach(Console.WriteLine);
  216. }
  217. private static void WriteWarn(string text)
  218. {
  219. text = $"[{DateTime.Now:yyyyMMddHHmmss},{DateTime.Now.Millisecond:d3}]-" + "[Thread:" + Thread.CurrentThread.ManagedThreadId.ToString() + "]" + text;
  220. Console.WriteLine(text);
  221. }
  222. private static void WriteWarn(List<string> textList)
  223. {
  224. textList.ForEach(Console.WriteLine);
  225. }
  226. private static void WriteError(string text)
  227. {
  228. text = $"[{DateTime.Now:yyyyMMddHHmmss},{DateTime.Now.Millisecond:d3}]-" + "[Thread:" + Thread.CurrentThread.ManagedThreadId.ToString() + "]" + text;
  229. Console.WriteLine(text);
  230. }
  231. private static void WriteError(List<string> textList)
  232. {
  233. textList.ForEach(Console.WriteLine);
  234. }
  235. private static void WriteTitle(List<string> textList)
  236. {
  237. WriteSplitLine();
  238. textList.ForEach(Console.WriteLine);
  239. WriteSplitLine();
  240. }
  241. private static void WriteSplitLine()
  242. {
  243. Console.WriteLine("=======================================");
  244. }
  245. #endregion
  246. }
  247. }