using System.Diagnostics; using System.Text; namespace ServiceLib.Handler { /// /// Core process processing class /// public class CoreHandler { private static readonly Lazy _instance = new(() => new()); public static CoreHandler Instance => _instance.Value; private Config _config; private Process? _process; private Process? _processPre; private int _linuxSudoPid = -1; private Action? _updateFunc; public async Task Init(Config config, Action 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 LoadCoreConfigSpeedtest(List 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) { await KillProcess(_process); _process.Dispose(); _process = null; } if (_processPre != null) { await KillProcess(_processPre); _processPre.Dispose(); _processPre = null; } if (_linuxSudoPid > 0) { await KillProcessAsLinuxSudo(); } _linuxSudoPid = -1; } catch (Exception ex) { Logging.SaveLog(ex.Message, ex); } } public async Task CoreStopPid(int pid) { try { var _p = Process.GetProcessById(pid); await KillProcess(_p); } 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 = _config.RunningCoreType = 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 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 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 KillProcess(Process? proc) { if (proc is null) { return; } try { proc?.Kill(); } catch (Exception) { // ignored } await Task.Delay(100); if (proc?.HasExited == false) { try { proc?.Kill(); } catch (Exception) { // ignored } } } #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 -9 {_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()) { 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); } var timeout = new CancellationTokenSource(TimeSpan.FromSeconds(10)); await proc.WaitForExitAsync(timeout.Token); await Task.Delay(1000); } private async Task CreateLinuxShellFile(string cmdLine, string fileName) { //Shell scripts var shFilePath = Utils.GetBinPath(fileName); File.Delete(shFilePath); var sb = new StringBuilder(); sb.AppendLine("#!/bin/sh"); 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 } }