既然選擇了遠方,便只顧風雨兼程 __ HANS許
在上篇文章,我們講了JWT在ASP.NET Core的實現,基於中間件來實現。這種方式有個缺點,就是所有的URL,要嘛需要驗證,要嘛不需要驗證,沒有辦法各取所需,因為我們某個API與另一個API的驗證方式不一樣。這就引導出“基於自定義策略形式下的驗證了”。
ASP.NET Core 的Authorization實現
我們使用Core自帶的Authorization(認證與授權)來實現。大家可以先看下微軟官方的策略授權
-
微軟官方例子:
1.1 定義策略- internal class MinimumAgeAuthorizeAttribute : AuthorizeAttribute
- {
- const string POLICY_PREFIX = "MinimumAge";
- public MinimumAgeAuthorizeAttribute(int age) => Age = age;
- // Get or set the Age property by manipulating the underlying Policy property
- public int Age
- {
- get
- {
- if (int.TryParse(Policy.Substring(POLICY_PREFIX.Length), out var age))
- {
- return age;
- }
- return default(int);
- }
- set
- {
- Policy = $"{POLICY_PREFIX}{value.ToString()}";
- }
- }
- }
1.2 使用策略
- [MinimumAgeAuthorize(10)]
- public IActionResult RequiresMinimumAge10()
這樣在執行
RequiresMinimumAge10
會先執行MinimumAgeAuthorize
策略,很像MVC的Attribute
特性,
但內部又不像,在這邊就不多做解釋了,微軟的Core官方文檔講的很清楚。大家去看下就清楚了。 -
JWT的自定義策略形式的實現
2.1 了解IAuthorizationRequirement
IAuthorizationRequirement
表示授權要求,用戶可以繼承這個接口,實現自己認證與授權的要求,比如上面的片段代碼,它就繼承該接口,並有個字段Age
,也就是這個策略有年齡的要求,這個要求可以帶到我們后面驗證的方法里面。我們往下看。2.2 繼承
IAuthorizationRequirement
所以我們實現了JwtAuthorizeBaseRequiremente
該接口,並繼承IAuthorizationRequirement
,可以看到我們的要求是一個叫validatePayLoad
的委托函數,委托函數的入參是字典,JWT,字典便是上篇文章說的JWT的負載部分了。而返回參數是bool,便代表我們自定義的策略驗證JWT是否成功。IJwtAuthorizRequiremente
繼承了IAuthorizationRequirement
- public class JwtAuthorizeBaseRequiremente : IJwtAuthorizRequiremente
- {
- protected internal Func<Dictionary<string, string>, JsonWebTokenSetting, bool> validatePayLoad = (a, b) =>
- {
- return true;
- };
- public virtual IJwtAuthorizRequiremente SetValidateFunc(Func<Dictionary<string, string>, JsonWebTokenSetting, bool> func)
- {
- this.validatePayLoad = func ?? this.validatePayLoad;
- return this;
- }
- }
2.3 了解
AuthorizationHandler
AuthorizationHandler
為特定需求類型調用的授權處理程序的基類。也就是說我們處理策略是會到這個基類來處理,並且判斷是否認證成功,也就是授權成功。2.4 繼承
AuthorizationHandler
JwtAuthorizeHandler
繼承AuthorizationHandler
並實現泛型JwtAuthorizeBaseRequiremente
的定義,這樣子我們的自定義的策略委托驗證函數就會傳遞到這個處理類。我們需要重寫HandleRequirementAsync
來自定已處理。可以看到,最終我們還是調用上篇文章所講的驗證函數_jsonWebTokenValidate.Validate
,大家不清楚可以去看上篇文章。而requirement.validatePayLoad
便是我們稍后再外面自定義的驗證函數了。- public class JwtAuthorizeHandler : AuthorizationHandler<JwtAuthorizeBaseRequiremente>
- {
- private readonly JsonWebTokenSetting _setting;
- private readonly IJsonWebTokenValidate _jsonWebTokenValidate;
- public JwtAuthorizeHandler(IOptions<JsonWebTokenSetting> setting, IJsonWebTokenValidate jsonWebTokenValidate)
- {
- this._setting = setting.Value;
- this._jsonWebTokenValidate = jsonWebTokenValidate;
- }
- /// <summary>
- /// 驗證JWT
- /// </summary>
- /// <param name="context"></param>
- /// <param name="requirement"></param>
- /// <returns></returns>
- protected override Task HandleRequirementAsync(AuthorizationHandlerContext context, JwtAuthorizeBaseRequiremente requirement)
- {
- var httpContext = (context.Resource as AuthorizationFilterContext).HttpContext;
- var result = httpContext.Request.Headers.TryGetValue("Authorization", out StringValues authStr);
- if (!result || string.IsNullOrEmpty(authStr.ToString()))
- {
- throw new UnauthorizedAccessException("未授權,請傳遞Header頭的Authorization參數。");
- }
- result = result && _jsonWebTokenValidate.Validate(authStr.ToString().Substring("Bearer ".Length).Trim(), _setting, requirement.validatePayLoad);
- if (!result)
- {
- throw new UnauthorizedAccessException("驗證失敗,請查看傳遞的參數是否正確或是否有權限訪問該地址。");
- }
- context.Succeed(requirement);
- return Task.CompletedTask;
- }
- }
2.5 怎么使用呢?
-
我們需要在
Startup.cs
文件進行注冊服務。其中CommonAuthorize
繼承JwtAuthorizeBaseRequiremente
,並將自定義的策略方法,傳遞進去。其中common
是策略名稱。可以多個定義策略- public void ConfigureServices(IServiceCollection services)
- {
- services.AddJwt(Configuration);
- services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);
- services.AddAuthorization(option =>
- {
- option.AddPolicy("common", policy => policy.Requirements.Add(new CommonAuthorize().
- SetValidateFunc((playLoad, sertting) =>
- {
- //每個策略自定義驗證函數,playLoad為帶過來的參數字典,setting為失效時間與秘鑰
- return true;
- })));
- }).AddAuthentication(option =>
- {
- option.DefaultScheme = JwtBearerDefaults.AuthenticationScheme;
- });
- }
-
接着我們在想要的
Controller
或者Action
的頭部使用[Authorize(Policy = "common")]
,這樣每次進到相對應的Controller
或者Action
,會先進行策略驗證,而我們這邊驗證的便是JWT了。
總結一下,我們在這篇文章是基於上篇文章的,所以JWT的生成與驗證我們就不講了。兩篇文章講了JWT的驗證,兩種方式有好有壞,大家可以根據自己的模式進行選擇。
1.使用管道的方式,感覺方便點,清晰點
2. 使用自定義策略的方式,效率稍微高一點,畢竟不是所有的請求都會進行是否可以匿名訪問運算和建立管道的消耗,只有加入Authorize屬性的Controller和Action的才會進入
最后附上源碼,或者直接到我的GitHub上看看。后面要是有時間,可以講下IdentityServer4
的OAuth2的授權與認證。