Files
netch/Netch/Controllers/VMessController.cs
2020-07-05 23:58:20 +08:00

266 lines
9.8 KiB
C#

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.IO;
using System.Threading;
using Netch.Forms;
using Netch.Models;
using Netch.Utils;
using Newtonsoft.Json;
using VMess = Netch.Models.Information.VMess;
namespace Netch.Controllers
{
public class VMessController
{
/// <summary>
/// 进程实例
/// </summary>
public Process Instance;
/// <summary>
/// 当前状态
/// </summary>
public State State = State.Waiting;
/// <summary>
/// 启动
/// </summary>
/// <param name="server">服务器</param>
/// <param name="mode">模式</param>
/// <returns>是否启动成功</returns>
public bool Start(Server server, Mode mode)
{
MainForm.Instance.StatusText(i18N.Translate("Starting V2ray"));
if (!File.Exists("bin\\v2ray.exe") || !File.Exists("bin\\v2ctl.exe"))
{
return false;
}
File.WriteAllText("data\\last.json", JsonConvert.SerializeObject(new VMess.Config
{
inbounds = new List<VMess.Inbounds>
{
new VMess.Inbounds
{
settings = new VMess.InboundSettings(),
port = Global.Settings.Socks5LocalPort,
listen = Global.Settings.LocalAddress
}
},
outbounds = new List<VMess.Outbounds>
{
new VMess.Outbounds
{
settings = new VMess.OutboundSettings
{
vnext = new List<VMess.VNext>
{
new VMess.VNext
{
address = server.Hostname,
port = server.Port,
users = new List<VMess.User>
{
new VMess.User
{
id = server.UserID,
alterId = server.AlterID,
security = server.EncryptMethod
}
}
}
}
},
streamSettings = new VMess.StreamSettings
{
network = server.TransferProtocol,
security = server.TLSSecure ? "tls" : "",
wsSettings = server.TransferProtocol == "ws" ? new VMess.WebSocketSettings
{
path = server.Path == "" ? "/" : server.Path,
headers = new VMess.WSHeaders
{
Host = server.Host == "" ? server.Hostname : server.Host
}
} : null,
tcpSettings = server.FakeType == "http" ? new VMess.TCPSettings
{
header = new VMess.TCPHeaders
{
type = server.FakeType,
request = new VMess.TCPRequest
{
path = server.Path == "" ? "/" : server.Path,
headers = new VMess.TCPRequestHeaders
{
Host = server.Host == "" ? server.Hostname : server.Host
}
}
}
} : null,
kcpSettings = server.TransferProtocol == "kcp" ? new VMess.KCPSettings
{
header = new VMess.TCPHeaders
{
type = server.FakeType
}
} : null,
quicSettings = server.TransferProtocol == "quic" ? new VMess.QUICSettings
{
security = server.QUICSecure,
key = server.QUICSecret,
header = new VMess.TCPHeaders
{
type = server.FakeType
}
} : null,
httpSettings = server.TransferProtocol == "h2" ? new VMess.HTTPSettings
{
host = server.Host == "" ? server.Hostname : server.Host,
path = server.Path == "" ? "/" : server.Path
} : null,
tlsSettings = new VMess.TLSSettings
{
allowInsecure = true,
serverName = server.Host == "" ? server.Hostname : server.Host
}
},
mux = new VMess.OutboundMux
{
enabled = server.UseMux
}
},
(mode.Type==0||mode.Type==1||mode.Type==2) ? new VMess.Outbounds
{
tag = "TUNTAP",
protocol = "freedom"
}: new VMess.Outbounds
{
tag = "direct",
protocol = "freedom"
}
},
routing = new VMess.Routing
{
rules = new List<VMess.RoutingRules>
{
mode.BypassChina ? new VMess.RoutingRules
{
type = "field",
ip = new List<string>
{
"geoip:cn",
"geoip:private"
},
domain = new List<string>
{
"geosite:cn"
},
outboundTag = "direct"
} : new VMess.RoutingRules
{
type = "field",
ip = new List<string>
{
"geoip:private"
},
outboundTag = "direct"
}
}
}
}));
// 清理上一次的日志文件,防止淤积占用磁盘空间
if (Directory.Exists("logging"))
{
if (File.Exists("logging\\v2ray.log"))
{
File.Delete("logging\\v2ray.log");
}
}
Instance = MainController.GetProcess();
Instance.StartInfo.FileName = "bin\\v2ray.exe";
Instance.StartInfo.Arguments = "-config ..\\data\\last.json";
Instance.OutputDataReceived += OnOutputDataReceived;
Instance.ErrorDataReceived += OnOutputDataReceived;
State = State.Starting;
Instance.Start();
Instance.BeginOutputReadLine();
Instance.BeginErrorReadLine();
for (var i = 0; i < 1000; i++)
{
Thread.Sleep(10);
if (State == State.Started)
{
if (File.Exists("data\\last.json"))
{
File.Delete("data\\last.json");
}
return true;
}
if (State == State.Stopped)
{
Logging.Info("V2Ray 进程启动失败");
Stop();
return false;
}
}
Logging.Info("V2Ray 进程启动超时");
Stop();
return false;
}
/// <summary>
/// 停止
/// </summary>
public void Stop()
{
try
{
if (Instance != null && !Instance.HasExited)
{
Instance.Kill();
Instance.WaitForExit();
}
}
catch (Exception e)
{
Logging.Info(e.ToString());
}
}
public void OnOutputDataReceived(object sender, DataReceivedEventArgs e)
{
if (!string.IsNullOrWhiteSpace(e.Data))
{
File.AppendAllText("logging\\v2ray.log", $"{e.Data}\r\n");
if (State == State.Starting)
{
if (Instance.HasExited)
{
State = State.Stopped;
}
else if (e.Data.Contains("started"))
{
State = State.Started;
}
else if (e.Data.Contains("config file not readable") || e.Data.Contains("failed to"))
{
State = State.Stopped;
}
}
}
}
}
}