.NET微信小程序用戶數(shù)據(jù)的簽名驗(yàn)證和解密代碼
微信小程序時(shí)下大熱,抱著學(xué)習(xí)的心態(tài)了解了一下,目前沒(méi)有搜到完整的.NET用戶數(shù)據(jù)簽名驗(yàn)證和解密代碼,于是就寫了一點(diǎn)。
簡(jiǎn)單使用方法:
1、客戶端調(diào)用wx.getUserInfo方法,服務(wù)端創(chuàng)建WeChatLoginInfo類的實(shí)例接收客戶端發(fā)來(lái)的數(shù)據(jù);
2、服務(wù)端新建WeChatAppDecrypt類的實(shí)例,初始化此類時(shí)需傳入appId與AppSecret用于驗(yàn)證;
3、調(diào)用WeChatAppDecrypt類中的Decrypt方法,傳入步驟1中獲取的WechatLoginInfo實(shí)例;
4、得到WechatUserInfo類的實(shí)例,其中就是解密好的數(shù)據(jù)。
話不多說(shuō),注釋比較詳盡,感興趣的朋友可以參考。
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ù)的簽名驗(yàn)證和解密
/// </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ā)來(lái)的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ù)微信小程序平臺(tái)提供的簽名驗(yàn)證算法驗(yàn)證用戶發(fā)來(lái)的數(shù)據(jù)是否有效
/// </summary>
/// <param name="rawData">公開的用戶資料</param>
/// <param name="signature">公開資料攜帶的簽名信息</param>
/// <param name="sessionKey">從服務(wù)端獲取的SessionKey</param>
/// <returns>True:資料有效,F(xiàn)alse:資料無(wú)效</returns>
public bool VaildateUserInfo(string rawData, string signature, string sessionKey)
{
//創(chuàng)建SHA1簽名類
SHA1 sha1 = new SHA1CryptoServiceProvider();
//編碼用于SHA1驗(yàn)證的源數(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();
//比對(duì),輸出驗(yàn)證結(jié)果
return signature == result;
}
/// <summary>
/// 根據(jù)微信小程序平臺(tái)提供的簽名驗(yàn)證算法驗(yàn)證用戶發(fā)來(lái)的數(shù)據(jù)是否有效
/// </summary>
/// <param name="loginInfo">登陸信息</param>
/// <param name="sessionKey">從服務(wù)端獲取的SessionKey</param>
/// <returns>True:資料有效,F(xiàn)alse:資料無(wú)效</returns>
public bool VaildateUserInfo(WechatLoginInfo loginInfo, string sessionKey)
{
return VaildateUserInfo(loginInfo.rawData, loginInfo.signature, sessionKey);
}
/// <summary>
/// 根據(jù)微信小程序平臺(tái)提供的簽名驗(yàn)證算法驗(yàn)證用戶發(fā)來(lái)的數(shù)據(jù)是否有效
/// </summary>
/// <param name="loginInfo">登陸信息</param>
/// <param name="idAndKey">包含OpenId和SessionKey的類</param>
/// <returns>True:資料有效,F(xiàn)alse:資料無(wú)效</returns>
public bool VaildateUserInfo(WechatLoginInfo loginInfo, OpenIdAndSessionKey idAndKey)
{
return VaildateUserInfo(loginInfo, idAndKey.session_key);
}
/// <summary>
/// 根據(jù)微信小程序平臺(tái)提供的解密算法解密數(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)建解密器生成工具實(shí)例
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ù)生成解密器實(shí)例
ICryptoTransform transform = aes.CreateDecryptor();
//解密
byte [] final = transform.TransformFinalBlock(byte_encryptedData, 0, byte_encryptedData.Length);
//生成結(jié)果
string result = Encoding.UTF8.GetString(final);
//反序列化結(jié)果,生成用戶信息實(shí)例
userInfo = JsonConvert.DeserializeObject<WechatUserInfo>(result);
return userInfo;
}
/// <summary>
/// 根據(jù)微信小程序平臺(tái)提供的解密算法解密數(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請(qǐng)求
/// </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; }
}
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
ASP.NET中Onclick與OnClientClick遇到的問(wèn)題
本文主要介紹了ASP.NET中Onclick與OnClientClick遇到的問(wèn)題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2025-04-04
ASP.NET頁(yè)面請(qǐng)求超時(shí)時(shí)間設(shè)置多種方法
這篇文章主要為大家詳細(xì)介紹了ASP.NET頁(yè)面請(qǐng)求超時(shí)時(shí)間設(shè)置Server.ScriptTimeOut executionTimeout多種方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-09-09
js獲取Treeview選中的節(jié)點(diǎn)(C#選中CheckBox項(xiàng))
方法網(wǎng)上有很多,試了一下都有瑕疵,于是設(shè)置斷點(diǎn)調(diào)試,各個(gè)屬性查找有用的字段,終于找到,接下來(lái)與大家分享解決方法,需要了解的朋友可以參考下2012-12-12
ASP.NET數(shù)據(jù)綁定之DataList控件實(shí)戰(zhàn)篇
這篇文章主要為大家介紹了ASP.NET數(shù)據(jù)綁定中的DataList控件,DataList控件以表的形式呈現(xiàn)數(shù)據(jù),通過(guò)該控件,您可以使用不同的布局來(lái)顯示數(shù)據(jù)記錄,對(duì)DataList控件感興趣的小伙伴們可以參考一下2016-01-01
Asp.Net Core利用xUnit進(jìn)行主機(jī)級(jí)別的網(wǎng)絡(luò)集成測(cè)試詳解
這篇文章主要給大家介紹了關(guān)于Asp.Net Core利用xUnit進(jìn)行主機(jī)級(jí)別的網(wǎng)絡(luò)集成測(cè)試的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們來(lái)一起看看吧2018-12-12
ASP.NET Core使用SkiaSharp實(shí)現(xiàn)驗(yàn)證碼的示例代碼
本篇文章主要介紹了ASP.NET Core使用SkiaSharp實(shí)現(xiàn)驗(yàn)證碼的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-12-12
導(dǎo)致Asp.Net站點(diǎn)重啟10個(gè)原因小結(jié)分析
Asp.Net站點(diǎn)有時(shí)候會(huì)莫名其妙的重啟,什么原因?qū)е碌膮s不得而知,經(jīng)過(guò)一番折騰后,我總結(jié)了導(dǎo)致Asp.Net站點(diǎn)重啟的10個(gè)原因,需要的朋友可以參考下。2011-08-08
通過(guò)Web Service實(shí)現(xiàn)IP地址查詢功能的示例
下面小編就為大家分享一篇通過(guò)Web Service實(shí)現(xiàn)IP地址查詢功能的示例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2017-12-12
輕量級(jí)ORM框架Dapper應(yīng)用支持操作函數(shù)和事物
這篇文章介紹了Dapper支持操作函數(shù)和事物的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-03-03
.net 動(dòng)態(tài)標(biāo)題實(shí)現(xiàn)方法
.net 實(shí)現(xiàn)動(dòng)態(tài)標(biāo)題方法,需要的朋友可以參考下。2009-11-11

