diff --git a/RevokeMsgPatcher/BusinessException.cs b/RevokeMsgPatcher/BusinessException.cs new file mode 100644 index 0000000..2b865e3 --- /dev/null +++ b/RevokeMsgPatcher/BusinessException.cs @@ -0,0 +1,14 @@ +using System; + +namespace RevokeMsgPatcher +{ + class BusinessException : ApplicationException + { + public string ErrorCode { get; protected set; } + + public BusinessException(string errcode, string message) : base(message) + { + ErrorCode = errcode; + } + } +} diff --git a/RevokeMsgPatcher/FormMain.Designer.cs b/RevokeMsgPatcher/FormMain.Designer.cs index d7bc826..b00b35d 100644 --- a/RevokeMsgPatcher/FormMain.Designer.cs +++ b/RevokeMsgPatcher/FormMain.Designer.cs @@ -116,6 +116,7 @@ // lblUpdatePachJson // this.lblUpdatePachJson.Cursor = System.Windows.Forms.Cursors.Hand; + this.lblUpdatePachJson.ForeColor = System.Drawing.SystemColors.Highlight; this.lblUpdatePachJson.Location = new System.Drawing.Point(166, 89); this.lblUpdatePachJson.Name = "lblUpdatePachJson"; this.lblUpdatePachJson.Size = new System.Drawing.Size(308, 12); diff --git a/RevokeMsgPatcher/FormMain.cs b/RevokeMsgPatcher/FormMain.cs index 183c1cf..41931e4 100644 --- a/RevokeMsgPatcher/FormMain.cs +++ b/RevokeMsgPatcher/FormMain.cs @@ -61,7 +61,7 @@ namespace RevokeMsgPatcher InitModifier(); InitControls(); - ga.RequestPageView("/main", "进入主界面"); + ga.RequestPageView($"/main/{thisVersion}", $"进入{thisVersion}版本主界面"); } private void InitControls() @@ -86,7 +86,9 @@ namespace RevokeMsgPatcher } // 记录点了什么应用的防撤回 - ga.RequestPageView(GetCheckedRadioButtonNameEn() + "/patch", "点击防撤回"); + string enName = GetCheckedRadioButtonNameEn(); // 应用英文名 + string version = modifier.GetVersion(); // 应用版本 + ga.RequestPageView($"{enName}/{version}/patch", "点击防撤回"); EnableAllButton(false); // a.重新初始化编辑器 @@ -96,28 +98,44 @@ namespace RevokeMsgPatcher { modifier.ValidateAndFindModifyInfo(); } + catch (BusinessException ex) + { + ga.RequestPageView($"{enName}/{version}/patch/sha1/ex/{ex.ErrorCode}", ex.Message); + MessageBox.Show(ex.Message); + return; + } + catch (IOException ex) + { + ga.RequestPageView($"{enName}/{version}/patch/sha1/ex/{ex.HResult.ToString("x4")}", ex.Message); + MessageBox.Show(ex.Message + " 请以管理员权限启动本程序,并确认当前应用(微信/QQ/TIM)处于关闭状态。"); + return; + } catch (Exception ex) { - ga.RequestPageView(GetCheckedRadioButtonNameEn() + "/patch/sha1/ex", ex.Message); + ga.RequestPageView($"{enName}/{version}/patch/sha1/ex/{ex.HResult.ToString("x4")}", ex.Message); MessageBox.Show(ex.Message); + return; + } + finally + { EnableAllButton(true); btnRestore.Enabled = modifier.BackupExists(); - return; } // c.打补丁 try { modifier.Patch(); - ga.RequestPageView(GetCheckedRadioButtonNameEn() + "/patch/succ", "防撤回成功"); + ga.RequestPageView($"{enName}/{version}/patch/succ", "防撤回成功"); MessageBox.Show("补丁安装成功!"); - EnableAllButton(true); - btnRestore.Enabled = modifier.BackupExists(); } catch (Exception ex) { Console.WriteLine(ex.Message); - ga.RequestPageView(GetCheckedRadioButtonNameEn() + "/patch/ex", ex.Message); - MessageBox.Show(ex.Message + " 请以管理员权限启动本程序,并确认微信处于关闭状态。"); + ga.RequestPageView($"{enName}/{version}/patch/ex/{ex.HResult.ToString("x4")}", ex.Message); + MessageBox.Show(ex.Message + " 请以管理员权限启动本程序,并确认当前应用(微信/QQ/TIM)处于关闭状态。"); + } + finally + { EnableAllButton(true); btnRestore.Enabled = modifier.BackupExists(); } @@ -183,7 +201,7 @@ namespace RevokeMsgPatcher string json = await GetPathJsonAsync(); if (string.IsNullOrEmpty(json)) { - lblUpdatePachJson.Text = "[ 获取失败 ]"; + lblUpdatePachJson.Text = "[ 获取最新补丁信息失败 ]"; } else @@ -205,7 +223,7 @@ namespace RevokeMsgPatcher else { needUpdate = false; - lblUpdatePachJson.Text = "[ 获取成功 ]"; + lblUpdatePachJson.Text = "[ 获取成功,点击查看更多信息 ]"; } } catch (Exception ex) @@ -259,7 +277,6 @@ namespace RevokeMsgPatcher private void radioButtons_CheckedChanged(object sender, EventArgs e) { - ga.RequestPageView(GetCheckedRadioButtonNameEn() + "/switch", "切换标签页"); EnableAllButton(false); RadioButton radioButton = sender as RadioButton; // 切换使用不同的防撤回对象 @@ -277,6 +294,7 @@ namespace RevokeMsgPatcher } txtPath.Text = modifier.FindInstallPath(); lblVersion.Text = modifier.GetVersion(); + ga.RequestPageView($"{GetCheckedRadioButtonNameEn()}/{lblVersion.Text}/switch", "切换标签页"); EnableAllButton(true); // 显示是否能够备份还原 if (!string.IsNullOrEmpty(txtPath.Text)) diff --git a/RevokeMsgPatcher/Modifier/AppModifier.cs b/RevokeMsgPatcher/Modifier/AppModifier.cs index 3c355ff..827520e 100644 --- a/RevokeMsgPatcher/Modifier/AppModifier.cs +++ b/RevokeMsgPatcher/Modifier/AppModifier.cs @@ -124,17 +124,17 @@ namespace RevokeMsgPatcher.Modifier // 补丁后SHA1匹配上,肯定已经打过补丁 if (matchingSHA1After != null) { - throw new Exception($"你已经安装过此补丁,文件路径:{editor.FilePath}"); + throw new BusinessException("installed", $"你已经安装过此补丁,文件路径:{editor.FilePath}"); } // 全部不匹配,说明不支持 if (matchingSHA1Before == null && matchingSHA1After == null && matchingVersion == null) { - throw new Exception($"不支持此版本:{editor.FileVersion},文件路径:{editor.FilePath}"); + throw new BusinessException("not_support", $"不支持此版本:{editor.FileVersion},文件路径:{editor.FilePath}"); } // SHA1不匹配,版本匹配,可能dll已经被其他补丁程序修改过 if ((matchingSHA1Before == null && matchingSHA1After == null) && matchingVersion != null) { - throw new Exception($"程序支持此版本:{editor.FileVersion}。但是文件校验不通过,请确认是否使用过其他补丁程序。文件路径:{editor.FilePath}"); + throw new BusinessException("maybe_modified", $"程序支持此版本:{editor.FileVersion}。但是文件校验不通过,请确认是否使用过其他补丁程序。文件路径:{editor.FilePath}"); } } } diff --git a/RevokeMsgPatcher/RevokeMsgPatcher.csproj b/RevokeMsgPatcher/RevokeMsgPatcher.csproj index bce1233..7e2e330 100644 --- a/RevokeMsgPatcher/RevokeMsgPatcher.csproj +++ b/RevokeMsgPatcher/RevokeMsgPatcher.csproj @@ -37,6 +37,7 @@ + @@ -49,6 +50,7 @@ + Form @@ -65,6 +67,7 @@ + diff --git a/RevokeMsgPatcher/Utils/Device.cs b/RevokeMsgPatcher/Utils/Device.cs new file mode 100644 index 0000000..be4d155 --- /dev/null +++ b/RevokeMsgPatcher/Utils/Device.cs @@ -0,0 +1,151 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Management; +using System.Net; +using System.Security.Cryptography; +using System.Text; + +namespace RevokeMsgPatcher.Utils +{ + public class Device + { + private static string macID = null; + private static string osVersion = null; + + private static string fingerPrint = null; + + #region PROP, get it only once + + public static string MacID + { + get + { + if (macID == null) + { + macID = ObtainMacID(); + } + return macID; + } + } + + public static string OSVersion + { + get + { + if (osVersion == null) + { + var name = (from x in new ManagementObjectSearcher("SELECT Caption FROM Win32_OperatingSystem").Get().Cast() + select x.GetPropertyValue("Caption")).FirstOrDefault(); + osVersion = name != null ? name.ToString() : "Unknown"; + } + return osVersion; + } + } + #endregion + + /// + /// Calculate GUID + /// + /// GUID + public static string Value() + { + if (fingerPrint == null) + { + fingerPrint = GetHash( + "MAC >> " + MacID + ); + } + return fingerPrint; + } + + private static string GetHash(string s) + { + MD5 sec = new MD5CryptoServiceProvider(); + ASCIIEncoding enc = new ASCIIEncoding(); + byte[] bt = enc.GetBytes(s); + return GetHexString(sec.ComputeHash(bt)); + } + + private static string GetHexString(byte[] bt) + { + string s = string.Empty; + for (int i = 0; i < bt.Length; i++) + { + byte b = bt[i]; + int n, n1, n2; + n = (int)b; + n1 = n & 15; + n2 = (n >> 4) & 15; + if (n2 > 9) + s += ((char)(n2 - 10 + (int)'A')).ToString(); + else + s += n2.ToString(); + if (n1 > 9) + s += ((char)(n1 - 10 + (int)'A')).ToString(); + else + s += n1.ToString(); + if ((i + 1) != bt.Length && (i + 1) % 2 == 0) s += "-"; + } + return s; + } + + + #region Original Device ID Getting Code + + public static string ObtainMacID() + { + return Identifier("Win32_NetworkAdapterConfiguration", "MACAddress", "IPEnabled"); + } + + private static string Identifier(string wmiClass, string wmiProperty, string wmiMustBeTrue) + { + string result = ""; + try + { + ManagementClass mc = new ManagementClass(wmiClass); + ManagementObjectCollection moc = mc.GetInstances(); + foreach (ManagementObject mo in moc) + { + if (mo[wmiMustBeTrue].ToString() == "True") + { + //Only get the first one + if (result == "") + { + result = mo[wmiProperty].ToString(); + break; + } + } + } + } + catch + { + } + return result; + } + + private static string Identifier(string wmiClass, string wmiProperty) + { + string result = ""; + try + { + ManagementClass mc = new ManagementClass(wmiClass); + ManagementObjectCollection moc = mc.GetInstances(); + foreach (ManagementObject mo in moc) + { + //Only get the first one + if (result == "") + { + result = mo[wmiProperty].ToString(); + break; + } + } + } + catch + { + } + return result; + } + #endregion + } +} diff --git a/RevokeMsgPatcher/Utils/GAHelper.cs b/RevokeMsgPatcher/Utils/GAHelper.cs index 2a4fefb..fa8fd79 100644 --- a/RevokeMsgPatcher/Utils/GAHelper.cs +++ b/RevokeMsgPatcher/Utils/GAHelper.cs @@ -25,7 +25,7 @@ namespace RevokeMsgPatcher.Utils // 根据实际使用分析账号设置 private const string tid = "UA-80358493-2"; // GA Tracking ID / Property ID. - private static readonly string cid = Guid.NewGuid().ToString(); // Anonymous Client ID. + private static readonly string cid = Device.Value(); // Anonymous Client ID. // Guid.NewGuid().ToString() // 屏幕分辨率(可选) private static readonly string sr = Screen.PrimaryScreen.Bounds.Width + "x" + Screen.PrimaryScreen.Bounds.Height;