通用?HTTP?簽名組件的另類實(shí)現(xiàn)方式
1、初衷
開發(fā)中經(jīng)常需要做一些接口的簽名生成和校驗(yàn)工作,最開始的時(shí)候都是每個(gè)接口去按照約定單獨(dú)實(shí)現(xiàn),久而久之就變的非常難維護(hù),因此就琢磨怎么能夠?qū)懥艘粋€(gè)比較通用的簽名生成工具。
2、思路
采用鏈?zhǔn)秸{(diào)用的方式,使得簽名的步驟可以動(dòng)態(tài)拼湊組合。
3、直接看效果
//設(shè)置數(shù)據(jù)源 var signSource = new Dictionary<string, string>() { { "param1", "1" }, { "param3", "3+" }, { "param2", "2" } }; var signer = new HttpSigner(); signer.SetSignData(signSource); //設(shè)置數(shù)據(jù)源并配置規(guī)則 signer.SetSignData(signSource, setting => { //按參數(shù)名排序 //result --> param1 param2 param3 setting.IsOrderByWithKey = false; //是否對(duì)簽名數(shù)據(jù)的參數(shù)值進(jìn)行UrlEncode setting.IsDoUrlEncodeForSourceValue = false; //簽名主體是否包含參數(shù)名 setting.IsSignTextContainKey = true; //簽名主體中參數(shù)和參數(shù)值的連接符(需要啟用IsSignTextContainKey) setting.SignTextKeyValueSeparator = "="; //簽名主體中不同參數(shù)項(xiàng)的連接符 setting.SignTextItemSeparator = "&"; //以上都開啟后 --> param1=1¶m2=2¶m3=3 //編碼 setting.DefaultEncoding = Encoding.UTF8; }); //簽名主體設(shè)置前綴 signer.SetSignData(signSource).SetSignTextPrefix("TestPrefix"); //簽名主體設(shè)置后綴 signer.SetSignData(signSource).SetSignTextSuffix("TestSuffix"); //簽名主體進(jìn)行Base64 signer.SetSignData(signSource).SetSignTextBase64(); //簽名主體進(jìn)行MD5,(方法參數(shù)為簽名結(jié)果是否轉(zhuǎn)小寫) signer.SetSignData(signSource).SetSignTextMD5(bool isToLower = true); //簽名主體進(jìn)行SHA1,(方法參數(shù)為簽名結(jié)果是否轉(zhuǎn)小寫) signer.SetSignData(signSource).SetSignTextSHA1(bool isToLower = true); //獲取簽名結(jié)果 string signString = signer.SetSignData(signSource).GetSignResult(); //組合調(diào)用 string signString = signer.SetSignData(signSource).SetSignTextBase64().SetSignTextMD5().SetSignTextSHA1();
4、代碼實(shí)現(xiàn)
HttpSignItem類
用于保存簽名的參數(shù)集合。
namespace JiuLing.CommonLibs.Security.HttpSign { internal class HttpSignItem { public string Key { get; set; } public string Value { get; set; } public HttpSignItem(string key, string value) { Key = key; Value = value; } } }
HttpSignSetting類
用于簽名的基本配置。
using System.Text; namespace JiuLing.CommonLibs.Security.HttpSign { /// <summary> /// 簽名配置 /// </summary> public class HttpSignSetting { /// <summary> /// 是否按參數(shù)名進(jìn)行排序 /// </summary> public bool IsOrderByWithKey { get; set; } = false; /// <summary> /// 是否對(duì)簽名數(shù)據(jù)的參數(shù)值進(jìn)行UrlEncode /// </summary> public bool IsDoUrlEncodeForSourceValue { get; set; } = false; /// <summary> /// 簽名主體是否包含參數(shù)名 /// </summary> public bool IsSignTextContainKey { get; set; } = true; /// <summary> /// 簽名主體中參數(shù)和參數(shù)值的連接符(需要啟用IsSignTextContainKey) /// </summary> public string SignTextKeyValueSeparator { get; set; } = "="; /// <summary> /// 簽名主體中不同參數(shù)項(xiàng)的連接符 /// </summary> public string SignTextItemSeparator { get; set; } = "&"; /// <summary> /// 編碼 /// </summary> public Encoding DefaultEncoding { get; set; } = Encoding.UTF8; } }
HttpSigner類
簽名組件的具體實(shí)現(xiàn)。
using System; using System.Collections.Generic; using System.Linq; namespace JiuLing.CommonLibs.Security.HttpSign { /// <summary> /// 網(wǎng)絡(luò)請(qǐng)求簽名工具 /// </summary> public class HttpSigner { /// <summary> /// 簽名配置 /// </summary> private readonly HttpSignSetting _setting = new HttpSignSetting(); /// <summary> /// 最終的簽名串 /// </summary> private string _signString; /// <summary> /// 設(shè)置簽名數(shù)據(jù) /// </summary> /// <param name="signSource">待簽名的鍵值對(duì)</param> /// <param name="setting">配置簽名規(guī)則</param> /// <returns></returns> /// <exception cref="ArgumentException"></exception> public HttpSigner SetSignData(Dictionary<string, string> signSource, Action<HttpSignSetting> setting = null) { setting?.Invoke(_setting); if (_setting == null) { throw new ArgumentNullException("無效的簽名配置", "setting"); } if (signSource == null || signSource.Count == 0) { throw new ArgumentException("待簽名數(shù)據(jù)異常", nameof(signSource)); } var signSourceList = new List<HttpSignItem>(signSource.Count); foreach (var item in signSource) { var itemValue = item.Value; if (_setting.IsDoUrlEncodeForSourceValue) { itemValue = System.Web.HttpUtility.UrlEncode(itemValue, _setting.DefaultEncoding); } signSourceList.Add(new HttpSignItem(item.Key, itemValue)); } if (_setting.IsOrderByWithKey) { signSourceList = signSourceList.OrderBy(x => x.Key).ToList(); } if (_setting.IsSignTextContainKey) { _signString = string.Join(_setting.SignTextItemSeparator, signSourceList.Select(x => $"{x.Key}{_setting.SignTextKeyValueSeparator}{x.Value}")); } else { _signString = string.Join(_setting.SignTextItemSeparator, signSourceList.Select(x => x.Value)); } return this; } /// <summary> /// 簽名主體設(shè)置前綴 /// </summary> /// <param name="input">前綴值</param> /// <returns></returns> public HttpSigner SetSignTextPrefix(string input) { _signString = $"{input}{_signString}"; return this; } /// <summary> /// 簽名主體設(shè)置后綴 /// </summary> /// <param name="input">后綴值</param> /// <returns></returns> public HttpSigner SetSignTextSuffix(string input) { _signString = $"{_signString}{input}"; return this; } /// <summary> /// 簽名主體設(shè)置后綴 /// </summary> /// <returns></returns> public HttpSigner SetUrlEncode() { _signString = System.Web.HttpUtility.UrlEncode(_signString, _setting.DefaultEncoding); return this; } /// <summary> /// 簽名主體進(jìn)行Base64 /// </summary> /// <returns></returns> public HttpSigner SetSignTextBase64() { _signString = Base64Utils.GetStringValue(_signString); return this; } /// <summary> /// 簽名主體進(jìn)行MD5 /// </summary> /// <param name="isToLower">簽名結(jié)果是否轉(zhuǎn)小寫</param> /// <returns></returns> public HttpSigner SetSignTextMD5(bool isToLower = true) { if (isToLower) { _signString = MD5Utils.GetStringValueToLower(_signString); } else { _signString = MD5Utils.GetStringValueToUpper(_signString); } return this; } /// <summary> /// 簽名主體進(jìn)行SHA1 /// </summary> /// <param name="isToLower">簽名結(jié)果是否轉(zhuǎn)小寫</param> /// <returns></returns> public HttpSigner SetSignTextSHA1(bool isToLower = true) { if (isToLower) { _signString = SHA1Utils.GetStringValueToLower(_signString); } else { _signString = SHA1Utils.GetStringValueToUpper(_signString); } return this; } /// <summary> /// 獲取簽名結(jié)果 /// </summary> /// <returns></returns> public string GetSignResult() { return _signString; } } }
5、附上倉庫地址
以上代碼包含在我的通用類庫中,可以直接Nuget搜索JiuLing.CommonLibs安裝。
GitHub類庫地址
文章代碼地址
到此這篇關(guān)于通用 HTTP 簽名組件的另類實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)HTTP 簽名組件內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
asp.net GridView和DataList實(shí)現(xiàn)鼠標(biāo)移到行行變色
在設(shè)計(jì)頁面添加了DataList控件后,我在使用DataList綁定數(shù)據(jù)時(shí)是通過單元格來綁定的,因此鼠標(biāo)效果就在源代碼頁面去實(shí)現(xiàn)2009-02-02ASP.net?Core微信平臺(tái)開發(fā)配置Token
這篇文章主要為大家介紹了ASP.net?Core微信平臺(tái)開發(fā)配置Token有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-04-04ASP.NET session.timeout設(shè)置案例詳解
這篇文章主要介紹了ASP.NET session.timeout設(shè)置案例詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08ASP.NET中GridView、DataList、DataGrid三個(gè)數(shù)據(jù)控件foreach遍歷用法示例
這篇文章主要介紹了ASP.NET中GridView、DataList、DataGrid三個(gè)數(shù)據(jù)控件foreach遍歷用法,結(jié)合實(shí)例形式分析了GridView、DataList、DataGrid使用foreach及for語句進(jìn)行數(shù)據(jù)遍歷的具體使用方法,需要的朋友可以參考下2016-08-08EFCore 通過實(shí)體Model生成創(chuàng)建SQL Server數(shù)據(jù)庫表腳本
這篇文章主要介紹了EFCore 通過實(shí)體Model生成創(chuàng)建SQL Server數(shù)據(jù)庫表腳本的示例,幫助大家更好的理解和學(xué)習(xí)使用.net框架,感興趣的朋友可以了解下2021-03-03.Net Core簡(jiǎn)單使用Mvc內(nèi)置的Ioc(續(xù))
怎樣直接獲取Ioc中的實(shí)例對(duì)象,而不是以構(gòu)造函數(shù)的方式進(jìn)行獲取呢?這篇文章繼續(xù)為大家介紹.Net Core簡(jiǎn)單使用Mvc內(nèi)置的Ioc2018-03-03asp.net中執(zhí)行存儲(chǔ)數(shù)據(jù)操作時(shí)數(shù)據(jù)被自動(dòng)截取的一種情況
asp.net中執(zhí)行存儲(chǔ)數(shù)據(jù)操作時(shí)數(shù)據(jù)被自動(dòng)截取的一種情況...2006-09-09