You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

Program.cs 7.9 kB

10 years ago
10 years ago
10 years ago
12 years ago
12 years ago
10 years ago
12 years ago
8 years ago
12 years ago
12 years ago
12 years ago
10 years ago
10 years ago
10 years ago
8 years ago
8 years ago
10 years ago
12 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
12 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. using System;
  2. using System.Diagnostics;
  3. using System.IO;
  4. using System.Threading;
  5. using System.Windows.Forms;
  6. using NLog;
  7. using Microsoft.Win32;
  8. using Shadowsocks.Controller;
  9. using Shadowsocks.Controller.Hotkeys;
  10. using Shadowsocks.Util;
  11. using Shadowsocks.View;
  12. namespace Shadowsocks
  13. {
  14. static class Program
  15. {
  16. private static Logger logger = LogManager.GetCurrentClassLogger();
  17. public static ShadowsocksController MainController { get; private set; }
  18. public static MenuViewController MenuController { get; private set; }
  19. public static string[] Args { get; private set; }
  20. /// <summary>
  21. /// 应用程序的主入口点。
  22. /// </summary>
  23. /// </summary>
  24. [STAThread]
  25. static void Main(string[] args)
  26. {
  27. Directory.SetCurrentDirectory(Application.StartupPath);
  28. // todo: initialize the NLog configuartion
  29. Model.NLogConfig.TouchAndApplyNLogConfig();
  30. // .NET Framework 4.7.2 on Win7 compatibility
  31. System.Net.ServicePointManager.SecurityProtocol |=
  32. System.Net.SecurityProtocolType.Tls | System.Net.SecurityProtocolType.Tls11 | System.Net.SecurityProtocolType.Tls12;
  33. // store args for further use
  34. Args = args;
  35. // Check OS since we are using dual-mode socket
  36. if (!Utils.IsWinVistaOrHigher())
  37. {
  38. MessageBox.Show(I18N.GetString("Unsupported operating system, use Windows Vista at least."),
  39. "Shadowsocks Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
  40. return;
  41. }
  42. // Check .NET Framework version
  43. if (!Utils.IsSupportedRuntimeVersion())
  44. {
  45. if (DialogResult.OK == MessageBox.Show(I18N.GetString("Unsupported .NET Framework, please update to {0} or later.", "4.7.2"),
  46. "Shadowsocks Error", MessageBoxButtons.OKCancel, MessageBoxIcon.Error))
  47. {
  48. //Process.Start("https://www.microsoft.com/download/details.aspx?id=53344"); // 4.6.2
  49. Process.Start("https://dotnet.microsoft.com/download/dotnet-framework/net472");
  50. }
  51. return;
  52. }
  53. Utils.ReleaseMemory(true);
  54. using (Mutex mutex = new Mutex(false, $"Global\\Shadowsocks_{Application.StartupPath.GetHashCode()}"))
  55. {
  56. Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException);
  57. // handle UI exceptions
  58. Application.ThreadException += Application_ThreadException;
  59. // handle non-UI exceptions
  60. AppDomain.CurrentDomain.UnhandledException += CurrentDomain_UnhandledException;
  61. Application.ApplicationExit += Application_ApplicationExit;
  62. SystemEvents.PowerModeChanged += SystemEvents_PowerModeChanged;
  63. Application.EnableVisualStyles();
  64. Application.SetCompatibleTextRenderingDefault(false);
  65. AutoStartup.RegisterForRestart(true);
  66. if (!mutex.WaitOne(0, false))
  67. {
  68. Process[] oldProcesses = Process.GetProcessesByName("Shadowsocks");
  69. if (oldProcesses.Length > 0)
  70. {
  71. Process oldProcess = oldProcesses[0];
  72. }
  73. MessageBox.Show(I18N.GetString("Find Shadowsocks icon in your notify tray.")
  74. + Environment.NewLine
  75. + I18N.GetString("If you want to start multiple Shadowsocks, make a copy in another directory."),
  76. I18N.GetString("Shadowsocks is already running."));
  77. return;
  78. }
  79. Directory.SetCurrentDirectory(Application.StartupPath);
  80. #if DEBUG
  81. // truncate privoxy log file while debugging
  82. string privoxyLogFilename = Utils.GetTempPath("privoxy.log");
  83. if (File.Exists(privoxyLogFilename))
  84. using (new FileStream(privoxyLogFilename, FileMode.Truncate)) { }
  85. #endif
  86. MainController = new ShadowsocksController();
  87. MenuController = new MenuViewController(MainController);
  88. HotKeys.Init(MainController);
  89. MainController.Start();
  90. Application.Run();
  91. }
  92. }
  93. private static int exited = 0;
  94. private static void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e)
  95. {
  96. if (Interlocked.Increment(ref exited) == 1)
  97. {
  98. string errMsg = e.ExceptionObject.ToString();
  99. logger.Error(errMsg);
  100. MessageBox.Show(
  101. $"{I18N.GetString("Unexpected error, shadowsocks will exit. Please report to")} https://github.com/shadowsocks/shadowsocks-windows/issues {Environment.NewLine}{errMsg}",
  102. "Shadowsocks non-UI Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
  103. Application.Exit();
  104. }
  105. }
  106. private static void Application_ThreadException(object sender, ThreadExceptionEventArgs e)
  107. {
  108. if (Interlocked.Increment(ref exited) == 1)
  109. {
  110. string errorMsg = $"Exception Detail: {Environment.NewLine}{e.Exception}";
  111. logger.Error(errorMsg);
  112. MessageBox.Show(
  113. $"{I18N.GetString("Unexpected error, shadowsocks will exit. Please report to")} https://github.com/shadowsocks/shadowsocks-windows/issues {Environment.NewLine}{errorMsg}",
  114. "Shadowsocks UI Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
  115. Application.Exit();
  116. }
  117. }
  118. private static void SystemEvents_PowerModeChanged(object sender, PowerModeChangedEventArgs e)
  119. {
  120. switch (e.Mode)
  121. {
  122. case PowerModes.Resume:
  123. logger.Info("os wake up");
  124. if (MainController != null)
  125. {
  126. System.Threading.Tasks.Task.Factory.StartNew(() =>
  127. {
  128. Thread.Sleep(10 * 1000);
  129. try
  130. {
  131. MainController.Start(false);
  132. logger.Info("controller started");
  133. }
  134. catch (Exception ex)
  135. {
  136. logger.LogUsefulException(ex);
  137. }
  138. });
  139. }
  140. break;
  141. case PowerModes.Suspend:
  142. if (MainController != null)
  143. {
  144. MainController.Stop();
  145. logger.Info("controller stopped");
  146. }
  147. logger.Info("os suspend");
  148. break;
  149. }
  150. }
  151. private static void Application_ApplicationExit(object sender, EventArgs e)
  152. {
  153. // detach static event handlers
  154. Application.ApplicationExit -= Application_ApplicationExit;
  155. SystemEvents.PowerModeChanged -= SystemEvents_PowerModeChanged;
  156. Application.ThreadException -= Application_ThreadException;
  157. HotKeys.Destroy();
  158. if (MainController != null)
  159. {
  160. MainController.Stop();
  161. MainController = null;
  162. }
  163. }
  164. }
  165. }