mirror of
https://github.com/2dust/v2rayN.git
synced 2025-04-19 21:52:25 +00:00
424 lines
No EOL
15 KiB
C#
424 lines
No EOL
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 = _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<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
|
|
}
|
|
} |