詳解ASP.NET中Identity的身份驗(yàn)證代碼
本篇內(nèi)容主要講述了實(shí)現(xiàn)基于微軟賬戶的第三方身份驗(yàn)證、實(shí)現(xiàn)雙因子身份驗(yàn)證、 驗(yàn)證碼機(jī)制這3個(gè)內(nèi)容。
實(shí)現(xiàn)基于微軟賬戶的第三方身份驗(yàn)證
在微軟提供的ASP.NET MVC模板代碼中,默認(rèn)添加了微軟、Google、twitter以及Facebook的賬戶登錄代碼(雖然被注釋了),另外針對國內(nèi)的一些社交賬戶提供了相應(yīng)的組件,所有組件都可以通過Nuget包管理器安裝:
從上圖中看到有優(yōu)酷、微信、QQ、微博等組件,其中一些是微軟提供的,一些是其它開發(fā)者提供的。而本文將使用微軟賬戶為例來介紹如何實(shí)現(xiàn)一個(gè)第三方登錄。
注:本章主要代碼參考ASP.NET MVC模板代碼,所以在文章中只列出關(guān)鍵代碼,其余代碼與模板中的完全一致。
組件安裝及Key申請
在開發(fā)之前首先需要通過Nuget安裝Microsoft.Owin.Security.MicrosoftAccount:
另外就是需要去微軟的開發(fā)者中心使用微軟賬戶創(chuàng)建一個(gè)自己的應(yīng)用信息https://apps.dev.microsoft.com/,并保存應(yīng)用的ID以及密鑰用于對身份驗(yàn)證中間件進(jìn)行配置:
創(chuàng)建過程如下:
1. 點(diǎn)擊添加應(yīng)用按鈕,進(jìn)入應(yīng)用程序注冊頁面,填寫應(yīng)用名稱并點(diǎn)擊Create按鈕(注:由于我已經(jīng)有一個(gè)名稱為My Blog的App,所以下面的Test App的創(chuàng)建流程僅僅是用于演示,后續(xù)的身份驗(yàn)證實(shí)際上是使用之前創(chuàng)建的My Blog):
2. 在后續(xù)頁面中點(diǎn)擊生成新密碼來生成密鑰(注:該密碼只顯示一次,需要在彈出框中復(fù)制并保存下來):
3. 添加平臺:點(diǎn)擊添加平臺按鈕,添加一個(gè)Web平臺,并在平臺的重定向Url中填入本地調(diào)試的地址信息(注:一定需要啟動(dòng)HTTPS并且地址后需要添加signin-microsoft,VS可以在項(xiàng)目的屬性中開啟SSL,并設(shè)置SSL的URL):
4. 保存更改。
添加中間件
在上一篇文章中介紹了,第三方賬戶身份驗(yàn)證除了特定賬戶身份驗(yàn)證中間件外,還需要添加一個(gè)消極模式的外部Cookie身份驗(yàn)證中間件,所以首先需要在項(xiàng)目的Startup文件中添加一下代碼:
然后再在該中間件后加入微軟身份驗(yàn)證中間件(注:中間件順序會(huì)影響處理流程,微軟身份驗(yàn)證中間件必須在外部Cookie中間件后),并設(shè)置上面創(chuàng)建的應(yīng)用ID及密鑰:
添加Controller及頁面的功能支持
現(xiàn)在可以說應(yīng)用中已經(jīng)支持微軟的賬戶身份驗(yàn)證了,但是在應(yīng)用中還未提供微軟身份驗(yàn)證的入口,以及登陸后用戶信息的補(bǔ)全等功能。
1. 在頁面上添加驗(yàn)證入口,在Login頁面上加入以下代碼,通過AuthenticationManager來獲取所有的第三方身份驗(yàn)證方式,并生成對應(yīng)鏈接:
2. 在AccountController中添加ExternalLogin Action方法(注:該方法主要目的是調(diào)用AuthenticationManager的Challenge方法來觸發(fā)微軟身份驗(yàn)證中間件的ResponseChallenge方法來完成頁面的跳轉(zhuǎn)):
其中ChallengeResult是一個(gè)自定義的ASP.NET MVC Reuslt類型:
3. 加入第三方驗(yàn)證后的回調(diào)方法ExternalLoginCallback,該回調(diào)方法是獲取第三方身份驗(yàn)證后的用戶信息,然后在本地?cái)?shù)據(jù)庫中查找該用戶,如果存在那么登錄成功,否則需要對該用戶信息進(jìn)行補(bǔ)全。
4. 添加第三方賬戶信息補(bǔ)全頁面及Action方法,其中action方法接收到補(bǔ)全的用戶信息后完成用戶注冊功能,但要注意的是第三方賬戶沒有密碼,僅僅是在AspNetUserLogins表中添加了第三方驗(yàn)證的信息:
運(yùn)行結(jié)果:
1.訪問登錄頁面出現(xiàn)Microsoft的按鈕(注:必須使用HTTPS地址才能正常的使用微軟身份驗(yàn)證):
2. 點(diǎn)擊微軟身份驗(yàn)證按鈕后,跳轉(zhuǎn)到微軟賬戶登錄頁面:
3. 完成登錄后,由于是第一次登錄,所以會(huì)跳轉(zhuǎn)到信息補(bǔ)全頁面:
輸入郵箱后將登錄成功:
數(shù)據(jù)庫中的信息:
上圖中可以看到無密碼,然后在Login表中有一條數(shù)據(jù):
實(shí)現(xiàn)雙因子身份驗(yàn)證
Identity的雙因子身份驗(yàn)證實(shí)際上是Identity的一個(gè)內(nèi)置功能,為什么說是內(nèi)置呢?因?yàn)橹恍枰獙?shí)現(xiàn)信息的發(fā)送(如郵件、短信等),然后再對Identity中的SignInManager進(jìn)行簡單的配置然后添加一些用于發(fā)送、填寫驗(yàn)證碼的頁面就可以完成。所以首先需要完成的就是實(shí)現(xiàn)信息發(fā)送功能。
注:這里信息發(fā)送功能使用將信息寫到硬盤的方式模擬。
1. 實(shí)現(xiàn)信息的發(fā)送:
在ASP.NET MVC默認(rèn)的模板中就為我們創(chuàng)建了如下代碼:
默認(rèn)的郵件及短信發(fā)送器,只不過它沒有實(shí)現(xiàn),僅僅是返回了一個(gè)空值,現(xiàn)在使用寫硬盤的方式將信息寫到硬盤上:
2. 完成UserManager的雙因子驗(yàn)證配置:
三個(gè)關(guān)鍵點(diǎn):1. TokenProvider,它用來生成驗(yàn)證碼。2. 信息格式。3. 信息發(fā)送服務(wù)。
3. 在身份驗(yàn)證管道中加入雙因子驗(yàn)證中間件:
兩個(gè)中間件前者用于處理二次驗(yàn)證,后者用于記住登錄狀態(tài),下次訪問系統(tǒng)時(shí)自動(dòng)登錄。
4. 添加驗(yàn)證碼發(fā)送方式選擇以及驗(yàn)證碼填寫頁面及相應(yīng)的Action方法(代碼略)。
5. 在數(shù)據(jù)庫中將演示用的用戶信息改為啟用二次驗(yàn)證(注:模板代碼中有用于管理個(gè)人信息的功能,此處省略了實(shí)現(xiàn),直接通過修改數(shù)據(jù)數(shù)據(jù)的方式開啟用戶的雙因子驗(yàn)證、添加電話號碼等):
6. 運(yùn)行結(jié)果:
登錄后需要選擇驗(yàn)證碼發(fā)送方式:
選擇后點(diǎn)擊提交按鈕,頁面調(diào)整到驗(yàn)證頁面的同時(shí),指定的文件中生成了需要的驗(yàn)證碼:
填寫驗(yàn)證碼后點(diǎn)擊提交按鈕,則登錄成功:
注:雙因子驗(yàn)證也可以應(yīng)用到第三方賬戶的登錄方式上,雙因子驗(yàn)證只與用戶有關(guān)與身份驗(yàn)證方式無關(guān)。
驗(yàn)證碼機(jī)制
對于雙因子驗(yàn)證來說,它實(shí)際上就是在普通驗(yàn)證或第三方賬戶驗(yàn)證的基礎(chǔ)上增加了驗(yàn)證碼的發(fā)送和驗(yàn)證兩個(gè)環(huán)節(jié),那么對于驗(yàn)證碼這個(gè)主體Identity是如何來維護(hù)的呢?
在上面的介紹中,有一個(gè)環(huán)節(jié)就是需要通過對UserManager進(jìn)行配置以支持雙因子驗(yàn)證的消息發(fā)送、消息生成等等:
根據(jù)這個(gè)代碼看來XXXTokenProvider是專門用來維護(hù)驗(yàn)證碼的,而XXXService是用來發(fā)送的,所以這里將對TokenProvider進(jìn)行說明,了解驗(yàn)證碼是如何維護(hù)的:
上圖是TokenProvider相關(guān)的一個(gè)簡單類圖,從類圖中可以看出TokenProvider實(shí)際上是實(shí)現(xiàn)了一個(gè)名為IUserTokenProvider的接口,該接口中有4個(gè)方法,它們的作用分別是:
● GenerateAsync:根據(jù)UserManager以及User信息來生成一個(gè)令牌(Token)。
● IsValidProviderForUserAsync:判斷這個(gè)Token提供器對這個(gè)用戶是否是有效的(如果使用短信驗(yàn)證,但是該用戶沒有設(shè)置手機(jī)號,那么就是無效的)。
● NotifyAsync:當(dāng)Token生成后調(diào)用該方法通知用戶,如短信或郵件通知。
● ValidateAsync:用于驗(yàn)證Token是否有效。
而TotpSecutityStampBasedTokenProvider是一個(gè)實(shí)現(xiàn)了IUserTokenProvider接口的,通過用戶安全戳生成驗(yàn)證碼的生成器:
從代碼中可以看到該算法是基于rfc6238(TOTP: Time-Based One-Time Password Algorithm,基于時(shí)間的一次性密碼算法) https://tools.ietf.org/html/rfc6238,然后通過用戶的安全戳以及GetUserModifierAsync方法生成特定的信息熵來完成密碼加密,關(guān)于信息熵可參考:https://www.zhihu.com/question/22178202,上面將生成后的令牌執(zhí)行ToString("D6")是將其轉(zhuǎn)換為一個(gè)6位數(shù)字的字符串。
而Token的驗(yàn)證方式和生成差不多都是通過用戶安全戳和信息熵來驗(yàn)證提交的驗(yàn)證碼(它實(shí)際上是一種hash算法):
以上已經(jīng)解釋了最初驗(yàn)證碼的生成和驗(yàn)證的問題,所以對于EmailTokenProvider和PhoneNumberTokenProvider只是對熵的生成、對Provider的有效性(是否存在Email或電話號碼)、通知方式進(jìn)行了修改,下面是PhoneNumberTokenProvider相關(guān)代碼:
小結(jié)
本章主要是使用代碼的形式實(shí)現(xiàn)了ASP.NET中的第三方驗(yàn)證和雙因子驗(yàn)證,文中的代碼都來自ASP.NET MVC的模板,所以文中僅僅是對關(guān)鍵的代碼進(jìn)行了介紹,一些細(xì)節(jié)的內(nèi)容可參考完整代碼。其中第三方驗(yàn)證使用的是微軟賬戶,如果有環(huán)境支持可以嘗試國內(nèi)的微信、QQ等身份驗(yàn)證。
另外在文章最后對驗(yàn)證碼的生成和校驗(yàn)代碼進(jìn)行了分析,知道了它是基于Hash算法的信息加密、驗(yàn)證的機(jī)制來實(shí)現(xiàn)的。
ASP.NET MVC基于Identity提供了非常完善、強(qiáng)大的用戶管理和身份驗(yàn)證功能,除了以上介紹的以外還有賬戶鎖定、注冊郵箱或短信驗(yàn)證功能,基本上已經(jīng)涵蓋了現(xiàn)在開發(fā)常用的功能,但這些功能被一個(gè)模板實(shí)現(xiàn)了,所以ASP.NET強(qiáng)大嗎?
上一篇:C#檢測移動(dòng)硬盤并獲取移動(dòng)硬盤盤符的方法
欄 目:C#教程
下一篇:基于數(shù)據(jù)類型轉(zhuǎn)換(裝箱與拆箱)與常量詳解
本文標(biāo)題:詳解ASP.NET中Identity的身份驗(yàn)證代碼
本文地址:http://mengdiqiu.com.cn/a1/C_jiaocheng/5374.html
您可能感興趣的文章
- 01-10C#通過反射獲取當(dāng)前工程中所有窗體并打開的方法
- 01-10C#實(shí)現(xiàn)Winform中打開網(wǎng)頁頁面的方法
- 01-10C#實(shí)現(xiàn)由四周向中心縮小的窗體退出特效
- 01-10Extjs4如何處理后臺json數(shù)據(jù)中日期和時(shí)間
- 01-10C#中DataGridView常用操作實(shí)例小結(jié)
- 01-10C#編程獲取資源文件中圖片的方法
- 01-10asp.net中XML如何做增刪改查操作
- 01-10C#利用反射技術(shù)實(shí)現(xiàn)去掉按鈕選中時(shí)的邊框效果
- 01-10C#中查找Dictionary中的重復(fù)值的方法
- 01-10C#中TreeView實(shí)現(xiàn)適合兩級節(jié)點(diǎn)的選中節(jié)點(diǎn)方法


閱讀排行
本欄相關(guān)
- 01-10C#通過反射獲取當(dāng)前工程中所有窗體并
- 01-10關(guān)于ASP網(wǎng)頁無法打開的解決方案
- 01-10WinForm限制窗體不能移到屏幕外的方法
- 01-10WinForm繪制圓角的方法
- 01-10C#實(shí)現(xiàn)txt定位指定行完整實(shí)例
- 01-10WinForm實(shí)現(xiàn)仿視頻 器左下角滾動(dòng)新
- 01-10C#停止線程的方法
- 01-10C#實(shí)現(xiàn)清空回收站的方法
- 01-10C#通過重寫Panel改變邊框顏色與寬度的
- 01-10C#實(shí)現(xiàn)讀取注冊表監(jiān)控當(dāng)前操作系統(tǒng)已
隨機(jī)閱讀
- 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
- 04-02jquery與jsp,用jquery
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-10C#中split用法實(shí)例總結(jié)
- 01-10delphi制作wav文件的方法
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 08-05DEDE織夢data目錄下的sessions文件夾有什