.NET Core授权失败如何自定义响应信息?

开发 后端
本文我们来看看在.NET 5中为何要出现针对授权失败的中间件接口?它是如何一步步衍生出来的呢?以及 对于授权失败根据实际需要如何自定义响应错误,以及如何获取对应路由信息等等。

 [[410575]]

本文转载自微信公众号「JeffckyShare」,作者Jeffcky。转载本文请联系JeffckyShare公众号。

本文我们来看看在.NET 5中为何要出现针对授权失败的中间件接口?它是如何一步步衍生出来的呢?以及 对于授权失败根据实际需要如何自定义响应错误,以及如何获取对应路由信息等等

授权失败自定义响应信息

如下是在.NET 5之前,对于授权处理,我们大多实现自定义的AuthorizationHandler

public class CustomAuthorizeHandler : AuthorizationHandler<CustomAuthorizationRequirement> 

    protected override Task HandleRequirementAsync(AuthorizationHandlerContext context, CustomAuthorizationRequirement requirement) 
    { 
        throw new NotImplementedException(); 
    } 

 
public class CustomAuthorizationRequirement : IAuthorizationRequirement 

    public CustomAuthorizationRequirement() 
    { 
    } 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.

但此时参数给予的是授权上下文,我们并不能拿到当前请求上下文中的相关信息,如果是在mvc中,想必大多是如下这般获取的

protected override Task HandleRequirementAsync(AuthorizationHandlerContext context, CustomAuthorizationRequirement requirement) 

    var context = context.Resource as HttpContext; 

  • 1.
  • 2.
  • 3.
  • 4.

但对于前后分离的web api中,若我没记错的话,这样是获取到的是空,于是乎我们借助于注入上下文接口实现,演变成如下这样

public class CustomAuthorizeHandler : AuthorizationHandler<CustomAuthorizationRequirement> 

    private readonly IHttpContextAccessor _accessor; 
    public CustomAuthorizeHandler(IHttpContextAccessor accessor) 
    { 
        _accessor = accessor; 
    } 
    protected async override Task HandleRequirementAsync(AuthorizationHandlerContext context, CustomAuthorizationRequirement requirement) 
    { 
        var httpContext = _accessor.HttpContext; 
 
        // 授权失败响应信息 
        await httpContext.Response.WriteAsync("授权失败"); 
 
        //响应失败调用 
        context.Fail(); 
 
    } 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.

通过上下文可以拿到比如用户声明信息等等,貌似已经基本满足我们实际业务需求,那要是我想获取路由信息又该如何呢?在3.0以下貌似只能通过Path自己解析(个人猜测)

从.NET Core 3.0+上,官方开放针对上下文的扩展方法,提供给我们获取路由节点元数据详细信息

在该终结点类存在一个元数据属性,该属性为集合,该元数据包含任何你想要的东东

这里必须强调一下,我最喜爱.NET Core的一点是,很多时候我们会封装类库,并在类库中使用到Web APi中相关的上下文一切信息等等,如果是以前.NET Framework怕是有点麻烦

比如如上在类库中获取上下文接口,如果你还是延续旧思想,查看vs智能提示你是否需要安装包,你会发现在Web APi中版本和你安装的版本是对应不上的,这可能是有问题的哈(具体细节我并未深入探究),但实际上我想安装的是.NET 5

在.NET Core类库中要实现.NET Core相关基础框架信息,只需要在类库项目文件中引入支持.NET Core应用程序包包即可,如此才和当前应用程序版本完全一致

<ItemGroup> 
   <FrameworkReference Include="Microsoft.AspNetCore.App" /> 
 </ItemGroup> 
  • 1.
  • 2.
  • 3.

 

面向不同群体读者,这里重点强调下,以免初学.NET Core童鞋路走偏了!话题扯远了,比如如上述我们想要获取到元数据中的控制器和action名称,该元数据集合参数都是object,所以我们想要对应的信息,需要稍微清楚一点.NET Core基本流程处理所提供的各个对象

public class CustomAuthorizeHandler : AuthorizationHandler<CustomAuthorizationRequirement> 

    private readonly IHttpContextAccessor _accessor; 
    public CustomAuthorizeHandler(IHttpContextAccessor accessor) 
    { 
        _accessor = accessor; 
    } 
    protected async override Task HandleRequirementAsync(AuthorizationHandlerContext context, CustomAuthorizationRequirement requirement) 
    { 
        var httpContext = _accessor.HttpContext; 
 
        var endPoint = httpContext.GetEndpoint(); 
 
        var controllerActionDescriptor = (ControllerActionDescriptor)endPoint.Metadata 
            .ToList().FirstOrDefault(d => d is ControllerActionDescriptor); 
 
        var controllerName = controllerActionDescriptor.ControllerName; 
 
        var actionName = controllerActionDescriptor.ActionName; 
 
    } 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.

讲到这里,实现对应抽象授权处理对象,基本上可满足我们的需求,即使上述拿到上下文并响应,但是在接口响应上我们是获取不到的,因为授权上下文,只提供Fail和Succeed方法,要是我们想根据业务失败后直接响应呢?所以最大的问题出在:我们无法完全控制响应,以及自定义响应

这个时候,经过开发者在github上激烈的反馈,官方在.NET 5给出了,针对授权处理的中间件接口,上下文也已直接对外暴露

public class CustomAuthorizationMiddlewareResultHandler 
        : IAuthorizationMiddlewareResultHandler 

 
    public async Task HandleAsync(RequestDelegate next
      HttpContext context, AuthorizationPolicy policy, PolicyAuthorizationResult authorizeResult) 
    { 
        var endPoint = context.GetEndpoint(); 
 
        var controllerActionDescriptor = (ControllerActionDescriptor)endPoint.Metadata 
          .ToList().FirstOrDefault(d => d is ControllerActionDescriptor); 
 
        var controllerName = controllerActionDescriptor.ControllerName; 
 
        var actionName = controllerActionDescriptor.ActionName; 
 
        if (!context.User.Identity.IsAuthenticated) 
        { 
            context.Response.StatusCode = (int)HttpStatusCode.Unauthorized; 
            await context.Response.WriteAsync("{\"data\":{\"succeeded\":false,\"code\":401,\"message\":\"登录已过期,请重新登录\"}}"); 
            return
        } 
        else if (!await HandleRequirementEvaluateAsync(context.User, controllerName, actionName)) 
        { 
            context.Response.StatusCode = (int)HttpStatusCode.Forbidden; 
            await context.Response.WriteAsync("{\"data\":{\"succeeded\":false,\"code\":403,\"message\":\"您暂无足够的权限执行该操作\"}}"); 
            return
        } 
        await next(context); 
    } 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.

自从.NET 5提供给了我们授权中间件接口,一切又是那么得心应手!

 

责任编辑:武晓燕 来源: JeffckyShare
相关推荐

2021-08-09 10:31:33

自定义授权响应

2009-09-03 13:34:03

.NET自定义控件

2009-07-06 13:49:29

2024-04-02 09:52:12

自定义返回类API开发

2009-08-06 17:13:56

ASP.NET自定义控

2009-11-12 16:14:28

ADO.NET自定义对

2009-08-04 13:35:16

ASP.NET自定义样

2009-09-07 22:00:15

LINQ自定义

2021-07-06 23:48:45

.NET用户信息

2013-06-27 11:10:01

iOS开发自定义UISlider

2009-08-10 14:16:59

ASP.NET自定义控

2009-07-28 09:32:41

ASP.NET自定义控

2015-02-12 15:33:43

微信SDK

2009-07-31 10:23:09

ASP.NET源码DateTimePic

2009-07-20 13:47:08

iBATIS.NET字

2010-01-15 15:26:46

VB.NET自定义类型

2010-01-18 15:43:35

VB.NET自定义属性

2023-12-28 08:22:33

响应数据转换

2015-02-12 15:38:26

微信SDK

2023-01-03 07:40:27

自定义滑块组件
点赞
收藏

51CTO技术栈公众号