.NET微信小程序用戶數(shù)據(jù)的簽名驗證和解密代碼
微信小程序時下大熱,抱著學(xué)習(xí)的心態(tài)了解了一下,目前沒有搜到完整的.NET用戶數(shù)據(jù)簽名驗證和解密代碼,于是就寫了一點。
簡單使用方法:
1、客戶端調(diào)用wx.getUserInfo方法,服務(wù)端創(chuàng)建WeChatLoginInfo類的實例接收客戶端發(fā)來的數(shù)據(jù);
2、服務(wù)端新建WeChatAppDecrypt類的實例,初始化此類時需傳入appId與AppSecret用于驗證;
3、調(diào)用WeChatAppDecrypt類中的Decrypt方法,傳入步驟1中獲取的WechatLoginInfo實例;
4、得到WechatUserInfo類的實例,其中就是解密好的數(shù)據(jù)。
話不多說,注釋比較詳盡,感興趣的朋友可以參考。
using System; using System.Net; using System.Net.Http; using System.Net.Http.Headers; using Newtonsoft.Json; using System.Security.Cryptography; using System.Text; namespace BroadSky.WeChatAppDecrypt { /// <summary> /// 處理微信小程序用戶數(shù)據(jù)的簽名驗證和解密 /// </summary> public class WeChatAppDecrypt { private string appId; private string appSecret; /// <summary> /// 構(gòu)造函數(shù) /// </summary> /// <param name="appId">應(yīng)用程序的AppId</param> /// <param name="appSecret">應(yīng)用程序的AppSecret</param> public WeChatAppDecrypt(string appId, string appSecret) { this.appId = appId; this.appSecret = appSecret; return; } /// <summary> /// 獲取OpenId和SessionKey的Json數(shù)據(jù)包 /// </summary> /// <param name="code">客戶端發(fā)來的code</param> /// <returns>Json數(shù)據(jù)包</returns> private string GetOpenIdAndSessionKeyString(string code) { string temp = "https://api.weixin.qq.com/sns/jscode2session?" + "appid=" + appId + "&secret=" + appSecret + "&js_code=" + code + "&grant_type=authorization_code"; return GetResponse(temp); } /// <summary> /// 反序列化包含OpenId和SessionKey的Json數(shù)據(jù)包 /// </summary> /// <param name="code">Json數(shù)據(jù)包</param> /// <returns>包含OpenId和SessionKey的類</returns> public OpenIdAndSessionKey DecodeOpenIdAndSessionKey(WechatLoginInfo loginInfo) { OpenIdAndSessionKey oiask = JsonConvert.DeserializeObject<OpenIdAndSessionKey>(GetOpenIdAndSessionKeyString(loginInfo.code)); if (!String.IsNullOrEmpty(oiask.errcode)) return null; return oiask; } /// <summary> /// 根據(jù)微信小程序平臺提供的簽名驗證算法驗證用戶發(fā)來的數(shù)據(jù)是否有效 /// </summary> /// <param name="rawData">公開的用戶資料</param> /// <param name="signature">公開資料攜帶的簽名信息</param> /// <param name="sessionKey">從服務(wù)端獲取的SessionKey</param> /// <returns>True:資料有效,F(xiàn)alse:資料無效</returns> public bool VaildateUserInfo(string rawData, string signature, string sessionKey) { //創(chuàng)建SHA1簽名類 SHA1 sha1 = new SHA1CryptoServiceProvider(); //編碼用于SHA1驗證的源數(shù)據(jù) byte[] source = Encoding.UTF8.GetBytes(rawData + sessionKey); //生成簽名 byte[] target = sha1.ComputeHash(source); //轉(zhuǎn)化為string類型,注意此處轉(zhuǎn)化后是中間帶短橫杠的大寫字母,需要剔除橫杠轉(zhuǎn)小寫字母 string result = BitConverter.ToString(target).Replace("-","").ToLower(); //比對,輸出驗證結(jié)果 return signature == result; } /// <summary> /// 根據(jù)微信小程序平臺提供的簽名驗證算法驗證用戶發(fā)來的數(shù)據(jù)是否有效 /// </summary> /// <param name="loginInfo">登陸信息</param> /// <param name="sessionKey">從服務(wù)端獲取的SessionKey</param> /// <returns>True:資料有效,F(xiàn)alse:資料無效</returns> public bool VaildateUserInfo(WechatLoginInfo loginInfo, string sessionKey) { return VaildateUserInfo(loginInfo.rawData, loginInfo.signature, sessionKey); } /// <summary> /// 根據(jù)微信小程序平臺提供的簽名驗證算法驗證用戶發(fā)來的數(shù)據(jù)是否有效 /// </summary> /// <param name="loginInfo">登陸信息</param> /// <param name="idAndKey">包含OpenId和SessionKey的類</param> /// <returns>True:資料有效,F(xiàn)alse:資料無效</returns> public bool VaildateUserInfo(WechatLoginInfo loginInfo, OpenIdAndSessionKey idAndKey) { return VaildateUserInfo(loginInfo, idAndKey.session_key); } /// <summary> /// 根據(jù)微信小程序平臺提供的解密算法解密數(shù)據(jù) /// </summary> /// <param name="encryptedData">加密數(shù)據(jù)</param> /// <param name="iv">初始向量</param> /// <param name="sessionKey">從服務(wù)端獲取的SessionKey</param> /// <returns></returns> public WechatUserInfo Decrypt(string encryptedData, string iv, string sessionKey) { WechatUserInfo userInfo; //創(chuàng)建解密器生成工具實例 AesCryptoServiceProvider aes = new AesCryptoServiceProvider(); //設(shè)置解密器參數(shù) aes.Mode = CipherMode.CBC; aes.BlockSize = 128; aes.Padding = PaddingMode.PKCS7; //格式化待處理字符串 byte[] byte_encryptedData = Convert.FromBase64String(encryptedData); byte[] byte_iv = Convert.FromBase64String(iv); byte[] byte_sessionKey = Convert.FromBase64String(sessionKey); aes.IV = byte_iv; aes.Key = byte_sessionKey; //根據(jù)設(shè)置好的數(shù)據(jù)生成解密器實例 ICryptoTransform transform = aes.CreateDecryptor(); //解密 byte [] final = transform.TransformFinalBlock(byte_encryptedData, 0, byte_encryptedData.Length); //生成結(jié)果 string result = Encoding.UTF8.GetString(final); //反序列化結(jié)果,生成用戶信息實例 userInfo = JsonConvert.DeserializeObject<WechatUserInfo>(result); return userInfo; } /// <summary> /// 根據(jù)微信小程序平臺提供的解密算法解密數(shù)據(jù),推薦直接使用此方法 /// </summary> /// <param name="loginInfo">登陸信息</param> /// <returns>用戶信息</returns> public WechatUserInfo Decrypt(WechatLoginInfo loginInfo) { if (loginInfo == null) return null; if (String.IsNullOrEmpty(loginInfo.code)) return null; OpenIdAndSessionKey oiask = DecodeOpenIdAndSessionKey(loginInfo); if (oiask == null) return null; if (!VaildateUserInfo(loginInfo, oiask)) return null; WechatUserInfo userInfo = Decrypt(loginInfo.encryptedData, loginInfo.iv, oiask.session_key); return userInfo; } /// <summary> /// GET請求 /// </summary> /// <param name="url"></param> /// <returns></returns> private string GetResponse(string url) { if (url.StartsWith("https")) System.Net.ServicePointManager.SecurityProtocol = SecurityProtocolType.Tls; HttpClient httpClient = new HttpClient(); httpClient.DefaultRequestHeaders.Accept.Add( new MediaTypeWithQualityHeaderValue("application/json")); HttpResponseMessage response = httpClient.GetAsync(url).Result; if (response.IsSuccessStatusCode) { string result = response.Content.ReadAsStringAsync().Result; return result; } return null; } } /// <summary> /// 微信小程序登錄信息結(jié)構(gòu) /// </summary> public class WechatLoginInfo { public string code { get; set; } public string encryptedData { get; set; } public string iv { get; set; } public string rawData { get; set; } public string signature { get; set; } } /// <summary> /// 微信小程序用戶信息結(jié)構(gòu) /// </summary> public class WechatUserInfo { public string openId { get; set; } public string nickName { get; set; } public string gender { get; set; } public string city { get; set; } public string province { get; set; } public string country { get; set; } public string avatarUrl { get; set; } public string unionId { get; set; } public Watermark watermark { get; set; } public class Watermark { public string appid { get; set; } public string timestamp { get; set; } } } /// <summary> /// 微信小程序從服務(wù)端獲取的OpenId和SessionKey信息結(jié)構(gòu) /// </summary> public class OpenIdAndSessionKey { public string openid { get; set; } public string session_key { get; set; } public string errcode { get; set; } public string errmsg { get; set; } } }
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
ASP.NET之Response.Cookies.Remove 無法刪除COOKIE的原因
在web開發(fā)中Cookie是必不可少的,.NET自然也有一個強(qiáng)大的Cookie操作類,我們用起來也非常方便,不過在使用中我們會發(fā)現(xiàn)一個事情Response.Cookies.Remove刪除不了Cookie,2013-06-06.net控件dropdownlist動態(tài)綁定數(shù)據(jù)具體過程分解
一、在頁面初始化時候?qū)⒓辖壎ǖ紻ropDownList;二、在頁面初始化的時候向DropDownList添加數(shù)據(jù);三、將DataReader讀取的數(shù)據(jù)動態(tài)綁定到DropDownList等等2013-05-05ASP.NET UserControl 通信的具體實現(xiàn)
下面我就用ASP.NET的UserControl模擬SharePoint UserControl通信,兩者的本質(zhì),思想和實現(xiàn)方式都不變。2013-06-06asp.net gridview的Rowcommand命令中獲取行索引的方法總結(jié)
asp.net gridview的Rowcommand命令中獲取行索引的方法總結(jié),需要的朋友可以參考下。2010-05-05asp.net Execl的添加,更新操作實現(xiàn)代碼
asp.net Execl的添加、修改等實現(xiàn)代碼。2009-02-02淺談ASP.NET Core 中間件詳解及項目實戰(zhàn)
這篇文章主要介紹了淺談ASP.NET Core 中間件詳解及項目實戰(zhàn),具有一定的參考價值,感興趣的小伙伴們可以參考一下。2016-12-12基于.NET中:自動將請求參數(shù)綁定到ASPX、ASHX和MVC的方法(菜鳥必看)
這篇文章的目的就是告訴初學(xué)者如何自動將客戶端用AJAX發(fā)送的參數(shù)自動綁定為強(qiáng)類型的成員屬性或方法參數(shù)2013-04-04