using System; using System.Collections.Generic; using System.ComponentModel; using System.Diagnostics; using System.IO; using System.Runtime.InteropServices; using System.Text; using System.Threading.Tasks; using Netch.Forms; using Netch.Models; using Netch.Utils; namespace Netch.Controllers { public static class MainController { /// /// 记录当前使用的端口 /// /// public static readonly List UsingPorts = new List(); public static EncryptedProxy EncryptedProxyController { get; private set; } public static ModeController ModeController { get; private set; } private static Server _savedServer; private static Mode _savedMode; public static string PortInfo { get { if (_savedMode == null || _savedServer == null) return string.Empty; if (_savedServer.Type == "Socks5" && _savedMode.Type != 3 && _savedMode.Type != 5) // 不可控Socks5, 不可控HTTP return string.Empty; var text = new StringBuilder(); if (_localAddress == "0.0.0.0") text.Append(i18N.Translate("Allow other Devices to connect") + " "); if (_savedServer.Type != "Socks5") // 可控Socks5 text.Append($"Socks5 {i18N.Translate("Local Port", ": ")}{_socks5Port}"); if (_savedMode.Type == 3 || _savedMode.Type == 5) // 有HTTP { if (_savedServer.Type != "Socks5") text.Append(" | "); text.Append($"HTTP {i18N.Translate("Local Port", ": ")}{_httpPort}"); } return $" ({text})"; } } /// /// NTT 控制器 /// public static readonly NTTController NTTController = new NTTController(); private static string _localAddress; private static int _redirectorTCPPort; private static int _httpPort; private static int _socks5Port; [DllImport("dnsapi", EntryPoint = "DnsFlushResolverCache")] public static extern uint FlushDNSResolverCache(); /// /// 启动 /// /// 服务器 /// 模式 /// 是否启动成功 public static async Task Start(Server server, Mode mode) { Logging.Info($"启动主控制器: {server.Type} [{mode.Type}]{mode.Remark}"); #region Record Settings _httpPort = Global.Settings.HTTPLocalPort; _socks5Port = server.Type != "Socks5" ? Global.Settings.Socks5LocalPort : server.Port; _redirectorTCPPort = Global.Settings.RedirectorTCPPort; _localAddress = server.Type != "Socks5" ? Global.Settings.LocalAddress : "127.0.0.1"; _savedServer = server; _savedMode = mode; #endregion FlushDNSResolverCache(); _ = Task.Run(Firewall.AddNetchFwRules); bool result; if (server.Type == "Socks5") { result = mode.Type != 4; } else { EncryptedProxyController = server.Type switch { "SS" => new SSController(), "SSR" => new SSRController(), "VMess" => new VMessController(), "Trojan" => new TrojanController(), _ => EncryptedProxyController }; KillProcessByName(EncryptedProxyController.MainFile); #region 检查端口是否被占用 var portNotAvailable = false; if (_savedServer.Type != "Socks5") { portNotAvailable |= PortCheckAndShowMessageBox(_socks5Port, "Socks5"); } switch (_savedMode.Type) { case 0: portNotAvailable |= PortCheckAndShowMessageBox(_redirectorTCPPort, "Redirector TCP"); break; case 3: case 5: portNotAvailable |= PortCheckAndShowMessageBox(_httpPort, "HTTP"); break; } if (portNotAvailable) { Logging.Error("主控制器启动失败: 端口被占用"); return false; } #endregion Global.MainForm.StatusText(i18N.Translate("Starting ", EncryptedProxyController.Name)); try { result = await Task.Run(() => EncryptedProxyController.Start(server, mode)); } catch (Exception e) { Logging.Error("加密代理启动失败,未处理异常: " + e); result = false; } } if (result) { // 加密代理成功启动 UsingPorts.Add(_socks5Port); switch (mode.Type) { case 0: // 进程代理模式 ModeController = new NFController(); break; case 1: // TUN/TAP 黑名单代理模式 case 2: // TUN/TAP 白名单代理模式 ModeController = new TUNTAPController(); break; case 3: case 5: ModeController = new HTTPController(); break; case 4: // Socks5 代理模式,不需要启动额外的Server result = true; break; } if (ModeController != null) { Global.MainForm.StatusText(i18N.Translate("Starting ", ModeController.Name)); try { result = await Task.Run(() => ModeController.Start(server, mode)); } catch (Exception e) { if (e is DllNotFoundException || e is FileNotFoundException) MessageBoxX.Show(e.Message + "\n\n" + i18N.Translate("Missing File or runtime components"), owner: Global.MainForm); else Logging.Error("模式启动失败,未处理异常: " + e); result = false; } } if (result) { // 成功启动 switch (mode.Type) // 记录使用端口 { case 0: UsingPorts.Add(_redirectorTCPPort); break; case 3: case 5: UsingPorts.Add(_httpPort); break; } switch (mode.Type) { case 0: case 1: case 2: NatTest(); break; } } } if (!result) { Logging.Error("主控制器启动失败"); try { await Stop(); } catch { // ignored } } return result; } public static bool NttTested; /// /// 测试 NAT /// public static void NatTest() { NttTested = false; Task.Run(() => { Global.MainForm.NatTypeStatusText(i18N.Translate("Starting NatTester")); // Thread.Sleep(1000); var (nttResult, natType, localEnd, publicEnd) = NTTController.Start(); if (nttResult) { var country = Utils.Utils.GetCityCode(publicEnd); Global.MainForm.NatTypeStatusText(natType, country); } else Global.MainForm.NatTypeStatusText(natType); NttTested = true; }); } /// /// 停止 /// public static async Task Stop() { _httpPort = _socks5Port = _redirectorTCPPort = 0; _localAddress = null; _savedMode = null; _savedServer = null; UsingPorts.Clear(); var tasks = new Task[] { Task.Run(() => EncryptedProxyController?.Stop()), Task.Run(() => ModeController?.Stop()), Task.Run(() => NTTController.Stop()) }; await Task.WhenAll(tasks); } public static void KillProcessByName(string name) { try { foreach (var p in Process.GetProcessesByName(name)) if (p.MainModule != null && p.MainModule.FileName.StartsWith(Global.NetchDir)) p.Kill(); } catch (Win32Exception e) { Logging.Error($"结束进程 {name} 错误:" + e.Message); } catch (Exception) { // ignored } } /// /// /// /// /// /// /// 端口是否被占用 private static bool PortCheckAndShowMessageBox(int port, string portName, PortType portType = PortType.Both) { if (!PortHelper.PortInUse(port, portType)) return false; MessageBoxX.Show(i18N.TranslateFormat("The {0} port is in use.", portName)); return true; } } }