如何从authrizationhandler .net核心获取params

Ram*_*ppy 14 c# .net-core asp.net-core

我正在使用授权处理程序将自定义授权放在.net核心的控制器中.如何从控制器获取参数并将其用于授权处理程序.

在旧的.net中,我可以像这样从Httpcontext请求参数中获取参数

var eventId = filterContext.RequestContext.HttpContext.Request.Params["id"];
Run Code Online (Sandbox Code Playgroud)

我不知道如何在.net核心中实现它

enter code here

public class HasAdminRoleFromAnySiteRequirement : AuthorizationHandler<HasAdminRoleFromAnySiteRequirement>, IAuthorizationRequirement
{

    public HasAdminRoleFromAnySiteRequirement()
    {

    }
    protected override Task HandleRequirementAsync(AuthorizationHandlerContext context,
        HasAdminRoleFromAnySiteRequirement requirement)
    {   

    //need to call get param from controller to used in the validation
    // something like this 
    //var eventId = filterContext.RequestContext.HttpContext.Request.Params["id"];
   // I tried the suggestion below but I can't get the parameter from routedata
   // var mvcContext = context.Resource as     Microsoft.AspNetCore.Mvc.Filters.AuthorizationFilterContext;            

        return Task.FromResult(0);
    }
}
Run Code Online (Sandbox Code Playgroud)

blo*_*art 19

在处理程序中,您可以执行以下操作

var mvcContext = context.Resource as 
    Microsoft.AspNetCore.Mvc.Filters.AuthorizationFilterContext;

if (mvcContext != null)
{
    // Examine MVC specific things like routing data.
}
Run Code Online (Sandbox Code Playgroud)

如果需要参数值,则在绑定发生之前运行授权属性块.相反,你会转移到控制器内的强制调用.这基本上是基于资源的授权,您的参数是资源.

您可以将授权服务注入您的控制器;

public class DocumentController : Controller
{
    IAuthorizationService _authorizationService;

    public DocumentController(IAuthorizationService authorizationService)
    {
        _authorizationService = authorizationService;
    }
}
Run Code Online (Sandbox Code Playgroud)

然后稍微改写你的处理程序;

public class DocumentAuthorizationHandler : AuthorizationHandler<MyRequirement, Document>
{
    public override Task HandleRequirementAsync(AuthorizationHandlerContext context,
                                                MyRequirement requirement,
                                                Document resource)
    {
        // Validate the requirement against the resource and identity.

        return Task.CompletedTask;
    }
}
Run Code Online (Sandbox Code Playgroud)

您可以看到此处理程序获取文档,这可以是您喜欢的任何内容,无论是ID的整数,还是某种类型的视图模型.

然后,您可以在HandleRequirementAsync()方法中访问它.

最后,一旦绑定发生,你就可以从你的控制器中调用它;

if (await authorizationService.AuthorizeAsync(
    User, 
    document,     
    yourRequirement))
{
}
Run Code Online (Sandbox Code Playgroud)

  • 行`context.Resource as Microsoft.AspNetCore.Mvc.Filters.AuthorizationFilterContext;`不再适用于`2.0.0-preview1`.我将`IHttpContextAccessor`注入我的`AuthorizationHandler`然后`_httpContextAccessor.HttpContext.GetRouteData()`. (6认同)
  • 我不知道.Net核心是否已更新,但显然现在可以访问RouteData值.这是真的@blowdart吗? (2认同)

Toh*_*hid 14

在ASP.NET Core 2.2中,您可以获取如下的路由参数值:

public class MyRequirementHandler : AuthorizationHandler<MyRequirement>
{
    protected override async Task HandleRequirementAsync(AuthorizationHandlerContext context, MyRequirement requirement)
    {
        var authContext = (AuthorizationFilterContext)context.Resource;
        var routeValueOfX = authContext.HttpContext.GetRouteValue("X");
        .
        .
        .
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 这在 aspnetcore 3.0 端点连接中会失败:在这种情况下,context.Resource 不是 AuthorizationFilterContext。IMO IHttpContextAccessor 是一个更强大的解决方案。 (3认同)
  • @Tohid是的,我建议注入IHttpContextAccessor;然后您可以避免有关 AuthorizationHandlerContext.Resource 类型的假设;-) (2认同)

Vah*_*idN 9

在启用了端点路由的ASP.NET Core 3.0中,您可以获取如下路由参数值:

public class MyRequirementHandler : AuthorizationHandler<MyRequirement>
{
    private readonly IHttpContextAccessor _httpContextAccessor;

    public MyRequirementHandler(IHttpContextAccessor httpContextAccessor)
    {
       _httpContextAccessor = httpContextAccessor ?? throw new ArgumentNullException(nameof(httpContextAccessor));
    }

    protected override async Task HandleRequirementAsync(AuthorizationHandlerContext context, MyRequirement requirement)
    {
        var routeData = _httpContextAccessor.HttpContext.GetRouteData();

        var areaName = routeData?.Values["area"]?.ToString();
        var area = string.IsNullOrWhiteSpace(areaName) ? string.Empty : areaName;

        var controllerName = routeData?.Values["controller"]?.ToString();
        var controller = string.IsNullOrWhiteSpace(controllerName) ? string.Empty : controllerName;

        var actionName = routeData?.Values["action"]?.ToString();
        var action = string.IsNullOrWhiteSpace(actionName) ? string.Empty : actionName;

        //...
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 这解决了我的问题谢谢!请注意,为了实现此目的,您需要使用以下方式注入 httpContextAccessor: services.AddTransient&lt;IHttpContextAccessor, HttpContextAccessor&gt;(); 在你的startup.cs中 (6认同)

小智 5

为了供将来参考,从 .NET Core 5.0 开始,现在改为传递 HttpContext,因此您可以执行以下操作:

if (context.Resource is HttpContext httpContext)
{
   var value = httpContext.GetRouteValue("key");
}
Run Code Online (Sandbox Code Playgroud)