using Netch.Forms; using System; using System.Diagnostics; using System.IO; using System.ServiceProcess; using System.Threading; using System.Threading.Tasks; namespace Netch.Controllers { public class NFController { /// /// 流量变动事件 /// public event BandwidthUpdateHandler OnBandwidthUpdated; /// /// 流量变动处理器 /// /// 上传 /// 下载 public delegate void BandwidthUpdateHandler(long upload, long download); /// /// 进程实例 /// public Process Instance; /// /// 当前状态 /// public Models.State State = Models.State.Waiting; /// /// 启动 /// /// 服务器 /// 模式 /// 是否成功 public bool Start(Models.Server server, Models.Mode mode) { MainForm.Instance.StatusText($"{Utils.i18N.Translate("Status")}{Utils.i18N.Translate(": ")}{Utils.i18N.Translate("Starting Redirector")}"); if (!File.Exists("bin\\Redirector.exe")) { return false; } // 生成驱动文件路径 var driver = string.Format("{0}\\drivers\\netfilter2.sys", Environment.SystemDirectory); // 检查驱动是否存在 if (!File.Exists(driver)) { // 生成系统版本 var version = $"{Environment.OSVersion.Version.Major.ToString()}.{Environment.OSVersion.Version.Minor.ToString()}"; // 检查系统版本并复制对应驱动 try { switch (version) { case "10.0": File.Copy("bin\\Win-10.sys", driver); Utils.Logging.Info("已复制 Win10 驱动"); break; case "6.3": case "6.2": File.Copy("bin\\Win-8.sys", driver); Utils.Logging.Info("已复制 Win8 驱动"); break; case "6.1": case "6.0": File.Copy("bin\\Win-7.sys", driver); Utils.Logging.Info("已复制 Win7 驱动"); break; default: Utils.Logging.Info($"不支持的系统版本:{version}"); return false; } } catch (Exception e) { Utils.Logging.Info("复制驱动文件失败"); Utils.Logging.Info(e.ToString()); return false; } // 注册驱动文件 var result = nfapinet.NFAPI.nf_registerDriver("netfilter2"); if (result != nfapinet.NF_STATUS.NF_STATUS_SUCCESS) { Utils.Logging.Info($"注册驱动失败,返回值:{result}"); return false; } } try { var service = new ServiceController("netfilter2"); if (service.Status == ServiceControllerStatus.Stopped) { MainForm.Instance.StatusText($"{Utils.i18N.Translate("Status")}{Utils.i18N.Translate(": ")}{Utils.i18N.Translate("Starting netfilter2 Service")}"); service.Start(); } } catch (Exception e) { Utils.Logging.Info(e.ToString()); var result = nfapinet.NFAPI.nf_registerDriver("netfilter2"); if (result != nfapinet.NF_STATUS.NF_STATUS_SUCCESS) { Utils.Logging.Info($"注册驱动失败,返回值:{result}"); return false; } } var processes = "NTT.exe,"; foreach (var proc in mode.Rule) { processes += proc; processes += ","; } processes = processes.Substring(0, processes.Length - 1); Instance = MainController.GetProcess(); Instance.StartInfo.FileName = "bin\\Redirector.exe"; var fallback = ""; if (server.Type != "Socks5") { fallback = $"-r 127.0.0.1:{Global.Settings.Socks5LocalPort} -p \"{processes}\""; } else { var result = Utils.DNS.Lookup(server.Hostname); if (result == null) { Utils.Logging.Info("无法解析服务器 IP 地址"); return false; } fallback = $"-r {result}:{server.Port} -p \"{processes}\""; if (!string.IsNullOrWhiteSpace(server.Username) && !string.IsNullOrWhiteSpace(server.Password)) { fallback += $" -username \"{server.Username}\" -password \"{server.Password}\""; } } Instance.StartInfo.Arguments = fallback; Instance.OutputDataReceived += OnOutputDataReceived; Instance.ErrorDataReceived += OnOutputDataReceived; State = Models.State.Starting; Instance.Start(); Instance.BeginOutputReadLine(); Instance.BeginErrorReadLine(); for (var i = 0; i < 1000; i++) { Thread.Sleep(10); if (State == Models.State.Started) { return true; } } Utils.Logging.Info("NF 进程启动超时"); Stop(); return false; } /// /// 停止 /// public void Stop() { try { if (Instance != null && !Instance.HasExited) { Instance.Kill(); Instance.WaitForExit(); } } catch (Exception e) { Utils.Logging.Info(e.ToString()); } } public void OnOutputDataReceived(object sender, DataReceivedEventArgs e) { if (!string.IsNullOrWhiteSpace(e.Data)) { File.AppendAllText("logging\\redirector.log", string.Format("{0}\r\n", e.Data)); if (State == Models.State.Starting) { if (Instance.HasExited) { State = Models.State.Stopped; } else if (e.Data.Contains("Redirect to")) { State = Models.State.Started; } else if (e.Data.Contains("Failed") || e.Data.Contains("Unable")) { State = Models.State.Stopped; } } else if (State == Models.State.Started) { if (e.Data.StartsWith("[Application][Bandwidth]")) { var splited = e.Data.Replace("[Application][Bandwidth]", "").Trim().Split(','); if (splited.Length == 2) { var uploadSplited = splited[0].Split(':'); var downloadSplited = splited[1].Split(':'); if (uploadSplited.Length == 2 && downloadSplited.Length == 2) { if (long.TryParse(uploadSplited[1], out var upload) && long.TryParse(downloadSplited[1], out var download)) { Task.Run(() => OnBandwidthUpdated(upload, download)); } } } } } } } } }