mirror of
https://github.com/bolucat/Archive.git
synced 2025-10-05 08:08:03 +08:00
Update On Tue Dec 31 19:31:58 CET 2024
This commit is contained in:
235
shadowsocks-windows/shadowsocks-csharp/Program.cs
Executable file
235
shadowsocks-windows/shadowsocks-csharp/Program.cs
Executable file
@@ -0,0 +1,235 @@
|
||||
using System;
|
||||
using System.Diagnostics;
|
||||
using System.IO;
|
||||
using System.IO.Pipes;
|
||||
using System.Net;
|
||||
using System.Reflection;
|
||||
using System.Text;
|
||||
using System.Threading;
|
||||
using System.Threading.Tasks;
|
||||
using System.Windows.Forms;
|
||||
using CommandLine;
|
||||
using Microsoft.Win32;
|
||||
using NLog;
|
||||
using ReactiveUI;
|
||||
using Shadowsocks.Controller;
|
||||
using Shadowsocks.Controller.Hotkeys;
|
||||
using Shadowsocks.Util;
|
||||
using Shadowsocks.View;
|
||||
using Splat;
|
||||
using WPFLocalizeExtension.Engine;
|
||||
|
||||
namespace Shadowsocks
|
||||
{
|
||||
internal static class Program
|
||||
{
|
||||
private static readonly Logger logger = LogManager.GetCurrentClassLogger();
|
||||
|
||||
public static ShadowsocksController MainController { get; private set; }
|
||||
public static MenuViewController MenuController { get; private set; }
|
||||
public static CommandLineOption Options { get; private set; }
|
||||
public static string[] Args { get; private set; }
|
||||
|
||||
// https://github.com/dotnet/runtime/issues/13051#issuecomment-510267727
|
||||
public static readonly string ExecutablePath = Process.GetCurrentProcess().MainModule?.FileName;
|
||||
public static readonly string WorkingDirectory = Path.GetDirectoryName(ExecutablePath);
|
||||
|
||||
private static readonly Mutex mutex = new Mutex(true, $"Shadowsocks_{ExecutablePath.GetHashCode()}");
|
||||
|
||||
/// <summary>
|
||||
/// 应用程序的主入口点。
|
||||
/// </summary>
|
||||
/// </summary>
|
||||
[STAThread]
|
||||
private static void Main(string[] args)
|
||||
{
|
||||
#region Single Instance and IPC
|
||||
bool hasAnotherInstance = !mutex.WaitOne(TimeSpan.Zero, true);
|
||||
|
||||
// store args for further use
|
||||
Args = args;
|
||||
Parser.Default.ParseArguments<CommandLineOption>(args)
|
||||
.WithParsed(opt => Options = opt)
|
||||
.WithNotParsed(e => e.Output());
|
||||
|
||||
if (hasAnotherInstance)
|
||||
{
|
||||
if (!string.IsNullOrWhiteSpace(Options.OpenUrl))
|
||||
{
|
||||
IPCService.RequestOpenUrl(Options.OpenUrl);
|
||||
}
|
||||
else
|
||||
{
|
||||
MessageBox.Show(I18N.GetString("Find Shadowsocks icon in your notify tray.")
|
||||
+ Environment.NewLine
|
||||
+ I18N.GetString("If you want to start multiple Shadowsocks, make a copy in another directory."),
|
||||
I18N.GetString("Shadowsocks is already running."));
|
||||
}
|
||||
return;
|
||||
}
|
||||
#endregion
|
||||
|
||||
#region Enviroment Setup
|
||||
Directory.SetCurrentDirectory(WorkingDirectory);
|
||||
// todo: initialize the NLog configuartion
|
||||
Model.NLogConfig.TouchAndApplyNLogConfig();
|
||||
|
||||
// .NET Framework 4.7.2 on Win7 compatibility
|
||||
ServicePointManager.SecurityProtocol |=
|
||||
SecurityProtocolType.Tls | SecurityProtocolType.Tls11 | SecurityProtocolType.Tls12;
|
||||
#endregion
|
||||
|
||||
#region Compactibility Check
|
||||
// Check OS since we are using dual-mode socket
|
||||
if (!Utils.IsWinVistaOrHigher())
|
||||
{
|
||||
MessageBox.Show(I18N.GetString("Unsupported operating system, use Windows Vista at least."),
|
||||
"Shadowsocks Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
|
||||
return;
|
||||
}
|
||||
|
||||
// Check .NET Framework version
|
||||
if (!Utils.IsSupportedRuntimeVersion())
|
||||
{
|
||||
if (DialogResult.OK == MessageBox.Show(I18N.GetString("Unsupported .NET Framework, please update to {0} or later.", "4.8"),
|
||||
"Shadowsocks Error", MessageBoxButtons.OKCancel, MessageBoxIcon.Error))
|
||||
{
|
||||
Process.Start("https://dotnet.microsoft.com/download/dotnet-framework/net48");
|
||||
}
|
||||
return;
|
||||
}
|
||||
#endregion
|
||||
|
||||
#region Event Handlers Setup
|
||||
Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException);
|
||||
// handle UI exceptions
|
||||
Application.ThreadException += Application_ThreadException;
|
||||
// handle non-UI exceptions
|
||||
AppDomain.CurrentDomain.UnhandledException += CurrentDomain_UnhandledException;
|
||||
Application.ApplicationExit += Application_ApplicationExit;
|
||||
SystemEvents.PowerModeChanged += SystemEvents_PowerModeChanged;
|
||||
Application.EnableVisualStyles();
|
||||
Application.SetCompatibleTextRenderingDefault(false);
|
||||
AutoStartup.RegisterForRestart(true);
|
||||
#endregion
|
||||
|
||||
// We would use this in v5.
|
||||
// Parameters would have to be dropped from views' constructors (VersionUpdatePromptView)
|
||||
//Locator.CurrentMutable.RegisterViewsForViewModels(Assembly.GetCallingAssembly());
|
||||
|
||||
// Workaround for hosting WPF controls in a WinForms app.
|
||||
// We have to manually set the culture for the LocalizeDictionary instance.
|
||||
// https://stackoverflow.com/questions/374518/localizing-a-winforms-application-with-embedded-wpf-user-controls
|
||||
// https://stackoverflow.com/questions/14668640/wpf-localize-extension-translate-window-at-run-time
|
||||
LocalizeDictionary.Instance.Culture = Thread.CurrentThread.CurrentCulture;
|
||||
|
||||
#if DEBUG
|
||||
// truncate privoxy log file while debugging
|
||||
string privoxyLogFilename = Utils.GetTempPath("privoxy.log");
|
||||
if (File.Exists(privoxyLogFilename))
|
||||
using (new FileStream(privoxyLogFilename, FileMode.Truncate)) { }
|
||||
#endif
|
||||
MainController = new ShadowsocksController();
|
||||
MenuController = new MenuViewController(MainController);
|
||||
|
||||
HotKeys.Init(MainController);
|
||||
MainController.Start();
|
||||
|
||||
// Update online config
|
||||
Task.Run(async () =>
|
||||
{
|
||||
await Task.Delay(10 * 1000);
|
||||
await MainController.UpdateAllOnlineConfig();
|
||||
});
|
||||
|
||||
#region IPC Handler and Arguement Process
|
||||
IPCService ipcService = new IPCService();
|
||||
Task.Run(() => ipcService.RunServer());
|
||||
ipcService.OpenUrlRequested += (_1, e) => MainController.AskAddServerBySSURL(e.Url);
|
||||
|
||||
if (!string.IsNullOrWhiteSpace(Options.OpenUrl))
|
||||
{
|
||||
MainController.AskAddServerBySSURL(Options.OpenUrl);
|
||||
}
|
||||
#endregion
|
||||
|
||||
Application.Run();
|
||||
|
||||
}
|
||||
|
||||
private static int exited = 0;
|
||||
private static void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e)
|
||||
{
|
||||
if (Interlocked.Increment(ref exited) == 1)
|
||||
{
|
||||
string errMsg = e.ExceptionObject.ToString();
|
||||
logger.Error(errMsg);
|
||||
MessageBox.Show(
|
||||
$"{I18N.GetString("Unexpected error, shadowsocks will exit. Please report to")} https://github.com/shadowsocks/shadowsocks-windows/issues {Environment.NewLine}{errMsg}",
|
||||
"Shadowsocks non-UI Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
|
||||
Application.Exit();
|
||||
}
|
||||
}
|
||||
|
||||
private static void Application_ThreadException(object sender, ThreadExceptionEventArgs e)
|
||||
{
|
||||
if (Interlocked.Increment(ref exited) == 1)
|
||||
{
|
||||
string errorMsg = $"Exception Detail: {Environment.NewLine}{e.Exception}";
|
||||
logger.Error(errorMsg);
|
||||
MessageBox.Show(
|
||||
$"{I18N.GetString("Unexpected error, shadowsocks will exit. Please report to")} https://github.com/shadowsocks/shadowsocks-windows/issues {Environment.NewLine}{errorMsg}",
|
||||
"Shadowsocks UI Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
|
||||
Application.Exit();
|
||||
}
|
||||
}
|
||||
|
||||
private static void SystemEvents_PowerModeChanged(object sender, PowerModeChangedEventArgs e)
|
||||
{
|
||||
switch (e.Mode)
|
||||
{
|
||||
case PowerModes.Resume:
|
||||
logger.Info("os wake up");
|
||||
if (MainController != null)
|
||||
{
|
||||
Task.Factory.StartNew(() =>
|
||||
{
|
||||
Thread.Sleep(10 * 1000);
|
||||
try
|
||||
{
|
||||
MainController.Start(true);
|
||||
logger.Info("controller started");
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
logger.LogUsefulException(ex);
|
||||
}
|
||||
});
|
||||
}
|
||||
break;
|
||||
case PowerModes.Suspend:
|
||||
if (MainController != null)
|
||||
{
|
||||
MainController.Stop();
|
||||
logger.Info("controller stopped");
|
||||
}
|
||||
logger.Info("os suspend");
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
private static void Application_ApplicationExit(object sender, EventArgs e)
|
||||
{
|
||||
// detach static event handlers
|
||||
Application.ApplicationExit -= Application_ApplicationExit;
|
||||
SystemEvents.PowerModeChanged -= SystemEvents_PowerModeChanged;
|
||||
Application.ThreadException -= Application_ThreadException;
|
||||
HotKeys.Destroy();
|
||||
if (MainController != null)
|
||||
{
|
||||
MainController.Stop();
|
||||
MainController = null;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user