在ASP.NET Core 中使用Cookie中間件
http://ASP.NET Core 提供了Cookie中間件來序列化用戶主題到一個加密的Cookie中並且在後來的請求中校驗這個Cookie,再現用戶並且分配到HttpContext對象的User屬性中。如果你想提供自己的登錄方式和用戶數據你可以使用Cookie中間件來實現獨立的功能。
添加和配置
第一步是增加Cookie中間件到你的應用中。首先使用nuget增加Microsoft.AspNetCore.Authentication.Cookies 程序包。然後添加下面的幾行代碼到Startup.cs文件的Configure方法中,且要在app.UseMvc()之前。
app.UseCookieAuthentication(new CookieAuthenticationOptions()
{
AuthenticationScheme = "MyCookieMiddlewareInstance",
LoginPath = new PathString("/Account/Unauthorized/"),
AccessDeniedPath = new PathString("/Account/Forbidden/"),
AutomaticAuthenticate = true,
AutomaticChallenge = true
});
上面的代碼片段配置了一下幾個選項;
- 認證方案:這是一個已知中間件的值,當有多個實例的中間件如果你想限制授權到一個實例時這個選項將會起作用。
- 登錄路徑:這是當用戶試圖訪問資源但未經過身份驗證時,程序將會將請求重定向到這個相對路徑。
- 禁止訪問路徑:當用戶試圖訪問資源時,但未通過該資源的任何授權策略,請求將被重定向到這個相對路徑。
- 自動認證:這個標誌表明中間件應該會在每個請求上進行驗證和重建他創建的序列化主體。
- 自動挑戰:這個標誌標明當中間件認證失敗時應該重定向瀏覽器到登錄路徑或者禁止訪問路徑。
其他選項包括設置中間件所創建的聲明的發行者,中間件存儲的cookie名稱,Cookie的域和cookie上的各種安全屬性。默認情況下Cookie中間件將使用適當的安全選項,設置HTTPONLY避免cookie在客戶端被JavaScript操作。當請求方式為HTTPS時限制Cookie的HTTPS操作。
創建Cookie
創建Cookie保存自己的信息,必須要初始化一個ClaimsPrincipal(類型)來序列化和保存你想保存的用戶信息到Cookie中。每一次的方法調用都會在你的Controller(控制器)中有一個合適的ClaimsPrincipal對象。
await HttpContext.Authentication.SignInAsync("MyCookieMiddlewareInstance", principal);
上面的代碼將會創建一個加密的Cookie並且增加到當前的請求響應中。AuthenticationScheme明確規定在配置期間
退出
退出當前用戶的登錄,刪除登錄的cookie信息,可以在控制器中調用下面的方法。
await HttpContext.Authentication.SignOutAsync("MyCookieMiddlewareInstance");
響應後端的變化
警告
一旦cookie創建就會成為身份單一認證的來源,即使在後臺系統已經不可用,中間件也是不知道的,並且始終保持登錄直到cookie失效。
Cookie認證中間件在他的選項類中提供了一系列的事件,其中 ValidateAsync() 事件可以用來中斷和重寫cookie認證的驗證方法。
考慮到後臺用戶的數據庫中可能會有‘最後的修改時間’這一列,為了在數據庫修改之後你可以廢止當前的Cookie,第一當創建這個Cookie時添加一個最後修改的聲明並包含當前的值,當數據庫中的數據改變時,這個值也同時更新。
實現一個ValidateAsync()的事件重寫你必須寫一個具有如下簽名的方法。
Task ValidateAsync(CookieValidatePrincipalContext context);
http://ASP.NET Core 認證在SecurityStampValidator中實現了這個驗證。下面是一個類似的例子:
public static class LastChangedValidator
{
public static async Task ValidateAsync(CookieValidatePrincipalContext context)
{
// Pull database from registered DI services.
var userRepository = context.HttpContext.RequestServices.GetRequiredService<IUserRepository>();
var userPrincipal = context.Principal;
// Look for the last changed claim.
string lastChanged;
lastChanged = (from c in userPrincipal.Claims
where c.Type == "LastUpdated"
select c.Value).FirstOrDefault();
if (string.IsNullOrEmpty(lastChanged) ||
!userRepository.ValidateLastChanged(userPrincipal, lastChanged))
{
context.RejectPrincipal();
await context.HttpContext.Authentication.SignOutAsync("MyCookieMiddlewareInstance");
}
}
}
這些要在Cookie中間件配置時進行註冊
app.UseCookieAuthentication(options =>
{
options.Events = new CookieAuthenticationEvents
{
// Set other options
OnValidatePrincipal = LastChangedValidator.ValidateAsync
};
});
如果你想非破壞性的更新用戶主體,例如,name更新了,要想以不影響安全的方式你可以調用context.ReplacePrincipal() 並且設置 context.ShouldRenew 為 true。
控制Cookie選項
CookieAuthenticationOptions 配備了各種各樣的配置選項是你能夠很好的調節創建的Cookie。
-
ClaimsIssuer - 被用來在任何中間件創建的屬性之上。(看不懂)
-
CookieDomain - 如果cookie domain被設置為 ** .http://contoso.com** 那麽contoso.com,http://www.contoso.com,staging.contoso.com 等等類似這樣的域名也會被允許。
-
CookieHttpOnly - 這個標誌指示這個 cookie 只會被服務端訪問。默認值是true,修改這個屬性將會開放你的應用造成 Cookie 盜竊,造成跨站腳本的bug。
-
CookiePath - 這個可以用來隔離運行在同一個 host 下的應用。如果你有一個應用運行在/app1 上,並且想限制 cookie 限制僅僅被發送給自己,那麽你應該設置 CookiePath 屬性為 /app1 ;Cookie將會明白只適用於道 /app1 或者他下面的請求。
-
ExpireTimeSpan - 這個 TimeSpan 時間段之後 Cookie 將會過期。
-
SlidingExpiration - 這個標誌標記了如果超過了過期時間的一半後被訪問那麽Cookie將會被重置。新的過期時間將會後移到當前時間加上ExpireTimespan之後。當調用SignInAsync 時可以通過 ** AuthenticationProperties ** 設置絕對的過期時間。通過限制驗證cookie有效的時間,絕對期滿可以提高應用程序的安全性。
持續性Cookie和絕對過期時間
您可能希望通過瀏覽器會話使cookie過期。也許你也想通過絕對過期時間和認證來結束cookie,那麽你可以在登錄認證和創建Cookie時使用HttpContext.Authentication.SignInAsync方法中的AuthenticationProperties參數類實現。AuthenticationProperties類在Microsoft.AspNetCore.Http.Authentication命名空間中。
例如
await HttpContext.Authentication.SignInAsync(
"MyCookieMiddlewareInstance",
principal,
new AuthenticationProperties
{
IsPersistent = true
});
這個代碼片段將會實現創建一個認證和相應的Cookie來實現即時瀏覽器關閉Cookie也能繼續保留。任何在cookie屬性中的過期時間的設置都將會保存下來。如果瀏覽器關閉時Cookie也過期了那麽在重新啟動瀏覽器是Cookie將會別清理。
await HttpContext.Authentication.SignInAsync(
"MyCookieMiddlewareInstance",
principal,
new AuthenticationProperties
{
ExpiresUtc = DateTime.UtcNow.AddMinutes(20)
});
這段代碼將創建一個身份認證和相應的cookie且將持續20分鐘。 任何在Cookie options中配置的動態選項都會被忽略。 ExpiresUtc 和 IsPersistent 這兩個屬性是相互獨立的。
其實上面bb了那麽多,都沒用! 不如來個demo
// 1. 在Startup.cs的Configure方法中加上
app.UseCookieAuthentication(new CookieAuthenticationOptions
{
AuthenticationScheme = "UserAuth", // Cookie 驗證方案名稱,在寫cookie時會用到。
AutomaticAuthenticate = true, // 是否自動啟用驗證,如果不啟用,則即便客服端傳輸了Cookie信息,服務端也不會主動解析。除了明確配置了 [Authorize(ActiveAuthenticationSchemes = "上面的方案名")] 屬性的地方,才會解析,此功能一般用在需要在同一應用中啟用多種驗證方案的時候。比如分Area.
LoginPath = "/User/Index" // 登錄頁
});
// 2. 新建UserController
// 3. 創建一個測試登錄的方法(這裏為了方便測是我用的是get方法,方便傳參請求)
public IActionResult Login(int userId, string userName)
{
WriteUser(userId, userName);
return Content("Write");
}
private async void WriteUser(int userId, string userName)
{
var identity = new ClaimsIdentity("Forms"); // 指定身份認證類型
identity.AddClaim(new Claim(ClaimTypes.Sid, userId.ToString())); // 用戶Id
identity.AddClaim(new Claim(ClaimTypes.Name, userName)); // 用戶名稱
var principal = new ClaimsPrincipal(identity);
await HttpContext.Authentication.SignInAsync("UserAuth", principal, new AuthenticationProperties { IsPersistent = true , ExpiresUtc = DateTime.UtcNow.AddMinutes(20) }); //過期時間20分鐘
}
// 4. 創建一個退出登錄的方法
public async Task<ActionResult> Logout()
{
await HttpContext.Authentication.SignOutAsync("UserAuth"); // Startup.cs中配置的驗證方案名
return RedirectToAction("User", "Index");
}
// 5. 創建一個獲取cookie用戶信息的方法方便調用
private int GetUserId()
{
//var userName = User.Identity.Name; //獲取登錄時存儲的用戶名稱
var userId = User.FindFirst(ClaimTypes.Sid).Value; // 獲取登錄時存儲的Id
if (string.IsNullOrEmpty(userId))
{
return 0;
}
else
{
return int.Parse(userId);
}
}
// 或者寫一個測試Action
public JsonResult CheckLogin()
{
var userName = User.Identity.Name; //獲取登錄時存儲的用戶名稱
var userId = User.FindFirst(ClaimTypes.Sid).Value; // 獲取登錄時存儲的Id
return Json({UserId