詳解ASP.NET Core 之 Identity 入門(二)
前言
在 上篇文章 中講了關(guān)于 Identity 需要了解的單詞以及相對應(yīng)的幾個知識點,并且知道了Identity處在整個登入流程中的位置,本篇主要是在 .NET 整個認(rèn)證系統(tǒng)中比較重要的一個環(huán)節(jié),就是 認(rèn)證(Authentication),因為想要把 Identity 講清楚,是繞不過 Authentication 的。
其實 Identity 也是認(rèn)證系統(tǒng)的一個具體使用,大家一定要把 Authentication 和 Identity 當(dāng)作是兩個東西,一旦混淆,你就容易陷入進去。
下面就來說一下 ASP.NET Core 中的認(rèn)證系統(tǒng)是怎么樣一回事。不要怕,其實很簡單,全是干貨~
Getting Started
大家應(yīng)該還記得在上一篇中的奧巴馬先生吧,他現(xiàn)在不住在華盛頓了,他到中國來旅游了,現(xiàn)在住在北京,這幾天聽說西湖風(fēng)景不錯,于是在 12306 定了一張北京到杭州的高鐵票。取到票之后,他向我們展示了一下:
今天是11.11號,奧巴馬很開心,原因你懂的??斓匠霭l(fā)的時間了,于是,拿著票走到了火車站檢票口,剛把身份證和火車票遞給檢票員?!癱ut”,導(dǎo)演喊了一聲。尼瑪原來是在拍電影呢~
導(dǎo)演說:奧巴馬,你演的太爛了,別演了,你來演檢票員吧,讓旁邊小李來演要出行路由的奧巴馬吧。奧巴馬不情愿的說了一聲:“好吧,希望小李能夠受的了你”。
“action”,導(dǎo)演又喊了一聲,故事開始了~
AuthenticationManager
奧巴馬當(dāng)了檢票員以后,特別高興,因為他有權(quán)利了呀,他可以控制別人能不能上車了,說不定還能偷偷放幾個人進去撈點外快呢。
得知了他能干什么以后,他覺得檢票員這個名字簡直太 low 了,很快,他就有了一個新的高大上的名字,叫:認(rèn)證管理員(AuthenticationManager),而且,他覺得他自己應(yīng)該處在整個核心位置,為什么呢?你想想看,那么龐大的一套鐵路載人系統(tǒng),能不能有收入有錢賺,全靠他給不給放人進去,如果一個人都不放進去,另外那一大幫人只能去喝西北風(fēng)了。
到這里,聰明的同學(xué)可能已經(jīng)知道奧巴馬把他自己放在怎么樣一個核心位置了。對,他把自己放到了 HttpContext 里面。怎么樣? 夠核心吧。
這里延伸第一個知識點:AuthenticationManager 所處的位置
有同學(xué)在上面的截圖里面發(fā)現(xiàn)了 public abstract ClaimsPrincipal User { get; set; }, 這不就是我們上一篇中講到的 “ 證件當(dāng)事人 ” ,現(xiàn)在小李扮演的那個角色么? 對,這個 User 就是本文中的小李,被你提前發(fā)現(xiàn)他躲著這里了,嘿嘿。
還有一個知識點,就是 AuthenticationScheme,什么意思呢? 且看
奧巴馬敢把自己放在這么核心的位置也是有他的能力的,怎么講呢? 比如說在檢票的時候,別人遞過來一張身份證和一張火車票,那怎么樣驗證這兩個證件是合法的呢? 以下就是奧巴馬提出的針對兩種證件的驗證方案:
方案1、針對身份證的驗證,可以查看其本人是否和身份證頭像是否一致,年齡是否符合當(dāng)事人具體年齡。
方案2、針對火車票的驗證,可以看車次,時間是否符合發(fā)車目標(biāo),另外可以看車票上的身份號碼是否和身份證一致。
其中,這每一種方案,就對應(yīng)一個 AuthenticationScheme(驗證方案名稱),是不是明白了。
這就是第二個知識點 AuthenticationScheme 很重要。
知道了奧巴馬的職責(zé)后,就很容易的把代碼寫出來了:
public abstract class AuthenticationManager { //AuthenticateContext包含了需要認(rèn)證的上下文,里面就有小李 public abstract Task AuthenticateAsync(AuthenticateContext context); //握手 public abstract Task ChallengeAsync(string authenticationScheme, AuthenticationProperties properties, ChallengeBehavior behavior); //登入 public abstract Task SignInAsync(string authenticationScheme, ClaimsPrincipal principal, AuthenticationProperties properties); //登出 public abstract Task SignOutAsync(string authenticationScheme, AuthenticationProperties properties); }
奧巴馬做為一個檢票員,有一個認(rèn)證方法,AuthenticateAsync() ,注意這是其一個核心功能,其他幾個都可以沒有,但是唯獨不能沒有這個功能,沒有的話他就不能稱之為一個檢票員了。
然后還有一個握手ChallengeAsync,登入SignInAsync和登出SignOutAsync,下面說說筆者對這三個方法的理解吧。
ChallengeAsync:是社區(qū)協(xié)議文件 RFC2167 定義的關(guān)于在HTTP Authentication 過程中的一種關(guān)于握手的一個過程,主要是摘要認(rèn)證(digest authentication)。
是不是有點專業(yè),看不懂,沒事,有通俗版本的。 小李要進站了,這個時候小李問了一下我們的檢票員奧巴馬先生。
- 小李:“你好,檢票員,我可以進站嗎?”
- 檢票員奧巴馬:“要趕火車嗎?可以啊,請出示你的證件?”
- 小李:“好的,這是我的證件,你檢查一下?”
- 檢票員奧巴馬:“嗯,證件沒問題,進去吧”
這樣一個過程就是握手(digest-challenge)或者叫問答的一個過程,明白了 ChallengeAsync 的原理了吧? 是不是很簡單。
SignInAsync,SignOutAsync:個人覺得這兩個不應(yīng)該放在這里,因為并不屬于認(rèn)證的職責(zé),也不屬于協(xié)議規(guī)定的內(nèi)容。但是這兩個方法確實需要抽象,應(yīng)該單獨抽取一個接口存放,至于為什么這樣做,或許是因為以下原因:
1、對登入登出的抽象是和認(rèn)證緊密結(jié)合的,大多數(shù)情況下認(rèn)證資料的保存是需要在SignIn進行的,比如 Cookies Authentication 中間件就在SignIn方法里面做了Cookie的保存。
2、 AuthenticationManager 這個對象是處在 HttpContext
上下文里面的,本著面向抽象和封裝的原則,放到其里面是合適的,這樣能夠很方便的用戶對其調(diào)用。
關(guān)于 AuthenticationManager 已經(jīng)介紹完了,是不是很簡單呢?
IAuthenticationHandler
有些同學(xué)可能會問了,如果 AuthenticationManager 不提供接口的話,只是一個抽象類的話,那如果自定義認(rèn)證方法就必須繼承它,這對于開發(fā)者來說是不友好的,也違背了面向接口編程的理念。嗯,確實是這樣,那么接口來了:
public interface IAuthenticationHandler { void GetDescriptions(DescribeSchemesContext context); Task AuthenticateAsync(AuthenticateContext context); Task ChallengeAsync(ChallengeContext context); Task SignInAsync(SignInContext context); Task SignOutAsync(SignOutContext context); }
這個接口是在 AuthenticationManager 實現(xiàn)類 DefaultAuthenticationManager 中延伸出來的,所以大家不用再去看里面的源碼了,記住以后如果需要重寫認(rèn)證相關(guān)的東西,實現(xiàn)IAuthenticationHandler就可以了。
Authentication 中間件
對 IAuthenticationHandler 的初步實現(xiàn),封裝了 AuthenticationHandler 這個抽象類,把具體的核心功能都交給下游去實現(xiàn)了,下面的CookieAuthentication 中間件核心類 CookieAuthenticationHandler 就是繼承自AuthenticationHandler, 知道這么多就夠了。
CookieAuthentication 中間件
故事還要繼續(xù),奧巴馬在接到小李遞來的身份證和火車票之后,首先拿著火車票在一個二維碼機器上掃描了一下,然后又拿著身份證在一個機器上刷了一下,經(jīng)過核查,發(fā)現(xiàn)都沒有問題。于是拿起印章在上面蓋了一個 “ 驗訖 ”。
這中間都發(fā)生了什么呢?
首先,在二維碼掃描的過程,這個過程二維碼機器會解析你火車票上的二維碼,如果發(fā)現(xiàn)解析失敗,會直接響應(yīng)認(rèn)證失敗。也就是你別想進站了。
如果解析成功,就會得到你這個票據(jù)中的信息了,然后拿到你票據(jù)里面的的當(dāng)事人信息進行驗證是否被列為了鐵路局黑名單中。
如果驗證通過,則會給你頒發(fā)一個識別碼,把符合你身份的一個識別碼寫入到你的火車票中和檢票員旁邊的電腦系統(tǒng)中,即 “ 驗訖 ”。
話說這個驗訖有點高級,它會向你的火車票芯片中寫入一些信息,那么都寫入些什么信息呢? 1、奧巴馬個人的信息。2、驗證途中的一些上下信息。3、使用的驗證方案。
知道了,這些之后,那么就很容易實現(xiàn)這個驗證方法了,對吧? 以下是 CookieAuthentication 中間件中的核心類 CookieAuthenticationHandler 的里面的核心方法HandleAuthenticateAsync(),同樣你可以理解為實現(xiàn)的 IAuthenticationHandler 接口的 AuthenticateAsync:
protected override async Task<AuthenticateResult> HandleAuthenticateAsync() { // 解析二維碼 var result = await EnsureCookieTicket(); if (!result.Succeeded) { return result; } // 從二維碼中拿當(dāng)事人信息進行驗證 var context = new CookieValidatePrincipalContext(Context, result.Ticket, Options); await Options.Events.ValidatePrincipal(context); if (context.Principal == null) { return AuthenticateResult.Fail("No principal."); } if (context.ShouldRenew) { RequestRefresh(result.Ticket); } // 驗訖, 寫入芯片 return AuthenticateResult.Success(new AuthenticationTicket(context.Principal, context.Properties, Options.AuthenticationScheme)); }
HandleSignInAsync
我們故事繼續(xù)……
奧巴馬檢票完成之后,把票就交給了小李,小李拿到票之后,導(dǎo)演又喊了一聲:“ cut ”……
怎么又停了,小李和奧巴馬一肚子的疑惑,導(dǎo)演說:“ 奧巴馬呀,你檢票員演的不錯,還是繼續(xù)扮演你的本職角色吧,演好了中午盒飯給你雙份,小李,你來演檢票員吧 ”。
可以吃兩份盒飯了,奧巴馬聽后心里還是很開心。
“action” 導(dǎo)演喊了一聲……
奧巴馬接過票,向著車站里面的列車停車處走去,走到了列車門口要進去的時候,又出現(xiàn)了一個人,奧巴馬知道,這個人就是做車內(nèi)乘客登記的(ps: 一般情況下,做乘客登記都是在列車行駛的過程中,在這里我們假設(shè)這個做乘客登記的人比較勤快,就在車門口守著),登記完成之后就讓奧巴馬進去了。
那么,登記這個過程中都干了些什么呢?
首先,登記員的手持設(shè)備會解析火車票票里面寫入芯片中的信息,發(fā)現(xiàn)沒有問題,就開始向自己手里面的登記本登記信息了,主要包含車票主人信息,過期時間,審核人等。
這樣整個過程就是 HandleSignInAsync 的一個過程,換成程序術(shù)語就是,組裝 Cookie 登入上下文信息,寫入到 Http 流的 header 中,也就寫入到了客戶端瀏覽器cookie。
至此,整個過程就完了,我們來看一下代碼:
//方法里面的流程,我只列出了核心部分,影響閱讀的全刪了 protected override async Task HandleSignInAsync(SignInContext signin) { // 解析芯片中的信息 var result = await EnsureCookieTicket(); // 組織登入上下文,設(shè)置過期時間等 // 使用 data protected 加密登記本上的信息 var cookieValue = Options.TicketDataFormat.Protect(ticket); // 寫入到瀏覽器header await ApplyHeaders(cookieValue); }
不想深入了解的可以忽略這部分內(nèi)容:
在 HandleSignInAsync 這個函數(shù)的源碼中,其中有一個很巧妙的設(shè)計, 就是 await Options.Events.SignedIn(signedInContext); 這樣一句代碼,干什么用的呢? 而且前后一共調(diào)用了兩次,有同學(xué)知道是為什么嗎? 我準(zhǔn)備在下一篇中給出答案。
還記得前面 HttpContext 中的ClaimsPrincipal User嗎? 就是小李臨時頂替的那個角色,現(xiàn)在有值了,他就是是奧巴馬了。
奧巴馬在座位上坐好之后,經(jīng)過6個小時的路程就從北京到杭州了,不得不佩服中國高鐵的速度呀,在欣賞晚西湖的風(fēng)景后,奧巴馬給我們傳來了一張照片:
至此,CookieAuthentication 中間件的整個工作流程已經(jīng)講完了,故事也結(jié)束了。
以上,就是這兩行代碼背后的故事:
var user = new ClaimsPrincipal(new ClaimsIdentity(new[] { new Claim(ClaimTypes.Name, "奧巴馬") }, CookieAuthenticationDefaults.AuthenticationScheme)); await HttpContext.Authentication.SignInAsync(CookieAuthenticationDefaults.AuthenticationScheme, user);
總結(jié)
在本篇中我們知道了 AuthenticationManager,也知道了 IAuthenticationHandler 并且簡單的介紹了一下 Authentication 中間件和 CookieAuthentication 中間件,其中 CookieAuthentication 中間件是我們以后使用最多的一個中間件了,本篇也對其做了一個詳細(xì)的介紹,我想通過本篇文章在以后使用的過程中應(yīng)該問題不大了。
有同學(xué)可能會問了,講了這么多認(rèn)證的東西它和 Identity 有什么關(guān)系呢? 難道我通篇都在隱藏他和 Identity 的關(guān)系你沒看出來?。。。。真的想知道? 看下一篇吧。
相關(guān)文章
asp.net實現(xiàn)取消頁面表單內(nèi)文本輸入框Enter響應(yīng)的方法
這篇文章主要介紹了asp.net實現(xiàn)取消頁面表單內(nèi)文本輸入框Enter響應(yīng)的方法,結(jié)合實例形式分析了asp.net文本框Enter響應(yīng)的原理與取消Enter響應(yīng)的相關(guān)實現(xiàn)技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-11-11asp.net 獲取TreeView中第一個選中的節(jié)點
今天做的項目中有一個要獲取TreeView中第一個選中的節(jié)點,當(dāng)然子節(jié)點己包含checkbox以前做過,用的時候又不知道怎么做了,花了點時間又寫了一下,記錄下來,以備下次用.2010-03-03Asp.NetCore3.1開源項目升級為.Net6.0的方法實現(xiàn)
自從.Net6.0出來后,一直想之前開發(fā)的項目升級.Net6.0,本文就詳細(xì)的介紹一下如何將Asp.NetCore3.1開源項目升級為.Net6.0,感興趣的小伙伴們可以參考一下2021-12-12.net 中的 StringBuilder 和 TextWriter 區(qū)別詳解
這篇文章主要介紹了.net 中的 StringBuilder 和 TextWriter 區(qū)別詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09ASP.NET web.config中 數(shù)據(jù)庫連接字符串加密解密
本文主要介紹利用aspnet_regiis.exe工具對web.config中connectionStrings節(jié)點進行加密和解密的過程,希望對大家有所幫助。2016-05-05