/** * Grasscutter Tools * Copyright (C) 2022 jie65535 * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published * by the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . * **/ using System; using System.Linq; using System.Reflection; using System.Text; using System.Threading; using System.Windows.Forms; using GrasscutterTools.OpenCommand; using GrasscutterTools.Properties; using GrasscutterTools.Utils; namespace GrasscutterTools { internal static class Program { private const string TAG = "Program"; static Program() { AppDomain.CurrentDomain.AssemblyResolve += OnResolveAssembly; } private static Assembly OnResolveAssembly(object sender, ResolveEventArgs args) { //var executingAssembly = Assembly.GetExecutingAssembly(); //var assemblyName = new AssemblyName(args.Name); //var path = assemblyName.Name + ".dll"; //if (assemblyName.CultureInfo.Equals(CultureInfo.InvariantCulture) == false) // path = $@"{assemblyName.CultureInfo}\{path}"; //using (var stream = executingAssembly.GetManifestResourceStream(path)) //{ // if (stream == null) return null; // var assemblyRawBytes = new byte[stream.Length]; // stream.Read(assemblyRawBytes, 0, assemblyRawBytes.Length); // return Assembly.Load(assemblyRawBytes); //} // 手工加载嵌入的dll文件 if (new AssemblyName(args.Name).Name == "Newtonsoft.Json") return Assembly.Load(Resources.Newtonsoft_Json); return null; } /// /// 应用程序的主入口点。 /// [STAThread] private static int Main(string[] args) { var result = HandleCommandLine(args); if (result != -1) return result; // 开启高DPI支持 if (Environment.OSVersion.Version.Major >= 6) // 至少需要Vista系统 { HighDPIUtil.SetDpiAwareness(); } Application.EnableVisualStyles(); Application.SetCompatibleTextRenderingDefault(false); //设置应用程序处理异常方式:ThreadException处理 Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException); //处理线程异常 Application.ThreadException += Application_ThreadException; //处理非UI线程异常 AppDomain.CurrentDomain.UnhandledException += CurrentDomain_UnhandledException; Logger.I(TAG, "Program startup"); try { // 初始化语言环境 if (!string.IsNullOrEmpty(Settings.Default.DefaultLanguage)) MultiLanguage.SetDefaultLanguage(Settings.Default.DefaultLanguage); } catch (Exception ex) { Logger.W(TAG, "Loading language error", ex); MessageBox.Show(Resources.SettingLoadError + ex.ToString(), Resources.Error, MessageBoxButtons.OK, MessageBoxIcon.Error); } Application.Run(new Forms.FormMain()); Logger.I(TAG, "Program end."); return 0; } #region - 命令行参数 - /// /// 处理命令行参数并返回处理结果 /// /// 命令行参数 /// 返回-1表示继续启动应用程序。返回其它值表示退出应用并将该值作为返回结果。 private static int HandleCommandLine(string[] args) { var parser = new ToggleParser(args); if (parser.IsEmpty) return -1; try { GuiRedirect.Redirect(); // 是否启动日志 if (parser.HasToggle("debug") || parser.HasToggle("log")) Logger.IsSaveLogs = true; if (parser.HasToggle("v") || parser.HasToggle("version")) { Console.WriteLine("v" + Common.AppVersion.ToString(3)); return 0; } if (parser.HasToggle("h") || parser.HasToggle("help") || parser.HasToggle("?")) { Console.WriteLine("Usages:"); Console.WriteLine(" GcTools.exe -help"); Console.WriteLine(" GcTools.exe -version"); Console.WriteLine(" GcTools.exe -c \"cmd arg\""); Console.WriteLine(" GcTools.exe -c \"cmd1 arg\" && GcTools -c \"cmd2 arg1 arg2\""); Console.WriteLine(" GcTools.exe -host http://127.0.0.1:443 -token 123456 -c \"cmd1 arg1 arg2 | cmd2 | cmd3 arg\""); return 0; } // 服务器地址 var host = parser.GetToggleValueOrDefault("host", Settings.Default.Host); // 服务器令牌 var token = parser.GetToggleValueOrDefault("token", Settings.Default.TokenCache); if (Settings.Default.Host != host || Settings.Default.TokenCache != token) { Settings.Default.Host = host; Settings.Default.TokenCache = token; Settings.Default.Save(); } #if DEBUG Logger.I(TAG, $"Host: {Settings.Default.Host} Token: {Settings.Default.TokenCache}"); #endif // UID //Settings.Default.RemoteUid = decimal.Parse(parser.GetToggleValueOrDefault("uid", Settings.Default.RemoteUid.ToString())); if (!string.IsNullOrEmpty(Settings.Default.Host) && !string.IsNullOrEmpty(Settings.Default.TokenCache)) { Common.OC = new OpenCommandAPI(Settings.Default.Host, Settings.Default.TokenCache); } // 解析并执行命令 var cmd = parser.GetToggleValueOrDefault("c", string.Empty); if (string.IsNullOrEmpty(cmd)) cmd = parser.GetToggleValueOrDefault("command", string.Empty); if (!string.IsNullOrEmpty(cmd)) { return RunCommand(cmd) ? 0 : 1; } } catch (Exception ex) { Logger.E(TAG, "Parse command failed!", ex); } return -1; } /// /// 执行命令 /// /// GC命令,由|分割多条命令 /// 返回是否执行成功 private static bool RunCommand(string commands) { if (Common.OC == null || !Common.OC.CanInvoke) { Console.WriteLine(Resources.RequireOpenCommandTip); Logger.E(TAG, Resources.RequireOpenCommandTip); return false; } try { Common.OC.Ping().Wait(1000); if (Common.OC.CanInvokeMultipleCmd) { var msg = Common.OC.Invoke(FormatCommand(commands)).Result; Console.WriteLine(string.IsNullOrEmpty(msg) ? "OK" : msg); } else { foreach (var cmd in commands.Split('|').Select(FormatCommand)) { var msg = Common.OC.Invoke(cmd).Result; Console.WriteLine(string.IsNullOrEmpty(msg) ? "OK" : msg); } } return true; } catch (Exception ex) { Console.WriteLine(ex); Logger.E(TAG, "RunCommand Error:", ex); return false; } } /// /// 格式化命令 /// (去除收尾空白,替换换行) /// /// 原始输入 /// 格式化后可执行命令 private static string FormatCommand(string raw) { return raw.Trim().Replace("\\r", "\r").Replace("\\n", "\n"); } #endregion - 命令行参数 - #region - 全局异常处理 - private static void Application_ThreadException(object sender, ThreadExceptionEventArgs e) { string str = GetExceptionMsg(e.Exception, e.ToString()); Logger.E(TAG, "Application_ThreadException"); Logger.E(TAG, str); MessageBox.Show(str, Resources.Error, MessageBoxButtons.OK, MessageBoxIcon.Error); } private static void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e) { string str = GetExceptionMsg(e.ExceptionObject as Exception, e.ToString()); Logger.E(TAG, "CurrentDomain_UnhandledException"); Logger.E(TAG, str); MessageBox.Show(str, Resources.Error, MessageBoxButtons.OK, MessageBoxIcon.Error); } /// /// 生成自定义异常消息 /// /// 异常对象 /// 备用异常消息:当ex为null时有效 /// 异常字符串文本 private static string GetExceptionMsg(Exception ex, string backStr) { StringBuilder sb = new StringBuilder(); sb.AppendLine("****************************异常文本****************************"); sb.AppendLine("【出现时间】:" + DateTime.Now.ToString()); if (ex != null) { sb.AppendLine("【异常类型】:" + ex.GetType().Name); sb.AppendLine("【异常信息】:" + ex.Message); #if DEBUG sb.AppendLine("【堆栈调用】:" + ex.StackTrace); #endif } else { sb.AppendLine("【未处理异常】:" + backStr); } sb.AppendLine("***************************************************************"); return sb.ToString(); } #endregion - 全局异常处理 - } }