mirror of
				https://github.com/bolucat/Archive.git
				synced 2025-10-31 11:57:05 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			424 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			424 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System.Diagnostics;
 | |
| using System.Text;
 | |
| 
 | |
| namespace ServiceLib.Handler
 | |
| {
 | |
|     /// <summary>
 | |
|     /// Core process processing class
 | |
|     /// </summary>
 | |
|     public class CoreHandler
 | |
|     {
 | |
|         private static readonly Lazy<CoreHandler> _instance = new(() => new());
 | |
|         public static CoreHandler Instance => _instance.Value;
 | |
|         private Config _config;
 | |
|         private Process? _process;
 | |
|         private Process? _processPre;
 | |
|         private int _linuxSudoPid = -1;
 | |
|         private Action<bool, string>? _updateFunc;
 | |
| 
 | |
|         public async Task Init(Config config, Action<bool, string> updateFunc)
 | |
|         {
 | |
|             _config = config;
 | |
|             _updateFunc = updateFunc;
 | |
| 
 | |
|             Environment.SetEnvironmentVariable("V2RAY_LOCATION_ASSET", Utils.GetBinPath(""), EnvironmentVariableTarget.Process);
 | |
|             Environment.SetEnvironmentVariable("XRAY_LOCATION_ASSET", Utils.GetBinPath(""), EnvironmentVariableTarget.Process);
 | |
| 
 | |
|             if (Utils.IsLinux() || Utils.IsOSX())
 | |
|             {
 | |
|                 var coreInfo = CoreInfoHandler.Instance.GetCoreInfo();
 | |
|                 foreach (var it in coreInfo)
 | |
|                 {
 | |
|                     if (it.CoreType == ECoreType.v2rayN)
 | |
|                     {
 | |
|                         if (Utils.UpgradeAppExists(out var fileName))
 | |
|                         {
 | |
|                             await Utils.SetLinuxChmod(fileName);
 | |
|                         }
 | |
|                         continue;
 | |
|                     }
 | |
| 
 | |
|                     foreach (var name in it.CoreExes)
 | |
|                     {
 | |
|                         var exe = Utils.GetBinPath(Utils.GetExeName(name), it.CoreType.ToString());
 | |
|                         if (File.Exists(exe))
 | |
|                         {
 | |
|                             await Utils.SetLinuxChmod(exe);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public async Task LoadCore(ProfileItem? node)
 | |
|         {
 | |
|             if (node == null)
 | |
|             {
 | |
|                 ShowMsg(false, ResUI.CheckServerSettings);
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             var fileName = Utils.GetConfigPath(Global.CoreConfigFileName);
 | |
|             var result = await CoreConfigHandler.GenerateClientConfig(node, fileName);
 | |
|             if (result.Success != true)
 | |
|             {
 | |
|                 ShowMsg(true, result.Msg);
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             ShowMsg(true, $"{node.GetSummary()}");
 | |
|             ShowMsg(false, $"{Environment.OSVersion} - {(Environment.Is64BitOperatingSystem ? 64 : 32)}");
 | |
|             ShowMsg(false, string.Format(ResUI.StartService, DateTime.Now.ToString("yyyy/MM/dd HH:mm:ss")));
 | |
|             await CoreStop();
 | |
|             await Task.Delay(100);
 | |
|             await CoreStart(node);
 | |
|             await CoreStartPreService(node);
 | |
|         }
 | |
| 
 | |
|         public async Task<int> LoadCoreConfigSpeedtest(List<ServerTestItem> selecteds)
 | |
|         {
 | |
|             var coreType = selecteds.Exists(t => t.ConfigType is EConfigType.Hysteria2 or EConfigType.TUIC or EConfigType.WireGuard) ? ECoreType.sing_box : ECoreType.Xray;
 | |
|             var configPath = Utils.GetConfigPath(Global.CoreSpeedtestConfigFileName);
 | |
|             var result = await CoreConfigHandler.GenerateClientSpeedtestConfig(_config, configPath, selecteds, coreType);
 | |
|             ShowMsg(false, result.Msg);
 | |
|             if (result.Success != true)
 | |
|             {
 | |
|                 return -1;
 | |
|             }
 | |
| 
 | |
|             ShowMsg(false, string.Format(ResUI.StartService, DateTime.Now.ToString("yyyy/MM/dd HH:mm:ss")));
 | |
|             ShowMsg(false, configPath);
 | |
|             return await CoreStartSpeedtest(configPath, coreType);
 | |
|         }
 | |
| 
 | |
|         public async Task CoreStop()
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 if (_process != null)
 | |
|                 {
 | |
|                     _process = await KillProcess(_process);
 | |
|                 }
 | |
| 
 | |
|                 if (_processPre != null)
 | |
|                 {
 | |
|                     _processPre = await KillProcess(_processPre);
 | |
|                 }
 | |
| 
 | |
|                 if (_linuxSudoPid > 0)
 | |
|                 {
 | |
|                     await KillProcessAsLinuxSudo();
 | |
|                 }
 | |
|                 _linuxSudoPid = -1;
 | |
|             }
 | |
|             catch (Exception ex)
 | |
|             {
 | |
|                 Logging.SaveLog(ex.Message, ex);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public async Task CoreStopPid(int pid)
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 await KillProcess(Process.GetProcessById(pid));
 | |
|             }
 | |
|             catch (Exception ex)
 | |
|             {
 | |
|                 Logging.SaveLog(ex.Message, ex);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         #region Private
 | |
| 
 | |
|         private string CoreFindExe(CoreInfo coreInfo)
 | |
|         {
 | |
|             var fileName = string.Empty;
 | |
|             foreach (var name in coreInfo.CoreExes)
 | |
|             {
 | |
|                 var vName = Utils.GetBinPath(Utils.GetExeName(name), coreInfo.CoreType.ToString());
 | |
|                 if (File.Exists(vName))
 | |
|                 {
 | |
|                     fileName = vName;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|             if (Utils.IsNullOrEmpty(fileName))
 | |
|             {
 | |
|                 var msg = string.Format(ResUI.NotFoundCore, Utils.GetBinPath("", coreInfo.CoreType.ToString()), string.Join(", ", coreInfo.CoreExes.ToArray()), coreInfo.Url);
 | |
|                 Logging.SaveLog(msg);
 | |
|                 ShowMsg(false, msg);
 | |
|             }
 | |
|             return fileName;
 | |
|         }
 | |
| 
 | |
|         private async Task CoreStart(ProfileItem node)
 | |
|         {
 | |
|             var coreType = _config.RunningCoreType = AppHandler.Instance.GetCoreType(node, node.ConfigType);
 | |
|             var coreInfo = CoreInfoHandler.Instance.GetCoreInfo(coreType);
 | |
| 
 | |
|             var displayLog = node.ConfigType != EConfigType.Custom || node.DisplayLog;
 | |
|             var proc = await RunProcess(coreInfo, Global.CoreConfigFileName, displayLog, true);
 | |
|             if (proc is null)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
|             _process = proc;
 | |
|         }
 | |
| 
 | |
|         private async Task CoreStartPreService(ProfileItem node)
 | |
|         {
 | |
|             if (_process != null && !_process.HasExited)
 | |
|             {
 | |
|                 var coreType = AppHandler.Instance.GetCoreType(node, node.ConfigType);
 | |
|                 var itemSocks = await ConfigHandler.GetPreSocksItem(_config, node, coreType);
 | |
|                 if (itemSocks != null)
 | |
|                 {
 | |
|                     var preCoreType = itemSocks.CoreType ?? ECoreType.sing_box;
 | |
|                     var fileName = Utils.GetConfigPath(Global.CorePreConfigFileName);
 | |
|                     var result = await CoreConfigHandler.GenerateClientConfig(itemSocks, fileName);
 | |
|                     if (result.Success)
 | |
|                     {
 | |
|                         var coreInfo = CoreInfoHandler.Instance.GetCoreInfo(preCoreType);
 | |
|                         var proc = await RunProcess(coreInfo, Global.CorePreConfigFileName, true, true);
 | |
|                         if (proc is null)
 | |
|                         {
 | |
|                             return;
 | |
|                         }
 | |
|                         _processPre = proc;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private async Task<int> CoreStartSpeedtest(string configPath, ECoreType coreType)
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 var coreInfo = CoreInfoHandler.Instance.GetCoreInfo(coreType);
 | |
|                 var proc = await RunProcess(coreInfo, Global.CoreSpeedtestConfigFileName, true, false);
 | |
|                 if (proc is null)
 | |
|                 {
 | |
|                     return -1;
 | |
|                 }
 | |
| 
 | |
|                 return proc.Id;
 | |
|             }
 | |
|             catch (Exception ex)
 | |
|             {
 | |
|                 Logging.SaveLog(ex.Message, ex);
 | |
|                 ShowMsg(false, ex.Message);
 | |
|                 return -1;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private void ShowMsg(bool notify, string msg)
 | |
|         {
 | |
|             _updateFunc?.Invoke(notify, msg);
 | |
|         }
 | |
| 
 | |
|         private bool IsNeedSudo(ECoreType eCoreType)
 | |
|         {
 | |
|             return _config.TunModeItem.EnableTun
 | |
|                    && eCoreType == ECoreType.sing_box
 | |
|                    && (Utils.IsLinux() || Utils.IsOSX())
 | |
|                 //&& _config.TunModeItem.LinuxSudoPwd.IsNotEmpty()
 | |
|                 ;
 | |
|         }
 | |
| 
 | |
|         #endregion Private
 | |
| 
 | |
|         #region Process
 | |
| 
 | |
|         private async Task<Process?> RunProcess(CoreInfo coreInfo, string configPath, bool displayLog, bool mayNeedSudo)
 | |
|         {
 | |
|             var fileName = CoreFindExe(coreInfo);
 | |
|             if (Utils.IsNullOrEmpty(fileName))
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 Process proc = new()
 | |
|                 {
 | |
|                     StartInfo = new()
 | |
|                     {
 | |
|                         FileName = fileName,
 | |
|                         Arguments = string.Format(coreInfo.Arguments, configPath),
 | |
|                         WorkingDirectory = Utils.GetConfigPath(),
 | |
|                         UseShellExecute = false,
 | |
|                         RedirectStandardOutput = displayLog,
 | |
|                         RedirectStandardError = displayLog,
 | |
|                         CreateNoWindow = true,
 | |
|                         StandardOutputEncoding = displayLog ? Encoding.UTF8 : null,
 | |
|                         StandardErrorEncoding = displayLog ? Encoding.UTF8 : null,
 | |
|                     }
 | |
|                 };
 | |
| 
 | |
|                 var isNeedSudo = mayNeedSudo && IsNeedSudo(coreInfo.CoreType);
 | |
|                 if (isNeedSudo)
 | |
|                 {
 | |
|                     await RunProcessAsLinuxSudo(proc, fileName, coreInfo, configPath);
 | |
|                 }
 | |
| 
 | |
|                 var startUpErrorMessage = new StringBuilder();
 | |
|                 var startUpSuccessful = false;
 | |
|                 if (displayLog)
 | |
|                 {
 | |
|                     proc.OutputDataReceived += (sender, e) =>
 | |
|                     {
 | |
|                         if (Utils.IsNullOrEmpty(e.Data)) return;
 | |
|                         ShowMsg(false, e.Data + Environment.NewLine);
 | |
|                     };
 | |
|                     proc.ErrorDataReceived += (sender, e) =>
 | |
|                     {
 | |
|                         if (Utils.IsNullOrEmpty(e.Data)) return;
 | |
|                         ShowMsg(false, e.Data + Environment.NewLine);
 | |
| 
 | |
|                         if (!startUpSuccessful)
 | |
|                         {
 | |
|                             startUpErrorMessage.Append(e.Data + Environment.NewLine);
 | |
|                         }
 | |
|                     };
 | |
|                 }
 | |
|                 proc.Start();
 | |
| 
 | |
|                 if (isNeedSudo && _config.TunModeItem.LinuxSudoPwd.IsNotEmpty())
 | |
|                 {
 | |
|                     var pwd = DesUtils.Decrypt(_config.TunModeItem.LinuxSudoPwd);
 | |
|                     await Task.Delay(10);
 | |
|                     await proc.StandardInput.WriteLineAsync(pwd);
 | |
|                     await Task.Delay(10);
 | |
|                     await proc.StandardInput.WriteLineAsync(pwd);
 | |
|                 }
 | |
|                 if (isNeedSudo) _linuxSudoPid = proc.Id;
 | |
| 
 | |
|                 if (displayLog)
 | |
|                 {
 | |
|                     proc.BeginOutputReadLine();
 | |
|                     proc.BeginErrorReadLine();
 | |
|                 }
 | |
| 
 | |
|                 if (proc.WaitForExit(1000))
 | |
|                 {
 | |
|                     proc.CancelErrorRead();
 | |
|                     throw new Exception(displayLog ? startUpErrorMessage.ToString() : "启动进程失败并退出 (Failed to start the process and exited)");
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     startUpSuccessful = true;
 | |
|                 }
 | |
| 
 | |
|                 AppHandler.Instance.AddProcess(proc.Handle);
 | |
|                 return proc;
 | |
|             }
 | |
|             catch (Exception ex)
 | |
|             {
 | |
|                 Logging.SaveLog(ex.Message, ex);
 | |
|                 ShowMsg(true, ex.Message);
 | |
|                 return null;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private async Task<Process?> KillProcess(Process? proc)
 | |
|         {
 | |
|             if (proc is null)
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
|             try { proc?.Kill(true); } catch { }
 | |
|             try { proc?.Close(); } catch { }
 | |
|             try { proc?.Dispose(); } catch { }
 | |
| 
 | |
|             await Task.Delay(100);
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         #endregion Process
 | |
| 
 | |
|         #region Linux
 | |
| 
 | |
|         private async Task RunProcessAsLinuxSudo(Process proc, string fileName, CoreInfo coreInfo, string configPath)
 | |
|         {
 | |
|             var cmdLine = $"{fileName.AppendQuotes()} {string.Format(coreInfo.Arguments, Utils.GetConfigPath(configPath).AppendQuotes())}";
 | |
| 
 | |
|             var shFilePath = await CreateLinuxShellFile(cmdLine, "run_as_sudo.sh");
 | |
|             proc.StartInfo.FileName = shFilePath;
 | |
|             proc.StartInfo.Arguments = "";
 | |
|             proc.StartInfo.WorkingDirectory = "";
 | |
|             if (_config.TunModeItem.LinuxSudoPwd.IsNotEmpty())
 | |
|             {
 | |
|                 proc.StartInfo.StandardInputEncoding = Encoding.UTF8;
 | |
|                 proc.StartInfo.RedirectStandardInput = true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private async Task KillProcessAsLinuxSudo()
 | |
|         {
 | |
|             var cmdLine = $"kill {_linuxSudoPid}";
 | |
|             var shFilePath = await CreateLinuxShellFile(cmdLine, "kill_as_sudo.sh");
 | |
|             Process proc = new()
 | |
|             {
 | |
|                 StartInfo = new()
 | |
|                 {
 | |
|                     FileName = shFilePath,
 | |
|                     UseShellExecute = false,
 | |
|                     CreateNoWindow = true,
 | |
|                     StandardInputEncoding = Encoding.UTF8,
 | |
|                     RedirectStandardInput = true
 | |
|                 }
 | |
|             };
 | |
|             proc.Start();
 | |
| 
 | |
|             if (_config.TunModeItem.LinuxSudoPwd.IsNotEmpty())
 | |
|             {
 | |
|                 try
 | |
|                 {
 | |
|                     var pwd = DesUtils.Decrypt(_config.TunModeItem.LinuxSudoPwd);
 | |
|                     await Task.Delay(10);
 | |
|                     await proc.StandardInput.WriteLineAsync(pwd);
 | |
|                     await Task.Delay(10);
 | |
|                     await proc.StandardInput.WriteLineAsync(pwd);
 | |
|                 }
 | |
|                 catch (Exception)
 | |
|                 {
 | |
|                     // ignored
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             var timeout = new CancellationTokenSource(TimeSpan.FromSeconds(10));
 | |
|             await proc.WaitForExitAsync(timeout.Token);
 | |
|             await Task.Delay(3000);
 | |
|         }
 | |
| 
 | |
|         private async Task<string> CreateLinuxShellFile(string cmdLine, string fileName)
 | |
|         {
 | |
|             //Shell scripts
 | |
|             var shFilePath = Utils.GetBinPath(AppHandler.Instance.IsAdministrator ? "root_" + fileName : fileName);
 | |
|             File.Delete(shFilePath);
 | |
|             var sb = new StringBuilder();
 | |
|             sb.AppendLine("#!/bin/sh");
 | |
|             if (AppHandler.Instance.IsAdministrator)
 | |
|             {
 | |
|                 sb.AppendLine($"{cmdLine}");
 | |
|             }
 | |
|             else if (_config.TunModeItem.LinuxSudoPwd.IsNullOrEmpty())
 | |
|             {
 | |
|                 sb.AppendLine($"pkexec {cmdLine}");
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 sb.AppendLine($"sudo -S {cmdLine}");
 | |
|             }
 | |
| 
 | |
|             await File.WriteAllTextAsync(shFilePath, sb.ToString());
 | |
|             await Utils.SetLinuxChmod(shFilePath);
 | |
|             Logging.SaveLog(shFilePath);
 | |
| 
 | |
|             return shFilePath;
 | |
|         }
 | |
| 
 | |
|         #endregion Linux
 | |
|     }
 | |
| } | 
![github-action[bot]](/assets/img/avatar_default.png)