From a6221c330a13623247df5c16f15f7cee59eb738c Mon Sep 17 00:00:00 2001 From: next-autumn Date: Thu, 13 May 2021 09:58:45 +0800 Subject: [PATCH] mvvmcross --- .../Developers/TrojanGoConfigBuilder.cs | 1 + .../Models/Developers/TrojanGoProject.cs | 1 + ProxySU_Core/ProxySU_Core.csproj | 6 +- ProxySU_Core/packages.config | 4 +- ProxySuper.Core/Helpers/DateTimeUtils.cs | 86 ++ ProxySuper.Core/Models/IProjectSettings.cs | 26 + ProxySuper.Core/Models/TrojanGoSettings.cs | 39 + ProxySuper.Core/Models/XraySettings.cs | 69 ++ ProxySuper.Core/Models/XraySettings_SS.cs | 26 + ProxySuper.Core/Models/XraySettings_Trojan.cs | 13 + ProxySuper.Core/Models/XraySettings_VLESS.cs | 41 + ProxySuper.Core/Models/XraySettings_VMESS.cs | 36 + ProxySuper.Core/Models/XrayType.cs | 34 + ProxySuper.Core/ProxySuper.Core.csproj | 22 + ProxySuper.Core/Services/ProjectBase.cs | 746 ++++++++++++++++++ .../Services/TrojanGoConfigBuilder.cs | 64 ++ ProxySuper.Core/Services/TrojanGoProject.cs | 136 ++++ ProxySuper.Core/Services/XrayConfigBuilder.cs | 234 ++++++ ProxySuper.Core/Services/XrayProject.cs | 321 ++++++++ ProxySuper.Core/ViewModels/MainViewModel.cs | 24 - ProxySuper.Core/packages.config | 1 + ProxySuper.WPF/App.xaml | 7 +- ProxySuper.WPF/MainWindow.xaml | 35 +- ProxySuper.WPF/MainWindow.xaml.cs | 2 +- ProxySuper.WPF/MvxMetroWindow .cs | 119 --- ProxySuper.WPF/ProxySuper.WPF.csproj | 35 +- ProxySuper.WPF/Resources/Languages/en.xaml | 16 + ProxySuper.WPF/Resources/Languages/zh_cn.xaml | 17 + ProxySuper.WPF/Resources/ProxySU.ico | Bin 0 -> 16958 bytes ProxySuper.WPF/Views/MainView.xaml | 37 +- ProxySuper.WPF/Views/MainView.xaml.cs | 5 + ProxySuper.WPF/Views/SecondView.xaml | 12 - ProxySuper.WPF/Views/SecondView.xaml.cs | 28 - ProxySuper.WPF/packages.config | 3 - 34 files changed, 2001 insertions(+), 245 deletions(-) create mode 100644 ProxySuper.Core/Helpers/DateTimeUtils.cs create mode 100644 ProxySuper.Core/Models/IProjectSettings.cs create mode 100644 ProxySuper.Core/Models/TrojanGoSettings.cs create mode 100644 ProxySuper.Core/Models/XraySettings.cs create mode 100644 ProxySuper.Core/Models/XraySettings_SS.cs create mode 100644 ProxySuper.Core/Models/XraySettings_Trojan.cs create mode 100644 ProxySuper.Core/Models/XraySettings_VLESS.cs create mode 100644 ProxySuper.Core/Models/XraySettings_VMESS.cs create mode 100644 ProxySuper.Core/Models/XrayType.cs create mode 100644 ProxySuper.Core/Services/ProjectBase.cs create mode 100644 ProxySuper.Core/Services/TrojanGoConfigBuilder.cs create mode 100644 ProxySuper.Core/Services/TrojanGoProject.cs create mode 100644 ProxySuper.Core/Services/XrayConfigBuilder.cs create mode 100644 ProxySuper.Core/Services/XrayProject.cs delete mode 100644 ProxySuper.WPF/MvxMetroWindow .cs create mode 100644 ProxySuper.WPF/Resources/Languages/en.xaml create mode 100644 ProxySuper.WPF/Resources/Languages/zh_cn.xaml create mode 100644 ProxySuper.WPF/Resources/ProxySU.ico delete mode 100644 ProxySuper.WPF/Views/SecondView.xaml delete mode 100644 ProxySuper.WPF/Views/SecondView.xaml.cs diff --git a/ProxySU_Core/Models/Developers/TrojanGoConfigBuilder.cs b/ProxySU_Core/Models/Developers/TrojanGoConfigBuilder.cs index 6ad81d3..c02a9d0 100644 --- a/ProxySU_Core/Models/Developers/TrojanGoConfigBuilder.cs +++ b/ProxySU_Core/Models/Developers/TrojanGoConfigBuilder.cs @@ -59,4 +59,5 @@ namespace ProxySU_Core.Models.Developers return caddyStr; } } + } diff --git a/ProxySU_Core/Models/Developers/TrojanGoProject.cs b/ProxySU_Core/Models/Developers/TrojanGoProject.cs index ef9bcb2..bf6cf19 100644 --- a/ProxySU_Core/Models/Developers/TrojanGoProject.cs +++ b/ProxySU_Core/Models/Developers/TrojanGoProject.cs @@ -131,4 +131,5 @@ namespace ProxySU_Core.Models.Developers RunCmd("systemctl restart caddy"); } } + } diff --git a/ProxySU_Core/ProxySU_Core.csproj b/ProxySU_Core/ProxySU_Core.csproj index 3dd467a..eae68c3 100644 --- a/ProxySU_Core/ProxySU_Core.csproj +++ b/ProxySU_Core/ProxySU_Core.csproj @@ -72,7 +72,7 @@ ..\packages\ControlzEx.5.0.0\lib\net452\ControlzEx.dll - ..\packages\MahApps.Metro.2.4.4\lib\net46\MahApps.Metro.dll + ..\packages\MahApps.Metro.2.4.5\lib\net46\MahApps.Metro.dll ..\packages\MaterialDesignColors.2.0.0\lib\net452\MaterialDesignColors.dll @@ -86,8 +86,8 @@ ..\packages\Microsoft.Xaml.Behaviors.Wpf.1.1.31\lib\net45\Microsoft.Xaml.Behaviors.dll - - ..\packages\Newtonsoft.Json.12.0.3\lib\net45\Newtonsoft.Json.dll + + ..\packages\Newtonsoft.Json.13.0.1\lib\net45\Newtonsoft.Json.dll ..\packages\QRCoder.1.4.1\lib\net40\QRCoder.dll diff --git a/ProxySU_Core/packages.config b/ProxySU_Core/packages.config index 003b776..5b9579c 100644 --- a/ProxySU_Core/packages.config +++ b/ProxySU_Core/packages.config @@ -1,12 +1,12 @@  - + - + \ No newline at end of file diff --git a/ProxySuper.Core/Helpers/DateTimeUtils.cs b/ProxySuper.Core/Helpers/DateTimeUtils.cs new file mode 100644 index 0000000..9d1673e --- /dev/null +++ b/ProxySuper.Core/Helpers/DateTimeUtils.cs @@ -0,0 +1,86 @@ + +using System; +using System.Collections.Generic; +using System.Globalization; +using System.Linq; +using System.Net; +using System.Runtime.InteropServices; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Helpers +{ + public static class DateTimeUtils + { + /// + /// 从国家授时中心获取标准GMT时间,读取https://www.tsa.cn + /// GMT时间与UTC时间没有差别,可以UTC=GMT + /// + /// 返回网络时间 + public static DateTime GetUTCTime() + { + DateTime time; + try + { + HttpWebRequest request = (HttpWebRequest)WebRequest.Create("https://www.tsa.cn"); + request.Method = "HEAD"; + request.AllowAutoRedirect = false; + HttpWebResponse reponse = (HttpWebResponse)request.GetResponse(); + string cc = reponse.GetResponseHeader("date"); + reponse.Close(); + + bool s = GMTStrParse(cc, out time); + return time; + } + catch + { + return new DateTime(1970, 1, 1, 0, 0, 0, 0); + } + } + + public static bool GMTStrParse(string gmtStr, out DateTime gmtTime) //抓取的date是GMT格式的字符串,这里转成datetime + { + CultureInfo enUS = new CultureInfo("en-US"); + bool s = DateTime.TryParseExact(gmtStr, "r", enUS, DateTimeStyles.None, out gmtTime); + return s; + } + + //设置系统时间的API函数 + [DllImport("kernel32.dll")] + private static extern bool SetLocalTime(ref SYSTEMTIME time); + + [StructLayout(LayoutKind.Sequential)] + private struct SYSTEMTIME + { + public short year; + public short month; + public short dayOfWeek; + public short day; + public short hour; + public short minute; + public short second; + public short milliseconds; + } + + /// + /// 设置系统时间 + /// + /// 需要设置的时间 + /// 返回系统时间设置状态,true为成功,false为失败 + public static bool SetDate(DateTime dt) + { + SYSTEMTIME st; + + st.year = (short)dt.Year; + st.month = (short)dt.Month; + st.dayOfWeek = (short)dt.DayOfWeek; + st.day = (short)dt.Day; + st.hour = (short)dt.Hour; + st.minute = (short)dt.Minute; + st.second = (short)dt.Second; + st.milliseconds = (short)dt.Millisecond; + bool rt = SetLocalTime(ref st); + return rt; + } + } +} diff --git a/ProxySuper.Core/Models/IProjectSettings.cs b/ProxySuper.Core/Models/IProjectSettings.cs new file mode 100644 index 0000000..2b049d0 --- /dev/null +++ b/ProxySuper.Core/Models/IProjectSettings.cs @@ -0,0 +1,26 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Models +{ + public interface IProjectSettings + { + /// + /// 端口 + /// + int Port { get; set; } + + /// + /// 域名 + /// + string Domain { get; set; } + + /// + /// 额外需要开放的端口 + /// + List FreePorts { get; } + } +} diff --git a/ProxySuper.Core/Models/TrojanGoSettings.cs b/ProxySuper.Core/Models/TrojanGoSettings.cs new file mode 100644 index 0000000..2646001 --- /dev/null +++ b/ProxySuper.Core/Models/TrojanGoSettings.cs @@ -0,0 +1,39 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Models +{ + public class TrojanGoSettings : IProjectSettings + { + public List FreePorts + { + get + { + return new List(); + } + } + + /// + /// 域名 + /// + public string Domain { get; set; } + + /// + /// 端口 + /// + public int Port { get; set; } + + /// + /// 密码 + /// + public string Password { get; set; } + + /// + /// 伪装域名 + /// + public string MaskDomain { get; set; } + } +} diff --git a/ProxySuper.Core/Models/XraySettings.cs b/ProxySuper.Core/Models/XraySettings.cs new file mode 100644 index 0000000..77c7ce2 --- /dev/null +++ b/ProxySuper.Core/Models/XraySettings.cs @@ -0,0 +1,69 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Models +{ + public partial class XraySettings : IProjectSettings + { + public List FreePorts + { + get + { + return new List + { + VLESS_KCP_Port, + VMESS_KCP_Port, + ShadowsocksPort, + }; + } + } + + /// + /// UUID + /// + public string UUID { get; set; } + + /// + /// 端口 + /// + public int Port { get; set; } + + /// + /// 域名 + /// + public string Domain { get; set; } + + /// + /// 伪装域名 + /// + public string MaskDomain { get; set; } + + /// + /// 安装类型 + /// + public List Types { get; set; } + + /// + /// 根据xray类型获取路径 + /// + /// + /// + public string GetPath(XrayType type) + { + switch (type) + { + case XrayType.VLESS_WS: + return VLESS_WS_Path; + case XrayType.VMESS_TCP: + return VMESS_TCP_Path; + case XrayType.VMESS_WS: + return VMESS_WS_Path; + default: + return string.Empty; + } + } + } +} diff --git a/ProxySuper.Core/Models/XraySettings_SS.cs b/ProxySuper.Core/Models/XraySettings_SS.cs new file mode 100644 index 0000000..0f0e4fb --- /dev/null +++ b/ProxySuper.Core/Models/XraySettings_SS.cs @@ -0,0 +1,26 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Models +{ + public partial class XraySettings + { + /// + /// ss password + /// + public string ShadowsocksPassword { get; set; } + + /// + /// ss method + /// + public string ShadowsocksMethod { get; set; } + + /// + /// ss port + /// + public int ShadowsocksPort { get; set; } + } +} diff --git a/ProxySuper.Core/Models/XraySettings_Trojan.cs b/ProxySuper.Core/Models/XraySettings_Trojan.cs new file mode 100644 index 0000000..1488665 --- /dev/null +++ b/ProxySuper.Core/Models/XraySettings_Trojan.cs @@ -0,0 +1,13 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Models +{ + public partial class XraySettings + { + public string TrojanPassword { get; set; } + } +} diff --git a/ProxySuper.Core/Models/XraySettings_VLESS.cs b/ProxySuper.Core/Models/XraySettings_VLESS.cs new file mode 100644 index 0000000..3bd71d4 --- /dev/null +++ b/ProxySuper.Core/Models/XraySettings_VLESS.cs @@ -0,0 +1,41 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Models +{ + public partial class XraySettings + { + /// + /// websocket path + /// + public string VLESS_WS_Path { get; set; } + + /// + /// kcp seed + /// + public string VLESS_KCP_Seed { get; set; } + + /// + /// kcp type + /// + public string VLESS_KCP_Type { get; set; } + + /// + /// kcp port + /// + public int VLESS_KCP_Port { get; set; } + + /// + /// grpc port + /// + public string VLESS_gRPC_Port { get; set; } + + /// + /// grpc service name + /// + public string VLESS_gRPC_ServiceName { get; set; } + } +} diff --git a/ProxySuper.Core/Models/XraySettings_VMESS.cs b/ProxySuper.Core/Models/XraySettings_VMESS.cs new file mode 100644 index 0000000..19a009b --- /dev/null +++ b/ProxySuper.Core/Models/XraySettings_VMESS.cs @@ -0,0 +1,36 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Models +{ + public partial class XraySettings + { + /// + /// vmess websocket path + /// + public string VMESS_WS_Path { get; set; } + + /// + /// mvess tcp path + /// + public string VMESS_TCP_Path { get; set; } + + /// + /// vmess kcp seed + /// + public string VMESS_KCP_Seed { get; set; } + + /// + /// vmess kcp type + /// + public string VMESS_KCP_Type { get; set; } + + /// + /// vmess kcp port + /// + public int VMESS_KCP_Port { get; set; } + } +} diff --git a/ProxySuper.Core/Models/XrayType.cs b/ProxySuper.Core/Models/XrayType.cs new file mode 100644 index 0000000..de770a5 --- /dev/null +++ b/ProxySuper.Core/Models/XrayType.cs @@ -0,0 +1,34 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Models +{ + public enum XrayType + { + // 入口 + VLESS_TCP_XTLS = 100, + + // VLESS 101开头 + VLESS_TCP = 101, + VLESS_WS = 102, + VLESS_H2 = 103, + VLESS_KCP = 104, + VLESS_gRPC = 110, + + // VMESS 201开头 + VMESS_TCP = 201, + VMESS_WS = 202, + VMESS_H2 = 203, + VMESS_KCP = 204, + + // Trojan 301开头 + Trojan_TCP = 301, + Trojan_WS = 302, + + // SS + ShadowsocksAEAD = 401 + } +} diff --git a/ProxySuper.Core/ProxySuper.Core.csproj b/ProxySuper.Core/ProxySuper.Core.csproj index b68f50e..d9d730d 100644 --- a/ProxySuper.Core/ProxySuper.Core.csproj +++ b/ProxySuper.Core/ProxySuper.Core.csproj @@ -34,6 +34,14 @@ ..\packages\MvvmCross.7.1.2\lib\net461\MvvmCross.dll + + False + ..\json.net\net40\Newtonsoft.Json.dll + + + + ..\packages\SSH.NET.2020.0.1\lib\net40\Renci.SshNet.dll + ..\packages\System.Console.4.3.1\lib\net46\System.Console.dll @@ -48,7 +56,21 @@ + + + + + + + + + + + + + + diff --git a/ProxySuper.Core/Services/ProjectBase.cs b/ProxySuper.Core/Services/ProjectBase.cs new file mode 100644 index 0000000..3905609 --- /dev/null +++ b/ProxySuper.Core/Services/ProjectBase.cs @@ -0,0 +1,746 @@ +using ProxySuper.Core.Helpers; +using ProxySuper.Core.Models; +using Renci.SshNet; +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using System.Windows; + +namespace ProxySuper.Core.Services +{ + public enum CmdType + { + None, + Apt, + Dnf, + Yum + } + + public abstract class ProjectBase where TSettings : IProjectSettings + { + private SshClient _sshClient; + + protected Action WriteOutput; + + protected CmdType CmdType { get; set; } + + protected bool IsSELinux { get; set; } + + protected bool OnlyIpv6 { get; set; } + + protected string IPv4 { get; set; } + + protected string IPv6 { get; set; } + + protected TSettings Parameters { get; set; } + + public ProjectBase(SshClient sshClient, TSettings parameters, Action writeOutput) + { + _sshClient = sshClient; + WriteOutput = writeOutput; + Parameters = parameters; + } + + protected string RunCmd(string cmdStr) + { + var cmd = _sshClient.CreateCommand(cmdStr); + WriteOutput(cmdStr); + + var result = cmd.Execute(); + WriteOutput(result); + return result; + } + + /// + /// 执行安装命令 + /// + public abstract void Install(); + + /// + /// 配置系统基础环境 + /// + protected void EnsureSystemEnv() + { + string cmd; + + // 确认安装命令 + if (CmdType == CmdType.None) + { + cmd = RunCmd("command -v apt-get"); + if (!string.IsNullOrEmpty(cmd)) + { + CmdType = CmdType.Apt; + } + } + + if (CmdType == CmdType.None) + { + cmd = RunCmd("command -v dnf"); + if (!string.IsNullOrEmpty(cmd)) + { + CmdType = CmdType.Dnf; + } + } + + if (CmdType == CmdType.None) + { + cmd = RunCmd("command -v yum"); + if (!string.IsNullOrEmpty(cmd)) + { + CmdType = CmdType.Yum; + } + } + + // systemctl + cmd = RunCmd("command -v systemctl"); + var hasSystemCtl = !string.IsNullOrEmpty(cmd); + + // SELinux + cmd = RunCmd("command -v getenforce"); + IsSELinux = !string.IsNullOrEmpty(cmd); + + if (CmdType == CmdType.None || !hasSystemCtl) + { + throw new Exception("系统缺乏必要的安装组件如:apt-get||dnf||yum||Syetemd,主机系统推荐使用:CentOS 7/8,Debian 8/9/10,Ubuntu 16.04及以上版本"); + } + + + // 判断是否启用了SELinux,如果启用了,并且工作在Enforcing模式下,则改为Permissive模式 + if (IsSELinux) + { + cmd = RunCmd("getenforce"); + + // 检测到系统启用SELinux,且工作在严格模式下,需改为宽松模式 + if (cmd.Contains("Enforcing")) + { + RunCmd("setenforce 0"); + RunCmd(@"sed -i 's/SELINUX=enforcing/SELINUX=permissive/' /etc/selinux/config"); + } + } + } + + /// + /// 确保Root账户登陆 + /// + protected void EnsureRootAuth() + { + // 禁止一些可能产生的干扰信息 + RunCmd(@"sed -i 's/echo/#echo/g' ~/.bashrc"); + RunCmd(@"sed -i 's/echo/#echo/g' ~/.profile"); + + + // 检测是否运行在Root权限下 + var cmd = RunCmd("id -u"); + if (!cmd.Equals("0\n")) + { + throw new Exception("请使用Root账户登陆主机"); + } + } + + /// + /// 配置IPV6环境 + /// + protected void ConfigureIPv6() + { + if (IsOnlyIpv6()) + { + SetNat64(); + } + } + + /// + /// 配置必要的软件 + /// + protected void ConfigureSoftware() + { + string cmd = RunCmd("command -v sudo"); + if (string.IsNullOrEmpty(cmd)) + { + RunCmd(GetInstallCmd("sudo")); + } + + // 安装curl,wget,unzip + cmd = RunCmd("command -v curl"); + if (string.IsNullOrEmpty(cmd)) + { + RunCmd(GetInstallCmd("curl")); + } + + cmd = RunCmd("command -v wget"); + if (string.IsNullOrEmpty(cmd)) + { + RunCmd(GetInstallCmd("wget")); + } + + cmd = RunCmd("command -v unzip"); + if (string.IsNullOrEmpty(cmd)) + { + RunCmd(GetInstallCmd("unzip")); + } + + // 安装dig + cmd = RunCmd("command -v dig"); + if (string.IsNullOrEmpty(cmd)) + { + if (CmdType == CmdType.Apt) + { + RunCmd(GetUpdateCmd()); + RunCmd(GetInstallCmd("dnsutils")); + } + else if (CmdType == CmdType.Dnf) + { + RunCmd(GetUpdateCmd()); + RunCmd(GetInstallCmd("bind-utils")); + } + else if (CmdType == CmdType.Yum) + { + RunCmd(GetUpdateCmd()); + RunCmd(GetInstallCmd("bind-utils")); + } + } + + + // 处理极其少见的xz-utils未安装的情况 + if (CmdType == CmdType.Apt) + { + RunCmd(GetInstallCmd("xz-utils")); + } + else + { + RunCmd(GetInstallCmd("xz-devel")); + } + + // 检测是否安装lsof + cmd = RunCmd("command -v lsof"); + if (string.IsNullOrEmpty(cmd)) + { + RunCmd(GetInstallCmd("lsof")); + } + } + + /// + /// 关闭端口 + /// + /// + protected void ClosePort(params int[] portList) + { + string cmd; + + cmd = RunCmd("command -v firewall-cmd"); + if (!string.IsNullOrEmpty(cmd)) + { + //有很奇怪的vps主机,在firewalld未运行时,端口是关闭的,无法访问。所以要先启动firewalld + //用于保证acme.sh申请证书成功 + cmd = RunCmd("firewall-cmd --state"); + if (cmd.Trim() != "running") + { + RunCmd("systemctl restart firewalld"); + } + + foreach (var port in portList) + { + RunCmd($"firewall-cmd --zone=public --remove-port={port}/tcp --permanent"); + RunCmd($"firewall-cmd --zone=public --remove-port={port}/udp --permanent"); + } + RunCmd("yes | firewall-cmd --reload"); + } + else + { + cmd = RunCmd("command -v ufw"); + if (!string.IsNullOrEmpty(cmd)) + { + foreach (var port in portList) + { + RunCmd($"ufw delete allow {port}/tcp"); + RunCmd($"ufw delete allow {port}/udp"); + } + RunCmd("yes | ufw reload"); + } + } + } + + /// + /// 开放端口 + /// + /// + protected void OpenPort(params int[] portList) + { + + string cmd; + + cmd = RunCmd("command -v firewall-cmd"); + if (!string.IsNullOrEmpty(cmd)) + { + //有很奇怪的vps主机,在firewalld未运行时,端口是关闭的,无法访问。所以要先启动firewalld + //用于保证acme.sh申请证书成功 + cmd = RunCmd("firewall-cmd --state"); + if (cmd.Trim() != "running") + { + RunCmd("systemctl restart firewalld"); + } + + foreach (var port in portList) + { + RunCmd($"firewall-cmd --zone=public --add-port={port}/tcp --permanent"); + RunCmd($"firewall-cmd --zone=public --add-port={port}/udp --permanent"); + } + RunCmd("yes | firewall-cmd --reload"); + } + else + { + cmd = RunCmd("command -v ufw"); + if (!string.IsNullOrEmpty(cmd)) + { + foreach (var port in portList) + { + RunCmd($"ufw allow {port}/tcp"); + RunCmd($"ufw allow {port}/udp"); + } + RunCmd("yes | ufw reload"); + } + } + } + + /// + /// 配置防火墙 + /// + protected void ConfigureFirewall() + { + var portList = new List(); + portList.Add(80); + portList.Add(Parameters.Port); + portList.AddRange(Parameters.FreePorts); + + OpenPort(portList.ToArray()); + } + + /// + /// 配置同步时间差 + /// + protected void SyncTimeDiff() + { + RunCmd("rm -f /etc/localtime"); + RunCmd("ln -s /usr/share/zoneinfo/UTC /etc/localtime"); + + var result = RunCmd("date +%s"); + var vpsSeconds = Convert.ToInt64(result); + var localSeconds = (int)(DateTime.Now.ToUniversalTime() - DateTime.Parse("1970-01-01")).TotalSeconds; + + if (Math.Abs(vpsSeconds - localSeconds) >= 90) + { + // 同步本地时间 + var netUtcTime = DateTimeUtils.GetUTCTime(); + DateTimeUtils.SetDate(netUtcTime.ToLocalTime()); + + // 同步VPS时间 + var utcTS = DateTimeUtils.GetUTCTime() - new DateTime(1970, 1, 1, 0, 0, 0, 0); + long timeStampVPS = Convert.ToInt64(utcTS.TotalSeconds); + RunCmd($"date --set=\"$(date \"+%Y-%m-%d %H:%M:%S\" -d @{timeStampVPS.ToString()})\""); + } + } + + /// + /// 验证域名是否绑定了主机 + /// + protected void ValidateDomain() + { + if (OnlyIpv6) + { + string cmdFilter = @"| grep -oE '(([0-9a-fA-F]{1,4}:){7,7}[0-9a-fA-F]{1,4}|([0-9a-fA-F]{1,4}:){1,7}:|([0-9a-fA-F]{1,4}:){1,6}:[0-9a-fA-F]{1,4}|([0-9a-fA-F]{1,4}:){1,5}(:[0-9a-fA-F]{1,4}){1,2}|([0-9a-fA-F]{1,4}:){1,4}(:[0-9a-fA-F]{1,4}){1,3}|([0-9a-fA-F]{1,4}:){1,3}(:[0-9a-fA-F]{1,4}){1,4}|([0-9a-fA-F]{1,4}:){1,2}(:[0-9a-fA-F]{1,4}){1,5}|[0-9a-fA-F]{1,4}:((:[0-9a-fA-F]{1,4}){1,6})|:((:[0-9a-fA-F]{1,4}){1,7}|:)|fe80:(:[0-9a-fA-F]{0,4}){0,4}%[0-9a-zA-Z]{1,}|::(ffff(:0{1,4}){0,1}:){0,1}((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\.){3,3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])|([0-9a-fA-F]{1,4}:){1,4}:((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\.){3,3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9]))' | head -n 1"; + var cmd = $"dig @resolver1.opendns.com AAAA {Parameters.Domain} +short -6 {cmdFilter}"; + var result = RunCmd(cmd).TrimEnd('\r', '\n'); + + if (result == IPv6) return; + } + + else + { + string cmdFilter = @"| grep -oE '[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+' | head -n 1"; + var cmd = $"dig @resolver1.opendns.com A {Parameters.Domain} +short -4 {cmdFilter}"; + var result = RunCmd(cmd).TrimEnd('\r', '\n'); + + if (result == IPv4) return; + + } + + + var btnResult = MessageBox.Show( + $"{Parameters.Domain}未能正常解析到服务器的IP,如果您使用了CDN请忽略,是否继续安装?", "提示", MessageBoxButton.YesNo); + + if (btnResult == MessageBoxResult.No) + { + throw new Exception($"域名解析失败,安装停止!"); + } + + } + + /// + /// 判断是否安装某个软件 + /// + /// + /// + protected bool FileExists(string path) + { + var cmdStr = $"if [[ -f {path} ]];then echo '1';else echo '0'; fi"; + var cmd = RunCmd(cmdStr); + return cmd.Trim() == "1"; + } + + /// + /// 安装 Caddy + /// + protected void InstallCaddy() + { + RunCmd("rm -rf caddy_install.sh"); + RunCmd("curl -o caddy_install.sh https://raw.githubusercontent.com/proxysu/shellscript/master/Caddy-Naive/caddy-naive-install.sh"); + RunCmd("yes | bash caddy_install.sh"); + RunCmd("rm -rf caddy_install.sh"); + RunCmd("systemctl enable caddy.service"); + } + + /// + /// 卸载 Caddy + /// + protected void UninstallCaddy() + { + RunCmd("rm -rf caddy_install.sh"); + RunCmd("curl -o caddy_install.sh https://raw.githubusercontent.com/proxysu/shellscript/master/Caddy-Naive/caddy-naive-install.sh"); + RunCmd("yes | bash caddy_install.sh uninstall"); + RunCmd("rm -rf caddy_install.sh"); + RunCmd("rm -rf /usr/share/caddy"); + } + + + #region 检测系统环境 + + private bool IsOnlyIpv6() + { + string cmd; + + cmd = RunCmd(@"curl -s https://api.ip.sb/ip --ipv4 --max-time 8"); + IPv4 = cmd.TrimEnd('\r', '\n'); + + if (!string.IsNullOrEmpty(IPv4)) + { + OnlyIpv6 = false; + return false; + } + + cmd = RunCmd(@"curl -s https://api.ip.sb/ip --ipv6 --max-time 8"); + IPv6 = cmd.TrimEnd('\r', '\n'); + + if (string.IsNullOrEmpty(IPv6)) + { + throw new Exception("未检测可用的的IP地址"); + } + + OnlyIpv6 = true; + return OnlyIpv6; + } + + private bool SetPortFree(int port, bool force = true) + { + string result = RunCmd($"lsof -n -P -i :{port} | grep LISTEN"); + + if (!string.IsNullOrEmpty(result)) + { + if (force) + { + var btnResult = MessageBox.Show($"{port}端口被占用,将强制停止占用{port}端口的程序?", "提示", MessageBoxButton.YesNo); + if (btnResult == MessageBoxResult.No) + { + throw new Exception($"{port}端口被占用,安装停止!"); + } + + string[] process = result.Split(' '); + RunCmd($"systemctl stop {process[0]}"); + RunCmd($"systemctl disable {process[0]}"); + RunCmd($"pkill {process[0]}"); + return SetPortFree(port, force: false); + } + else + { + return false; + } + } + + return true; + } + + public void ConfigurePort() + { + if (Parameters.Port == 80 || Parameters.Port == 443) + { + SetPortFree(80); + SetPortFree(443); + } + else + { + SetPortFree(80); + SetPortFree(443); + SetPortFree(Parameters.Port); + + Parameters.FreePorts.ForEach(port => + { + SetPortFree(port); + }); + } + } + + protected void SetNat64() + { + var dns64List = FilterFastestIP(); + if (dns64List.Count == 0) + { + throw new Exception("未找到有效的Nat64网关"); + } + + var exists = FileExists("/etc/resolv.conf.proxysu"); + if (!exists) + { + var cmdStr = @"mv /etc/resolv.conf /etc/resolv.conf.proxysu"; + RunCmd(cmdStr); + } + + foreach (var gateip in dns64List) + { + RunCmd($"echo \"nameserver {gateip}\" > /etc/resolv.conf"); + } + } + + protected void RemoveNat64() + { + RunCmd("rm /etc/resolv.conf"); + RunCmd("mv /etc/resolv.conf.proxysu /etc/resolv.conf"); + } + + private List FilterFastestIP() + { + string[] gateNat64 = { + "2a01:4f9:c010:3f02::1", + "2001:67c:2b0::4", + "2001:67c:2b0::6", + "2a09:11c0:f1:bbf0::70", + "2a01:4f8:c2c:123f::1", + "2001:67c:27e4:15::6411", + "2001:67c:27e4::64", + "2001:67c:27e4:15::64", + "2001:67c:27e4::60", + "2a00:1098:2b::1", + "2a03:7900:2:0:31:3:104:161", + "2a00:1098:2c::1", + "2a09:11c0:100::53", + }; + + Dictionary dns64List = new Dictionary(); + foreach (var gateip in gateNat64) + { + var cmdStr = $"ping6 -c4 {gateip} | grep avg | awk '{{print $4}}'|cut -d/ -f2"; + var cmd = RunCmd(cmdStr); + if (!string.IsNullOrEmpty(cmd)) + { + if (float.TryParse(cmd, out float delay)) + { + dns64List.Add(gateip, delay); + } + } + } + + return dns64List.Keys.ToList(); + } + + #endregion + + + #region BBR + private bool CheckKernelVersionBBR(string kernelVer) + { + string[] linuxKernelCompared = kernelVer.Split('.'); + if (int.Parse(linuxKernelCompared[0]) > 4) + { + return true; + } + else if (int.Parse(linuxKernelCompared[0]) < 4) + { + return false; + } + else if (int.Parse(linuxKernelCompared[0]) == 4) + { + if (int.Parse(linuxKernelCompared[1]) >= 9) + { + return true; + } + else if (int.Parse(linuxKernelCompared[1]) < 9) + { + return false; + } + + } + return false; + + } + + protected void EnableBBR() + { + var osVersion = RunCmd("uname -r"); + var canInstallBBR = CheckKernelVersionBBR(osVersion.Split('-')[0]); + + var bbrInfo = RunCmd("sysctl net.ipv4.tcp_congestion_control | grep bbr"); + var installed = bbrInfo.Contains("bbr"); + if (canInstallBBR && !installed) + { + RunCmd(@"bash -c 'echo ""net.core.default_qdisc=fq"" >> /etc/sysctl.conf'"); + RunCmd(@"bash -c 'echo ""net.ipv4.tcp_congestion_control=bbr"" >> /etc/sysctl.conf'"); + RunCmd(@"sysctl -p"); + + if (OnlyIpv6) + { + RemoveNat64(); + } + WriteOutput("BBR启动成功"); + } + + if (!canInstallBBR) + { + WriteOutput("****** 系统不满足启用BBR条件,启动失败。 ******"); + } + + } + #endregion + + /// + /// 安装证书 + /// + /// + /// + protected void InstallCert(string dirPath, string certName, string keyName) + { + string certPath = Path.Combine(dirPath, certName); + string keyPath = Path.Combine(dirPath, keyName); + + // 安装依赖 + RunCmd(GetInstallCmd("socat")); + + // 解决搬瓦工CentOS缺少问题 + RunCmd(GetInstallCmd("automake autoconf libtool")); + + // 安装Acme + var result = RunCmd($"curl https://get.acme.sh yes | sh"); + if (result.Contains("Install success")) + { + WriteOutput("安装 acme.sh 成功"); + } + else + { + WriteOutput("安装 acme.sh 失败,请联系开发者!"); + throw new Exception("安装 acme.sh 失败,请联系开发者!"); + } + + RunCmd("cd ~/.acme.sh/"); + RunCmd("alias acme.sh=~/.acme.sh/acme.sh"); + + // 申请证书 + if (OnlyIpv6) + { + var cmd = $"/root/.acme.sh/acme.sh --force --debug --issue --standalone -d {Parameters.Domain} --listen-v6"; + result = RunCmd(cmd); + } + else + { + var cmd = $"/root/.acme.sh/acme.sh --force --debug --issue --standalone -d {Parameters.Domain}"; + result = RunCmd(cmd); + } + + if (result.Contains("success")) + { + WriteOutput("申请证书成功"); + } + else + { + WriteOutput("申请证书失败,如果申请次数过多请更换二级域名,或联系开发者!"); + throw new Exception("申请证书失败,如果申请次数过多请更换二级域名,或联系开发者!"); + } + + // 安装证书 + RunCmd($"mkdir -p {dirPath}"); + RunCmd($"/root/.acme.sh/acme.sh --installcert -d {Parameters.Domain} --certpath {certPath} --keypath {keyPath} --capath {certPath}"); + + result = RunCmd($@"if [ ! -f ""{keyPath}"" ]; then echo ""0""; else echo ""1""; fi | head -n 1"); + + if (result.Contains("1")) + { + WriteOutput("安装证书成功"); + } + else + { + WriteOutput("安装证书失败,请联系开发者!"); + throw new Exception("安装证书失败,请联系开发者!"); + } + + RunCmd($"chmod 755 {dirPath}"); + } + + /// + /// 上传文件 + /// + /// + /// + protected void UploadFile(Stream stream, string path) + { + using (var sftp = new SftpClient(_sshClient.ConnectionInfo)) + { + sftp.Connect(); + sftp.UploadFile(stream, path, true); + sftp.Disconnect(); + } + } + + /// + /// 根据系统环境匹配更新命令 + /// + /// + protected string GetUpdateCmd() + { + if (CmdType == CmdType.Apt) + { + return "apt-get update"; + } + else if (CmdType == CmdType.Dnf) + { + return "dnf clean all;dnf makecache"; + } + else if (CmdType == CmdType.Yum) + { + return "yum clean all;yum makecache"; + } + + throw new Exception("未识别的系统"); + } + + /// + /// 根据系统匹配安装命令 + /// + /// + /// + protected string GetInstallCmd(string soft) + { + if (CmdType == CmdType.Apt) + { + return "echo y | apt-get install " + soft; + } + else if (CmdType == CmdType.Dnf) + { + return "echo y | dnf -y install " + soft; + } + else if (CmdType == CmdType.Yum) + { + return "echo y | yum -y install " + soft; + } + + throw new Exception("未识别的系统"); + } + } +} diff --git a/ProxySuper.Core/Services/TrojanGoConfigBuilder.cs b/ProxySuper.Core/Services/TrojanGoConfigBuilder.cs new file mode 100644 index 0000000..2c8a488 --- /dev/null +++ b/ProxySuper.Core/Services/TrojanGoConfigBuilder.cs @@ -0,0 +1,64 @@ +using Newtonsoft.Json; +using Newtonsoft.Json.Linq; +using ProxySuper.Core.Models; +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Services +{ + public class TrojanGoConfigBuilder + { + public static readonly int WebPort = 8088; + + public static readonly string TrojanGoSettingPath = @"Templates\trojan-go\trojan-go.json"; + + public static readonly string CaddyFilePath = @"Templates\trojan-go\base.caddyfile"; + + public static string BuildTrojanGoConfig(TrojanGoSettings parameters) + { + var jsonStr = File.ReadAllText(TrojanGoSettingPath); + var settings = JToken.FromObject(JsonConvert.DeserializeObject(jsonStr)); + + settings["remote_port"] = WebPort; + settings["password"][0] = parameters.Password; + settings["ssl"]["sni"] = parameters.Domain; + + return JsonConvert.SerializeObject(settings, Formatting.Indented, new JsonSerializerSettings() + { + NullValueHandling = NullValueHandling.Ignore + }); + } + + public static string BuildCaddyConfig(TrojanGoSettings parameters, bool useCustomWeb = false) + { + var caddyStr = File.ReadAllText(CaddyFilePath); + caddyStr.Replace("##domain##", parameters.Domain); + caddyStr.Replace("##port##", WebPort.ToString()); + + if (!useCustomWeb && !string.IsNullOrEmpty(parameters.MaskDomain)) + { + var prefix = "http://"; + if (parameters.MaskDomain.StartsWith("https://")) + { + prefix = "https://"; + } + var domain = parameters.MaskDomain + .TrimStart("http://".ToCharArray()) + .TrimStart("https://".ToCharArray()); + + caddyStr = caddyStr.Replace("##reverse_proxy##", $"reverse_proxy {prefix}{domain} {{ \n header_up Host {domain} \n }}"); + } + else + { + caddyStr = caddyStr.Replace("##reverse_proxy##", ""); + } + + return caddyStr; + } + } + +} diff --git a/ProxySuper.Core/Services/TrojanGoProject.cs b/ProxySuper.Core/Services/TrojanGoProject.cs new file mode 100644 index 0000000..3722eef --- /dev/null +++ b/ProxySuper.Core/Services/TrojanGoProject.cs @@ -0,0 +1,136 @@ +using ProxySuper.Core.Models; +using Renci.SshNet; +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using System.Windows; + +namespace ProxySuper.Core.Services +{ + public class TrojanGoProject : ProjectBase + { + public TrojanGoProject(SshClient sshClient, TrojanGoSettings parameters, Action writeOutput) : base(sshClient, parameters, writeOutput) + { + } + + public override void Install() + { + try + { + EnsureRootAuth(); + + if (FileExists("/usr/local/bin/xray")) + { + var btnResult = MessageBox.Show("已经安装Xray,是否需要重装?", "提示", MessageBoxButton.YesNo); + if (btnResult == MessageBoxResult.No) + { + MessageBox.Show("安装终止", "提示"); + return; + } + } + + WriteOutput("检测安装系统环境..."); + EnsureSystemEnv(); + WriteOutput("检测安装系统环境完成"); + + WriteOutput("配置服务器端口..."); + ConfigurePort(); + WriteOutput("端口配置完成"); + + WriteOutput("安装必要的系统工具..."); + ConfigureSoftware(); + WriteOutput("系统工具安装完成"); + + WriteOutput("检测IP6..."); + ConfigureIPv6(); + WriteOutput("检测IP6完成"); + + WriteOutput("配置防火墙..."); + ConfigureFirewall(); + WriteOutput("防火墙配置完成"); + + WriteOutput("同步系统和本地时间..."); + SyncTimeDiff(); + WriteOutput("时间同步完成"); + + WriteOutput("检测域名是否绑定本机IP..."); + ValidateDomain(); + WriteOutput("域名检测完成"); + + WriteOutput("安装Trojan-Go..."); + InstallTrojanGo(); + WriteOutput("Trojan-Go安装完成"); + + WriteOutput("安装Caddy..."); + InstallCaddy(); + WriteOutput("Caddy安装完成"); + + WriteOutput("启动BBR"); + EnableBBR(); + + UploadCaddyFile(); + WriteOutput("************"); + WriteOutput("安装完成,尽情享用吧......"); + WriteOutput("************"); + } + catch (Exception ex) + { + var errorLog = "安装终止," + ex.Message; + WriteOutput(errorLog); + MessageBox.Show(errorLog); + } + } + + private void InstallTrojanGo() + { + WriteOutput("安装Trojan-Go"); + RunCmd(@"curl https://raw.githubusercontent.com/proxysu/shellscript/master/trojan-go.sh yes | bash"); + var success = FileExists("/usr/local/etc/trojan-go"); + if (success == false) + { + throw new Exception("trojan-go 安装失败,请联系开发者!"); + } + + RunCmd($"sed -i 's/User=nobody/User=root/g' /etc/systemd/system/xray.service"); + RunCmd($"sed -i 's/CapabilityBoundingSet=/#CapabilityBoundingSet=/g' /etc/systemd/system/xray.service"); + RunCmd($"sed -i 's/AmbientCapabilities=/#AmbientCapabilities=/g' /etc/systemd/system/xray.service"); + RunCmd($"systemctl daemon-reload"); + + RunCmd("systemctl enable trojan-go"); + RunCmd("systemctl start trojan-go"); + WriteOutput("Trojan-Go 安装完成"); + + InstallCert( + dirPath: "/usr/local/etc/trojan-go", + certName: "trojan-go.crt", + keyName: "trojan-go.key"); + + if (FileExists("/usr/local/etc/trojan-go/config.json")) + { + RunCmd("mv /usr/local/etc/trojan-go/config.json config.json.old"); + } + + // 上传配置 + var settings = TrojanGoConfigBuilder.BuildTrojanGoConfig(Parameters); + var stream = new MemoryStream(Encoding.UTF8.GetBytes(settings)); + UploadFile(stream, "/usr/local/etc/trojan-go/config.json"); + RunCmd("systemctl restart trojan-go"); + } + + private void UploadCaddyFile(bool useCustomWeb = false) + { + var config = TrojanGoConfigBuilder.BuildCaddyConfig(Parameters, useCustomWeb); + var stream = new MemoryStream(Encoding.UTF8.GetBytes(config)); + if (FileExists("/etc/caddy/Caddyfile")) + { + RunCmd("mv /etc/caddy/Caddyfile /etc/caddy/Caddyfile.back"); + } + UploadFile(stream, "/etc/caddy/Caddyfile"); + RunCmd("systemctl restart caddy"); + } + } + +} diff --git a/ProxySuper.Core/Services/XrayConfigBuilder.cs b/ProxySuper.Core/Services/XrayConfigBuilder.cs new file mode 100644 index 0000000..998bfe1 --- /dev/null +++ b/ProxySuper.Core/Services/XrayConfigBuilder.cs @@ -0,0 +1,234 @@ +using Newtonsoft.Json; +using Newtonsoft.Json.Linq; +using ProxySuper.Core.Models; +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace ProxySuper.Core.Services +{ + public class XrayConfigBuilder + { + private const string ServerLogDir = @"Templates\xray\server\00_log"; + private const string ServerApiDir = @"Templates\xray\server\01_api"; + private const string ServerDnsDir = @"Templates\xray\server\02_dns"; + private const string ServerRoutingDir = @"Templates\xray\server\03_routing"; + private const string ServerPolicyDir = @"Templates\xray\server\04_policy"; + private const string ServerInboundsDir = @"Templates\xray\server\05_inbounds"; + private const string ServerOutboundsDir = @"Templates\xray\server\06_outbounds"; + private const string ServerTransportDir = @"Templates\xray\server\07_transport"; + private const string ServerStatsDir = @"Templates\xray\server\08_stats"; + private const string ServerReverseDir = @"Templates\xray\server\09_reverse"; + private const string CaddyFileDir = @"Templates\xray\caddy"; + + public static int VLESS_TCP_Port = 1110; + public static int VLESS_WS_Port = 1111; + public static int VLESS_H2_Port = 1112; + public static int VLESS_mKCP_Port = 1113; + + public static int VMESS_TCP_Port = 1210; + public static int VMESS_WS_Port = 1211; + public static int VMESS_H2_Port = 1212; + + public static int Trojan_TCP_Port = 1310; + public static int Trojan_WS_Port = 1311; + + public static int FullbackPort = 8080; + + + + public static dynamic LoadXrayConfig() + { + dynamic logObj = LoadJsonObj(Path.Combine(ServerLogDir, "00_log.json")); + dynamic apiObj = LoadJsonObj(Path.Combine(ServerApiDir, "01_api.json")); + dynamic dnsObj = LoadJsonObj(Path.Combine(ServerDnsDir, "02_dns.json")); + dynamic routingObj = LoadJsonObj(Path.Combine(ServerRoutingDir, "03_routing.json")); + dynamic policyObj = LoadJsonObj(Path.Combine(ServerPolicyDir, "04_policy.json")); + dynamic inboundsObj = LoadJsonObj(Path.Combine(ServerInboundsDir, "05_inbounds.json")); + dynamic outboundsObj = LoadJsonObj(Path.Combine(ServerOutboundsDir, "06_outbounds.json")); + dynamic transportObj = LoadJsonObj(Path.Combine(ServerTransportDir, "07_transport.json")); + dynamic statsObj = LoadJsonObj(Path.Combine(ServerStatsDir, "08_stats.json")); + dynamic reverseObj = LoadJsonObj(Path.Combine(ServerReverseDir, "09_reverse.json")); + + return new + { + log = logObj["log"], + //api = apiObj["api"], api不能为空 + dns = dnsObj["dns"], + routing = routingObj["routing"], + policy = policyObj["policy"], + inbounds = inboundsObj["inbounds"], + outbounds = outboundsObj["outbounds"], + transport = transportObj["transport"], + stats = statsObj["stats"], + reverse = reverseObj["reverse"] + }; + } + + public static string BuildCaddyConfig(XraySettings parameters, bool useCustomWeb = false) + { + var caddyStr = File.ReadAllText(Path.Combine(CaddyFileDir, "base.caddyfile")); + caddyStr = caddyStr.Replace("##domain##", parameters.Domain); + caddyStr = caddyStr.Replace("##port##", FullbackPort.ToString()); + + if (!useCustomWeb && !string.IsNullOrEmpty(parameters.MaskDomain)) + { + var prefix = "http://"; + if (parameters.MaskDomain.StartsWith("https://")) + { + prefix = "https://"; + } + var domain = parameters.MaskDomain + .TrimStart("http://".ToCharArray()) + .TrimStart("https://".ToCharArray()); + + caddyStr = caddyStr.Replace("##reverse_proxy##", $"reverse_proxy {prefix}{domain} {{ \n header_up Host {domain} \n }}"); + } + else + { + caddyStr = caddyStr.Replace("##reverse_proxy##", ""); + } + + return caddyStr; + } + + public static string BuildXrayConfig(XraySettings parameters) + { + var xrayConfig = LoadXrayConfig(); + var baseBound = GetBound("VLESS_TCP_XTLS.json"); + baseBound.port = parameters.Port; + baseBound.settings.fallbacks.Add(JToken.FromObject(new + { + dest = FullbackPort + })); + xrayConfig.inbounds.Add(baseBound); + baseBound.settings.clients[0].id = parameters.UUID; + + if (parameters.Types.Contains(XrayType.VLESS_WS)) + { + var wsInbound = GetBound("VLESS_WS.json"); + wsInbound.port = VLESS_WS_Port; + wsInbound.settings.clients[0].id = parameters.UUID; + wsInbound.streamSettings.wsSettings.path = parameters.VLESS_WS_Path; + baseBound.settings.fallbacks.Add(JToken.FromObject(new + { + dest = VLESS_WS_Port, + path = parameters.VLESS_WS_Path, + xver = 1, + })); + xrayConfig.inbounds.Add(JToken.FromObject(wsInbound)); + } + + if (parameters.Types.Contains(XrayType.VLESS_gRPC)) + { + var gRPCInBound = GetBound("VLESS_gRPC.json"); + gRPCInBound.port = parameters.VLESS_gRPC_Port; + gRPCInBound.settings.clients[0].id = parameters.UUID; + gRPCInBound.streamSettings.grpcSettings.serviceName = parameters.VLESS_gRPC_ServiceName; + xrayConfig.inbounds.Add(JToken.FromObject(gRPCInBound)); + } + + if (parameters.Types.Contains(XrayType.VLESS_KCP)) + { + var kcpBound = GetBound("VLESS_KCP.json"); + kcpBound.port = parameters.VLESS_KCP_Port; + kcpBound.settings.clients[0].id = parameters.UUID; + kcpBound.streamSettings.kcpSettings.header.type = parameters.VLESS_KCP_Type; + kcpBound.streamSettings.kcpSettings.seed = parameters.VLESS_KCP_Seed; + xrayConfig.inbounds.Add(JToken.FromObject(kcpBound)); + } + + if (parameters.Types.Contains(XrayType.VMESS_TCP)) + { + var mtcpBound = GetBound("VMESS_TCP.json"); + mtcpBound.port = VMESS_TCP_Port; + mtcpBound.settings.clients[0].id = parameters.UUID; + mtcpBound.streamSettings.tcpSettings.header.request.path = parameters.VMESS_TCP_Path; + baseBound.settings.fallbacks.Add(JToken.FromObject(new + { + dest = VMESS_TCP_Port, + path = parameters.VMESS_TCP_Path, + xver = 1, + })); + xrayConfig.inbounds.Add(JToken.FromObject(mtcpBound)); + } + + if (parameters.Types.Contains(XrayType.VMESS_WS)) + { + var mwsBound = GetBound("VMESS_WS.json"); + mwsBound.port = VMESS_WS_Port; + mwsBound.settings.clients[0].id = parameters.UUID; + mwsBound.streamSettings.wsSettings.path = parameters.VMESS_WS_Path; + baseBound.settings.fallbacks.Add(JToken.FromObject(new + { + dest = VMESS_WS_Port, + path = parameters.VMESS_WS_Path, + xver = 1, + })); + xrayConfig.inbounds.Add(JToken.FromObject(mwsBound)); + } + + if (parameters.Types.Contains(XrayType.VMESS_KCP)) + { + var kcpBound = GetBound("VMESS_KCP.json"); + kcpBound.port = parameters.VMESS_KCP_Port; + kcpBound.settings.clients[0].id = parameters.UUID; + kcpBound.streamSettings.kcpSettings.header.type = parameters.VMESS_KCP_Type; + kcpBound.streamSettings.kcpSettings.seed = parameters.VMESS_KCP_Seed; + xrayConfig.inbounds.Add(JToken.FromObject(kcpBound)); + } + + if (parameters.Types.Contains(XrayType.Trojan_TCP)) + { + var trojanTcpBound = GetBound("Trojan_TCP.json"); + trojanTcpBound.port = Trojan_TCP_Port; + trojanTcpBound.settings.clients[0].password = parameters.TrojanPassword; + trojanTcpBound.settings.fallbacks[0].dest = FullbackPort; + baseBound.settings.fallbacks[0] = JToken.FromObject(new + { + dest = Trojan_TCP_Port, + xver = 1, + }); + xrayConfig.inbounds.Add(JToken.FromObject(trojanTcpBound)); + } + + + if (parameters.Types.Contains(XrayType.ShadowsocksAEAD)) + { + var ssBound = GetBound("Shadowsocks-AEAD.json"); + ssBound.port = parameters.ShadowsocksPort; + ssBound.settings.clients[0].password = parameters.ShadowsocksPassword; + ssBound.settings.clients[0].method = parameters.ShadowsocksMethod; + xrayConfig.inbounds.Add(JToken.FromObject(ssBound)); + } + + return JsonConvert.SerializeObject( + xrayConfig, + Formatting.Indented, + new JsonSerializerSettings() + { + NullValueHandling = NullValueHandling.Ignore + }); + } + + private static dynamic GetBound(string name) + { + return LoadJsonObj(Path.Combine(ServerInboundsDir, name)); + } + + private static dynamic LoadJsonObj(string path) + { + if (File.Exists(path)) + { + var jsonStr = File.ReadAllText(path, Encoding.UTF8); + return JToken.FromObject(JsonConvert.DeserializeObject(jsonStr)); + } + return null; + } + + } + +} diff --git a/ProxySuper.Core/Services/XrayProject.cs b/ProxySuper.Core/Services/XrayProject.cs new file mode 100644 index 0000000..9bfcd1d --- /dev/null +++ b/ProxySuper.Core/Services/XrayProject.cs @@ -0,0 +1,321 @@ +using Newtonsoft.Json; +using ProxySuper.Core.Models; +using Renci.SshNet; +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using System.Windows; + +namespace ProxySuper.Core.Services +{ + public class XrayProject : ProjectBase + { + + private const string ServerLogDir = @"Templates\xray\server\00_log"; + private const string ServerApiDir = @"Templates\xray\server\01_api"; + private const string ServerDnsDir = @"Templates\xray\server\02_dns"; + private const string ServerRoutingDir = @"Templates\xray\server\03_routing"; + private const string ServerPolicyDir = @"Templates\xray\server\04_policy"; + private const string ServerInboundsDir = @"Templates\xray\server\05_inbounds"; + private const string ServerOutboundsDir = @"Templates\xray\server\06_outbounds"; + private const string ServerTransportDir = @"Templates\xray\server\07_transport"; + private const string ServerStatsDir = @"Templates\xray\server\08_stats"; + private const string ServerReverseDir = @"Templates\xray\server\09_reverse"; + private const string CaddyFileDir = @"Templates\xray\caddy"; + + public XrayProject(SshClient sshClient, XraySettings parameters, Action writeOutput) : base(sshClient, parameters, writeOutput) + { + } + + /// + /// 安装Xray + /// + public override void Install() + { + try + { + EnsureRootAuth(); + + if (FileExists("/usr/local/bin/xray")) + { + var btnResult = MessageBox.Show("已经安装Xray,是否需要重装?", "提示", MessageBoxButton.YesNo); + if (btnResult == MessageBoxResult.No) + { + MessageBox.Show("安装终止", "提示"); + return; + } + } + + WriteOutput("检测安装系统环境..."); + EnsureSystemEnv(); + WriteOutput("检测安装系统环境完成"); + + WriteOutput("配置服务器端口..."); + ConfigurePort(); + WriteOutput("端口配置完成"); + + WriteOutput("安装必要的系统工具..."); + ConfigureSoftware(); + WriteOutput("系统工具安装完成"); + + WriteOutput("检测IP6..."); + ConfigureIPv6(); + WriteOutput("检测IP6完成"); + + WriteOutput("配置防火墙..."); + ConfigureFirewall(); + WriteOutput("防火墙配置完成"); + + WriteOutput("同步系统和本地时间..."); + SyncTimeDiff(); + WriteOutput("时间同步完成"); + + WriteOutput("检测域名是否绑定本机IP..."); + ValidateDomain(); + WriteOutput("域名检测完成"); + + WriteOutput("安装Xray-Core..."); + InstallXrayWithCert(); + WriteOutput("Xray-Core安装完成"); + + WriteOutput("安装Caddy..."); + InstallCaddy(); + WriteOutput("Caddy安装完成"); + + WriteOutput("启动BBR"); + EnableBBR(); + + UploadCaddyFile(); + WriteOutput("************"); + WriteOutput("安装完成,尽情享用吧......"); + WriteOutput("************"); + } + catch (Exception ex) + { + var errorLog = "安装终止," + ex.Message; + WriteOutput(errorLog); + MessageBox.Show(errorLog); + } + } + + public void UninstallProxy() + { + EnsureRootAuth(); + WriteOutput("卸载Caddy"); + UninstallCaddy(); + WriteOutput("卸载Xray"); + UninstallXray(); + WriteOutput("卸载证书"); + UninstallAcme(); + WriteOutput("关闭端口"); + ClosePort(Parameters.ShadowsocksPort, Parameters.VMESS_KCP_Port); + + WriteOutput("************ 卸载完成 ************"); + } + + /// + /// 更新xray内核 + /// + public void UpdateXrayCore() + { + EnsureRootAuth(); + EnsureSystemEnv(); + RunCmd("bash -c \"$(curl -L https://github.com/XTLS/Xray-install/raw/main/install-release.sh)\" @ install"); + RunCmd("systemctl restart xray"); + WriteOutput("************ 更新xray内核完成 ************"); + } + + /// + /// 更新xray配置 + /// + public void UpdateXraySettings() + { + EnsureRootAuth(); + EnsureSystemEnv(); + ConfigureFirewall(); + var configJson = XrayConfigBuilder.BuildXrayConfig(Parameters); + var stream = new MemoryStream(Encoding.UTF8.GetBytes(configJson)); + RunCmd("rm -rf /usr/local/etc/xray/config.json"); + UploadFile(stream, "/usr/local/etc/xray/config.json"); + ConfigurePort(); + UploadCaddyFile(string.IsNullOrEmpty(Parameters.MaskDomain)); + RunCmd("systemctl restart xray"); + WriteOutput("************ 更新Xray配置成功,更新配置不包含域名,如果域名更换请重新安装。 ************"); + } + + /// + /// 重装Caddy + /// + public void DoUninstallCaddy() + { + EnsureRootAuth(); + UninstallCaddy(); + WriteOutput("************ 卸载Caddy完成 ************"); + } + + /// + /// 安装证书 + /// + public void InstallCertToXray() + { + EnsureRootAuth(); + EnsureSystemEnv(); + InstallCert( + dirPath: "/usr/local/etc/xray/ssl", + certName: "xray_ssl.crt", + keyName: "xray_ssl.key"); + + RunCmd("systemctl restart xray"); + WriteOutput("************ 安装证书完成 ************"); + } + + /// + /// 上传证书 + /// + /// + /// + public void UploadCert(Stream stream) + { + EnsureRootAuth(); + EnsureSystemEnv(); + + // 转移旧文件 + var oldFileName = $"ssl_{DateTime.Now.Ticks}"; + RunCmd($"mv /usr/local/etc/xray/ssl /usr/local/etc/xray/{oldFileName}"); + + // 上传新文件 + RunCmd("mkdir /usr/local/etc/xray/ssl"); + UploadFile(stream, "/usr/local/etc/xray/ssl/ssl.zip"); + RunCmd("unzip /usr/local/etc/xray/ssl/ssl.zip -d /usr/local/etc/xray/ssl"); + + // 改名 + var crtFiles = RunCmd("find /usr/local/etc/xray/ssl/*.crt").Split("\n".ToCharArray(), StringSplitOptions.RemoveEmptyEntries); + if (crtFiles.Length > 0) + { + RunCmd($"mv {crtFiles[0]} /usr/local/etc/xray/ssl/xray_ssl.crt"); + } + else + { + WriteOutput("************ 上传证书失败,请联系开发者 ************"); + RunCmd("rm -rf /usr/local/etc/xray/ssl"); + RunCmd($"mv /usr/local/etc/xray/ssl{oldFileName} /usr/local/etc/xray/ssl"); + WriteOutput("操作已回滚"); + return; + } + + var keyFiles = RunCmd("find /usr/local/etc/xray/ssl/*.key").Split("\n".ToCharArray(), StringSplitOptions.RemoveEmptyEntries); + if (keyFiles.Length > 0) + { + RunCmd($"mv {keyFiles[0]} /usr/local/etc/xray/ssl/xray_ssl.key"); + } + else + { + WriteOutput("************ 上传证书失败,请联系开发者 ************"); + RunCmd("rm -rf /usr/local/etc/xray/ssl"); + RunCmd($"mv /usr/local/etc/xray/ssl{oldFileName} /usr/local/etc/xray/ssl"); + WriteOutput("操作已回滚"); + return; + } + + RunCmd("systemctl restart xray"); + WriteOutput("************ 上传证书完成 ************"); + } + + /// + /// 上传静态网站 + /// + /// + public void UploadWeb(Stream stream) + { + EnsureRootAuth(); + EnsureSystemEnv(); + if (!FileExists("/usr/share/caddy")) + { + RunCmd("mkdir /usr/share/caddy"); + } + RunCmd("rm -rf /usr/share/caddy/*"); + UploadFile(stream, "/usr/share/caddy/caddy.zip"); + RunCmd("unzip /usr/share/caddy/caddy.zip -d /usr/share/caddy"); + RunCmd("chmod -R 777 /usr/share/caddy"); + UploadCaddyFile(useCustomWeb: true); + WriteOutput("************ 上传网站模板完成 ************"); + } + + + private void UploadCaddyFile(bool useCustomWeb = false) + { + var configJson = XrayConfigBuilder.BuildCaddyConfig(Parameters, useCustomWeb); + var stream = new MemoryStream(Encoding.UTF8.GetBytes(configJson)); + if (FileExists("/etc/caddy/Caddyfile")) + { + RunCmd("mv /etc/caddy/Caddyfile /etc/caddy/Caddyfile.back"); + } + UploadFile(stream, "/etc/caddy/Caddyfile"); + RunCmd("systemctl restart caddy"); + } + + + + private void UninstallXray() + { + RunCmd("bash -c \"$(curl -L https://github.com/XTLS/Xray-install/raw/main/install-release.sh)\" @ remove"); + } + + private void UninstallAcme() + { + RunCmd("acme.sh --uninstall"); + RunCmd("rm -r ~/.acme.sh"); + } + + private void InstallXrayWithCert() + { + RunCmd("bash -c \"$(curl -L https://github.com/XTLS/Xray-install/raw/main/install-release.sh)\" @ install"); + + if (!FileExists("/usr/local/bin/xray")) + { + WriteOutput("Xray-Core安装失败,请联系开发者"); + throw new Exception("Xray-Core安装失败,请联系开发者"); + } + + RunCmd($"sed -i 's/User=nobody/User=root/g' /etc/systemd/system/xray.service"); + RunCmd($"sed -i 's/CapabilityBoundingSet=/#CapabilityBoundingSet=/g' /etc/systemd/system/xray.service"); + RunCmd($"sed -i 's/AmbientCapabilities=/#AmbientCapabilities=/g' /etc/systemd/system/xray.service"); + RunCmd($"systemctl daemon-reload"); + + if (FileExists("/usr/local/etc/xray/config.json")) + { + RunCmd(@"mv /usr/local/etc/xray/config.json /usr/local/etc/xray/config.json.1"); + } + + WriteOutput("安装TLS证书"); + InstallCertToXray(); + WriteOutput("TLS证书安装完成"); + + + var configJson = XrayConfigBuilder.BuildXrayConfig(Parameters); + var stream = new MemoryStream(Encoding.UTF8.GetBytes(configJson)); + UploadFile(stream, "/usr/local/etc/xray/config.json"); + RunCmd("systemctl restart xray"); + } + + private int GetRandomPort() + { + var random = new Random(); + return random.Next(10001, 60000); + } + + private dynamic LoadJsonObj(string path) + { + if (File.Exists(path)) + { + var jsonStr = File.ReadAllText(path, Encoding.UTF8); + return JsonConvert.DeserializeObject(jsonStr); + } + return null; + } + + } +} diff --git a/ProxySuper.Core/ViewModels/MainViewModel.cs b/ProxySuper.Core/ViewModels/MainViewModel.cs index 728263d..3e43785 100644 --- a/ProxySuper.Core/ViewModels/MainViewModel.cs +++ b/ProxySuper.Core/ViewModels/MainViewModel.cs @@ -18,30 +18,6 @@ namespace ProxySuper.Core.ViewModels _navigationService = navigationService; } - private int _count = 1; - public int Count - { - get - { - return _count; - } - set - { - SetProperty(ref _count, value); - } - } - - public IMvxCommand PlusCommand => new MvxCommand(Plus); - - public void Plus() - { - this.Count++; - - if (this.Count >= 2) - { - _navigationService.Navigate(); - } - } } } diff --git a/ProxySuper.Core/packages.config b/ProxySuper.Core/packages.config index 90d5258..294bf72 100644 --- a/ProxySuper.Core/packages.config +++ b/ProxySuper.Core/packages.config @@ -1,5 +1,6 @@  + \ No newline at end of file diff --git a/ProxySuper.WPF/App.xaml b/ProxySuper.WPF/App.xaml index 7a0206e..939ca3d 100644 --- a/ProxySuper.WPF/App.xaml +++ b/ProxySuper.WPF/App.xaml @@ -7,11 +7,8 @@ - - - - - + + diff --git a/ProxySuper.WPF/MainWindow.xaml b/ProxySuper.WPF/MainWindow.xaml index 6a6bc9e..8557ed9 100644 --- a/ProxySuper.WPF/MainWindow.xaml +++ b/ProxySuper.WPF/MainWindow.xaml @@ -1,34 +1,15 @@ - - - - - - - - - - - - - - - - - - + FontSize="14" + Icon="/Resources/ProxySU.ico" + Title="主机列表" Height="600" Width="1000"> + - + - + diff --git a/ProxySuper.WPF/MainWindow.xaml.cs b/ProxySuper.WPF/MainWindow.xaml.cs index f96e441..1421f43 100644 --- a/ProxySuper.WPF/MainWindow.xaml.cs +++ b/ProxySuper.WPF/MainWindow.xaml.cs @@ -19,7 +19,7 @@ namespace ProxySuper.WPF /// /// MainWindow.xaml 的交互逻辑 /// - public partial class MainWindow : MvxMetroWindow + public partial class MainWindow : MvxWindow { public MainWindow() { diff --git a/ProxySuper.WPF/MvxMetroWindow .cs b/ProxySuper.WPF/MvxMetroWindow .cs deleted file mode 100644 index d36c5e2..0000000 --- a/ProxySuper.WPF/MvxMetroWindow .cs +++ /dev/null @@ -1,119 +0,0 @@ -using System; -using System.Windows; -using MahApps.Metro.Controls; -using MvvmCross; -using MvvmCross.Binding.BindingContext; -using MvvmCross.Platforms.Wpf.Views; -using MvvmCross.ViewModels; - -namespace ProxySuper.WPF -{ - public class MvxMetroWindow : MetroWindow, IMvxWindow, IMvxWpfView, IDisposable - { - private IMvxViewModel _viewModel; - private IMvxBindingContext _bindingContext; - private bool _unloaded = false; - - public IMvxViewModel ViewModel - { - get => _viewModel; - set - { - _viewModel = value; - DataContext = value; - BindingContext.DataContext = value; - } - } - - public string Identifier { get; set; } - - public IMvxBindingContext BindingContext - { - get - { - if (_bindingContext != null) - return _bindingContext; - - if (Mvx.IoCProvider != null) - this.CreateBindingContext(); - - return _bindingContext; - } - set => _bindingContext = value; - } - - public MvxMetroWindow() - { - Closed += MvxWindow_Closed; - Unloaded += MvxWindow_Unloaded; - Loaded += MvxWindow_Loaded; - Initialized += MvxWindow_Initialized; - } - - private void MvxWindow_Initialized(object sender, EventArgs e) - { - if (this == Application.Current.MainWindow) - { - (Application.Current as MvvmCross.Platforms.Wpf.Views.MvxApplication).ApplicationInitialized(); - } - } - - private void MvxWindow_Closed(object sender, EventArgs e) => Unload(); - - private void MvxWindow_Unloaded(object sender, RoutedEventArgs e) => Unload(); - - private void MvxWindow_Loaded(object sender, RoutedEventArgs e) - { - ViewModel?.ViewAppearing(); - ViewModel?.ViewAppeared(); - } - - private void Unload() - { - if (!_unloaded) - { - ViewModel?.ViewDisappearing(); - ViewModel?.ViewDisappeared(); - ViewModel?.ViewDestroy(); - _unloaded = true; - } - } - - public void Dispose() - { - Dispose(true); - GC.SuppressFinalize(this); - } - - ~MvxMetroWindow() - { - Dispose(false); - } - - protected virtual void Dispose(bool disposing) - { - if (disposing) - { - Unloaded -= MvxWindow_Unloaded; - Loaded -= MvxWindow_Loaded; - Closed -= MvxWindow_Closed; - } - } - } - - public class MvxWindow : MvxWindow, IMvxWpfView - where TViewModel : class, IMvxViewModel - { - public new TViewModel ViewModel - { - get { return (TViewModel)base.ViewModel; } - set { base.ViewModel = value; } - } - - public MvxFluentBindingDescriptionSet, TViewModel> CreateBindingSet() - { - return this.CreateBindingSet, TViewModel>(); - } - } -} - diff --git a/ProxySuper.WPF/ProxySuper.WPF.csproj b/ProxySuper.WPF/ProxySuper.WPF.csproj index 24ba0cf..81408b4 100644 --- a/ProxySuper.WPF/ProxySuper.WPF.csproj +++ b/ProxySuper.WPF/ProxySuper.WPF.csproj @@ -14,6 +14,8 @@ 4 true true + + AnyCPU @@ -35,15 +37,6 @@ 4 - - ..\packages\ControlzEx.4.4.0\lib\net45\ControlzEx.dll - - - ..\packages\MahApps.Metro.2.4.5\lib\net46\MahApps.Metro.dll - - - ..\packages\Microsoft.Xaml.Behaviors.Wpf.1.1.19\lib\net45\Microsoft.Xaml.Behaviors.dll - ..\packages\MvvmCross.7.1.2\lib\net461\MvvmCross.dll @@ -77,13 +70,9 @@ MSBuild:Compile Designer - MainView.xaml - - SecondView.xaml - MSBuild:Compile Designer @@ -96,11 +85,17 @@ MainWindow.xaml Code - + MSBuild:Compile - - Designer + PreserveNewest + + + MSBuild:Compile + Designer + PreserveNewest + + MSBuild:Compile @@ -137,5 +132,13 @@ ProxySuper.Core + + + PreserveNewest + + + + + \ No newline at end of file diff --git a/ProxySuper.WPF/Resources/Languages/en.xaml b/ProxySuper.WPF/Resources/Languages/en.xaml new file mode 100644 index 0000000..14e42fb --- /dev/null +++ b/ProxySuper.WPF/Resources/Languages/en.xaml @@ -0,0 +1,16 @@ + + + Add Host + Actions + Export Settings + Export Subscribe + + Language + English + 中文 + + Helper + Github + \ No newline at end of file diff --git a/ProxySuper.WPF/Resources/Languages/zh_cn.xaml b/ProxySuper.WPF/Resources/Languages/zh_cn.xaml new file mode 100644 index 0000000..3036a34 --- /dev/null +++ b/ProxySuper.WPF/Resources/Languages/zh_cn.xaml @@ -0,0 +1,17 @@ + + + 添加主机 + 操作 + 导出配置 + 导出配置 + + 语言 + English + 中文 + 帮助 + Github + + + \ No newline at end of file diff --git a/ProxySuper.WPF/Resources/ProxySU.ico b/ProxySuper.WPF/Resources/ProxySU.ico new file mode 100644 index 0000000000000000000000000000000000000000..9d7a128c59c121d3e72c9d81575b05bde4971b2c GIT binary patch literal 16958 zcmeI3cZd{87r^H@@cug$Gw6C4PZ87;b6Nu!K~#*$f&u+QF=1A8F<`=+bIxMUaTRk~ z^O@J2vl!M?SHCw~{q^+pj2mWFclUcsZB18Kzk2z-s+sn9e$#)Ekskg3FVEjKJ)WE% zkLPdQ^pGc7H^v1by;Wo^0Om?CjyuFCOSI;zUM)K6XFQ zOq1i%a-2`9-r&{BjrudWGNVCvV*{G;*l;rc{rh)C`}XZyIG0H(?pXNO4^h*G4dUbD zRoAXv)zYO)Rj*#X)WU@e)uKg3I0?p<1zGg**#R+W!6f z)u&IN5{*MoH1&K^o=NgtXfyEj>sNL7@L`e1k|j&jh7B9k+qZ9(;b-Q|nd;uXduF;{ z|LWB%HDksMwQ18P_5S^P>pu2%?bp<;xefY}qn(>JrXh7}GSIwU$ebLLFI7>JDU z4Fkvty&8bW&6_uyOX;6Hd7{>=S>rQiF0h}ICr^qU1c|$5|85@pDLu9^efo4u{}6$` z?fkD=waUlenM3?=+5_VN&*KN@&Ye?7j~?|7`u~5Shx`rFe_q%|@aajDCP^HyF9Q7w z7A&xaun)!z@EM;E20V}M?eW(!|EKXzd`73}*iDD0PMz|Jf#}-Fzg<2KzT|S(u3eH# z>~`cfFSW*?|Hh3Q5~GQ4FJHcNU7B2IkB3Zh@=wkT({Rr8?!%a$D10T$?=mXPxgynW^H&y zF4R8M2yV1G^-p}KjtB-yw(Vi+|Geh~~-uU_pVzkU1mDe`vi;Puji=s@7%d#N9+0-zd3#Sw7PKN zg6sWo4^r1U*TyDq-@YAiX*c^PChpuxx#orA$B(PamoJ;kfIYeI>C>lvwNt%nsQP0=`UWtRq7C9J*jg1xG zz(3sDty{Ou)SuEne*9SN*s(+GnfQ#)lc&rWc1_Fz$cy+3@jp;IaNvN*^`|fs=0!Q< zLGLtdqgAU`a`r+LsgF+Jwz8= zvv=6GZJX2wpshkj$mWd+1OKo0tJ+FAMSi^!h{Jjm-&1S1f19B z&YfH3&!1nlZQEAG*=jR#9OK|HXBzN@Z6Q~DoN@Ls$cQoK!tch78`Y>$qvV=#YKfEr zaRgj6&Y^+<^*ww+54p&$Kl2=8b?ep@eAGTsQBkUB(V{+Qe((VfY?(F5>0I06IyE3V zVZ3kOzN%TXW-5F3?5cnN{xZh6y+$_cC$)#(3Kc2{&XFTW27R2l(8$^};-M3mQ?6V& zl{s@}In$@MAzpLtX@EK1yLT5H$fjn^nqm{=76a^ejqV2w7$CByHq4nbr_Z>VpOSvi zpg}@Q_3G7Kb!x{=-Ddc&Ql*ODcTe!vxN&2lkvhi6qd|iPKDap}D^;qLg=_5CvG!$D zoc0;RKXC^D7j=+XziHE^J~(|4B;kT{ID@lVwQ8zgzkYH?<#t{{jZO|U^Xu2IFLf)l z(b#VdA3j{}6Ym+TRH>3QKHGu4vB86l+wb$;pW(kojT+WHVrZd4h2*)?rAu4mcDzab z44=doXn_vw0pDXCY%fmh!q7(Fu3bBYt)NpwW66>w#m~__`*q?(!-frI?}LpG9z58O zte!o4%9@dpk^JqElh1(z2g-c>DimjZ@I?NDfBX`_K8Fn(rgG)VCH~vH zcW>Flakj>OjqkUqPpL_wqoWn~*|VDO{P_L~8D`6tO=LmdvY+7+yOB|uGG!!A6e&_f z6)afLsx6_ZR;^k>6S6h(hc@_P_u58I0KFLA$Gmb!O zL$hgd+J>N;eK7mfV#SK75+zDlIc)FVy&@}Y#^@B=gQnWGYx~dyPHYC*hU$Fd{{8#1 z7VDAc?0_#;tXNU*bKa5?AZJkL5X-4?48Rj-9IQWj^l0IieIGt(KMOPSiLLCBLj|l~ zzI=IEpZ$OnsA-@rCAGg3r}G*bIL9L|@ZCTdARe697rsrbYT2@-_%dfurncn##R;6V zz(3zxgd31U7y#b+4%n(8;m56E>%=s4fsGLxsFCW`tCuLh4+HI?Knd^i`v-iuMT-^^ zpH2R9lY<#!PW$%lef%{P1f?<#c;9K(#Kzd~T)cQu&KtXQ>7tr6X(Bm?*h=o9=Ee@p zxw?5ajrQ>Vzal&U6UfE108`THFuX^lRpLWHH@ruL08d&)A@2%l;=}s;*I!gP^843> z-@tkc{9lyNFp*{d{d0!?{duZQZe+uD@CV6z*lcy zU2f|Muf5lqZ+rVcOdsRD{@wJ^-s|6-y?6Nk{}U?#r~J%yj2D3=@7vz%KTL1*Wb|hA MX!L67IYICL0XR=#Gynhq literal 0 HcmV?d00001 diff --git a/ProxySuper.WPF/Views/MainView.xaml b/ProxySuper.WPF/Views/MainView.xaml index f7e0e0d..0f33c95 100644 --- a/ProxySuper.WPF/Views/MainView.xaml +++ b/ProxySuper.WPF/Views/MainView.xaml @@ -3,11 +3,38 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" - xmlns:local="clr-namespace:ProxySuper.WPF.Views" - xmlns:views="clr-namespace:MvvmCross.Platforms.Wpf.Views;assembly=MvvmCross.Platforms.Wpf" - mc:Ignorable="d" Height="450" Width="800"> + xmlns:views="clr-namespace:MvvmCross.Platforms.Wpf.Views;assembly=MvvmCross.Platforms.Wpf" + mc:Ignorable="d"> - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ProxySuper.WPF/Views/MainView.xaml.cs b/ProxySuper.WPF/Views/MainView.xaml.cs index 3269c09..a7328e7 100644 --- a/ProxySuper.WPF/Views/MainView.xaml.cs +++ b/ProxySuper.WPF/Views/MainView.xaml.cs @@ -24,5 +24,10 @@ namespace ProxySuper.WPF.Views { InitializeComponent(); } + + private void LaunchGitHubSite(object sender, RoutedEventArgs e) + { + System.Diagnostics.Process.Start("explorer.exe", "https://github.com/proxysu/ProxySU"); + } } } diff --git a/ProxySuper.WPF/Views/SecondView.xaml b/ProxySuper.WPF/Views/SecondView.xaml deleted file mode 100644 index cb503bd..0000000 --- a/ProxySuper.WPF/Views/SecondView.xaml +++ /dev/null @@ -1,12 +0,0 @@ - - - - - - diff --git a/ProxySuper.WPF/Views/SecondView.xaml.cs b/ProxySuper.WPF/Views/SecondView.xaml.cs deleted file mode 100644 index abfca03..0000000 --- a/ProxySuper.WPF/Views/SecondView.xaml.cs +++ /dev/null @@ -1,28 +0,0 @@ -using MvvmCross.Platforms.Wpf.Views; -using System; -using System.Collections.Generic; -using System.Linq; -using System.Text; -using System.Threading.Tasks; -using System.Windows; -using System.Windows.Controls; -using System.Windows.Data; -using System.Windows.Documents; -using System.Windows.Input; -using System.Windows.Media; -using System.Windows.Media.Imaging; -using System.Windows.Shapes; - -namespace ProxySuper.WPF.Views -{ - /// - /// SecondView.xaml 的交互逻辑 - /// - public partial class SecondView : MvxMetroWindow - { - public SecondView() - { - InitializeComponent(); - } - } -} diff --git a/ProxySuper.WPF/packages.config b/ProxySuper.WPF/packages.config index cbb419c..9383125 100644 --- a/ProxySuper.WPF/packages.config +++ b/ProxySuper.WPF/packages.config @@ -1,8 +1,5 @@  - - -