mirror of
https://github.com/2dust/v2rayN.git
synced 2025-04-19 21:52:25 +00:00
399 lines
No EOL
14 KiB
C#
399 lines
No EOL
14 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 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())
|
|
{
|
|
var coreInfo = CoreInfoHandler.Instance.GetCoreInfo();
|
|
foreach (var it in coreInfo)
|
|
{
|
|
if (it.CoreType == ECoreType.v2rayN)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
foreach (var vName in it.CoreExes)
|
|
{
|
|
var exe = Utils.GetExeName(Utils.GetBinPath(vName, 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);
|
|
ShowMsg(false, result.Msg);
|
|
if (result.Success != true)
|
|
{
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
ShowMsg(true, $"{node.GetSummary()}");
|
|
await CoreStop();
|
|
await CoreStart(node);
|
|
|
|
//In tun mode, do a delay check and restart the core
|
|
//if (_config.tunModeItem.enableTun)
|
|
//{
|
|
// Observable.Range(1, 1)
|
|
// .Delay(TimeSpan.FromSeconds(15))
|
|
// .Subscribe(x =>
|
|
// {
|
|
// {
|
|
// if (_process == null || _process.HasExited)
|
|
// {
|
|
// CoreStart(node);
|
|
// ShowMsg(false, "Tun mode restart the core once");
|
|
// Logging.SaveLog("Tun mode restart the core once");
|
|
// }
|
|
// }
|
|
// });
|
|
//}
|
|
}
|
|
}
|
|
|
|
public async Task<int> LoadCoreConfigSpeedtest(List<ServerTestItem> selecteds)
|
|
{
|
|
var pid = -1;
|
|
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)
|
|
{
|
|
pid = await CoreStartSpeedtest(configPath, coreType);
|
|
}
|
|
return pid;
|
|
}
|
|
|
|
public async Task CoreStop()
|
|
{
|
|
try
|
|
{
|
|
bool hasProc = false;
|
|
if (_process != null)
|
|
{
|
|
await KillProcess(_process);
|
|
_process.Dispose();
|
|
_process = null;
|
|
hasProc = true;
|
|
}
|
|
|
|
if (_processPre != null)
|
|
{
|
|
await KillProcess(_processPre);
|
|
_processPre.Dispose();
|
|
_processPre = null;
|
|
hasProc = true;
|
|
}
|
|
|
|
if (!hasProc)
|
|
{
|
|
var coreInfo = CoreInfoHandler.Instance.GetCoreInfo();
|
|
foreach (var it in coreInfo)
|
|
{
|
|
if (it.CoreType == ECoreType.v2rayN)
|
|
{
|
|
continue;
|
|
}
|
|
foreach (string vName in it.CoreExes)
|
|
{
|
|
var existing = Process.GetProcessesByName(vName);
|
|
foreach (Process p in existing)
|
|
{
|
|
string? path = p.MainModule?.FileName;
|
|
if (path == Utils.GetExeName(Utils.GetBinPath(vName, it.CoreType.ToString())))
|
|
{
|
|
await KillProcess(p);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
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)
|
|
{
|
|
string fileName = string.Empty;
|
|
foreach (string name in coreInfo.CoreExes)
|
|
{
|
|
string vName = Utils.GetExeName(name);
|
|
vName = Utils.GetBinPath(vName, coreInfo.CoreType.ToString());
|
|
if (File.Exists(vName))
|
|
{
|
|
fileName = vName;
|
|
break;
|
|
}
|
|
}
|
|
if (Utils.IsNullOrEmpty(fileName))
|
|
{
|
|
string 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)
|
|
{
|
|
ShowMsg(false, $"{Environment.OSVersion} - {(Environment.Is64BitOperatingSystem ? 64 : 32)}");
|
|
ShowMsg(false, string.Format(ResUI.StartService, DateTime.Now.ToString("yyyy/MM/dd HH:mm:ss")));
|
|
|
|
//ECoreType coreType;
|
|
//if (node.configType != EConfigType.Custom && _config.tunModeItem.enableTun)
|
|
//{
|
|
// coreType = ECoreType.sing_box;
|
|
//}
|
|
//else
|
|
//{
|
|
// coreType = LazyConfig.Instance.GetCoreType(node, node.configType);
|
|
//}
|
|
var coreType = AppHandler.Instance.GetCoreType(node, node.ConfigType);
|
|
_config.RunningCoreType = coreType;
|
|
var coreInfo = CoreInfoHandler.Instance.GetCoreInfo(coreType);
|
|
|
|
var displayLog = node.ConfigType != EConfigType.Custom || node.DisplayLog;
|
|
var proc = await RunProcess(node, coreInfo, "", displayLog);
|
|
if (proc is null)
|
|
{
|
|
return;
|
|
}
|
|
_process = proc;
|
|
|
|
//start a pre service
|
|
if (_process != null && !_process.HasExited)
|
|
{
|
|
ProfileItem? itemSocks = null;
|
|
var preCoreType = ECoreType.sing_box;
|
|
if (node.ConfigType != EConfigType.Custom && coreType != ECoreType.sing_box && _config.TunModeItem.EnableTun)
|
|
{
|
|
itemSocks = new ProfileItem()
|
|
{
|
|
CoreType = preCoreType,
|
|
ConfigType = EConfigType.SOCKS,
|
|
Address = Global.Loopback,
|
|
Sni = node.Address, //Tun2SocksAddress
|
|
Port = AppHandler.Instance.GetLocalPort(EInboundProtocol.socks)
|
|
};
|
|
}
|
|
else if ((node.ConfigType == EConfigType.Custom && node.PreSocksPort > 0))
|
|
{
|
|
preCoreType = _config.TunModeItem.EnableTun ? ECoreType.sing_box : ECoreType.Xray;
|
|
itemSocks = new ProfileItem()
|
|
{
|
|
CoreType = preCoreType,
|
|
ConfigType = EConfigType.SOCKS,
|
|
Address = Global.Loopback,
|
|
Port = node.PreSocksPort.Value,
|
|
};
|
|
_config.RunningCoreType = preCoreType;
|
|
}
|
|
if (itemSocks != null)
|
|
{
|
|
string fileName2 = Utils.GetConfigPath(Global.CorePreConfigFileName);
|
|
var result = await CoreConfigHandler.GenerateClientConfig(itemSocks, fileName2);
|
|
if (result.Success)
|
|
{
|
|
var coreInfo2 = CoreInfoHandler.Instance.GetCoreInfo(preCoreType);
|
|
var proc2 = await RunProcess(node, coreInfo2, $" -c {Global.CorePreConfigFileName}", true);
|
|
if (proc2 is not null)
|
|
{
|
|
_processPre = proc2;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private async Task<int> CoreStartSpeedtest(string configPath, ECoreType coreType)
|
|
{
|
|
ShowMsg(false, string.Format(ResUI.StartService, DateTime.Now.ToString("yyyy/MM/dd HH:mm:ss")));
|
|
|
|
ShowMsg(false, configPath);
|
|
try
|
|
{
|
|
var coreInfo = CoreInfoHandler.Instance.GetCoreInfo(coreType);
|
|
var proc = await RunProcess(new(), coreInfo, $" -c {Global.CoreSpeedtestConfigFileName}", true);
|
|
if (proc is null)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return proc.Id;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Logging.SaveLog(ex.Message, ex);
|
|
string msg = ex.Message;
|
|
ShowMsg(false, msg);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
private void ShowMsg(bool notify, string msg)
|
|
{
|
|
_updateFunc?.Invoke(notify, msg);
|
|
}
|
|
|
|
#endregion Private
|
|
|
|
#region Process
|
|
|
|
private async Task<Process?> RunProcess(ProfileItem node, CoreInfo coreInfo, string configPath, bool displayLog)
|
|
{
|
|
try
|
|
{
|
|
string fileName = CoreFindExe(coreInfo);
|
|
if (Utils.IsNullOrEmpty(fileName))
|
|
{
|
|
return null;
|
|
}
|
|
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 startUpErrorMessage = new StringBuilder();
|
|
var startUpSuccessful = false;
|
|
if (displayLog)
|
|
{
|
|
proc.OutputDataReceived += (sender, e) =>
|
|
{
|
|
if (Utils.IsNotEmpty(e.Data))
|
|
{
|
|
string msg = e.Data + Environment.NewLine;
|
|
ShowMsg(false, msg);
|
|
}
|
|
};
|
|
proc.ErrorDataReceived += (sender, e) =>
|
|
{
|
|
if (Utils.IsNotEmpty(e.Data))
|
|
{
|
|
string msg = e.Data + Environment.NewLine;
|
|
ShowMsg(false, msg);
|
|
|
|
if (!startUpSuccessful)
|
|
{
|
|
startUpErrorMessage.Append(msg);
|
|
}
|
|
}
|
|
};
|
|
}
|
|
proc.Start();
|
|
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);
|
|
string msg = ex.Message;
|
|
ShowMsg(true, msg);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
private async Task KillProcess(Process? proc)
|
|
{
|
|
if (proc is null)
|
|
{
|
|
return;
|
|
}
|
|
try
|
|
{
|
|
proc.Kill();
|
|
proc.WaitForExit(100);
|
|
if (!proc.HasExited)
|
|
{
|
|
proc.Kill();
|
|
proc.WaitForExit(100);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Logging.SaveLog(ex.Message, ex);
|
|
}
|
|
}
|
|
|
|
#endregion Process
|
|
}
|
|
} |