ASP.NET Core Web API自定义AuthorizeAttribute问题

azh*_*ahi 3 c# asp.net-core asp.net-core-webapi

我正在研究 ASP.NET Core Web API。我正在尝试创建自定义Authorize属性,但我陷入困境。我不明白我错过了什么。我有以下属性和过滤器代码Authorize

public class AuthorizeAttribute : TypeFilterAttribute
{
    public AuthorizeAttribute(params string[] claim) : base(typeof(AuthorizeFilter))
    {
        Arguments = new object[] { claim };
    }
}

public class AuthorizeFilter : IAuthorizationFilter
{
    readonly string[] _claim;

    public AuthorizeFilter(params string[] claim)
    {
        _claim = claim;
    }

    public void OnAuthorization(AuthorizationFilterContext context)
    {
        var IsAuthenticated = context.HttpContext.User.Identity.IsAuthenticated;
        var claimsIndentity = context.HttpContext.User.Identity as ClaimsIdentity;

        if (IsAuthenticated)
        {
            bool flagClaim = false;
            foreach (var item in _claim)
            {
                if (context.HttpContext.User.HasClaim("Role", item))
                    flagClaim = true;
            }

            if (!flagClaim)
            {
                //if (context.HttpContext.Request.IsAjaxRequest())
                    context.HttpContext.Response.StatusCode = (int)HttpStatusCode.Forbidden; //Set HTTP 403 
                //else
                //    context.Result = new RedirectResult("~/Login/Index");
            }
        }
        else
        {
            //if (context.HttpContext.Request.IsAjaxRequest())
            //{
                context.HttpContext.Response.StatusCode = (int)HttpStatusCode.Unauthorized; //Set HTTP 401 -   
            //}
            //else
            //{
            //    context.Result = new RedirectResult("~/Login/Index");
            //}
        }
        return;
    }
}
Run Code Online (Sandbox Code Playgroud)

我从某处复制了这段代码并注释了不必要的行。

这是我的控制器类,我试图将其放入其中:

[Route("api/[controller]/[action]")]
[ApiController]
[Authorize]
public class JobController : ControllerBase
{
    // GET: api/<JobController>
    [HttpGet]
    [ActionName("GetAll")]
    public List<Job> Get()
    {
        return JobDataLog.GetAllJobQueue();
    }

    // GET api/<JobController>/5
    [HttpGet("{ID}")]
    [ActionName("GetByID")]
    public Job Get(Guid ID)
    {
        return JobDataLog.GetJob(ID);
    }

    // GET api/<JobController>/5
    [HttpGet]
    [ActionName("GetCount")]
    public int GetCount()
    {
        return JobDataLog.GetJobTotal();
    }
}
Run Code Online (Sandbox Code Playgroud)

还有Startup.cs的Configure和ConfigureService方法

// This method gets called by the runtime. Use this method to add services to the container.
        public void ConfigureServices(IServiceCollection services)
        {
            services.AddControllers();
            services.AddDistributedMemoryCache();
            services.AddSession(options =>
            {
                options.IdleTimeout = TimeSpan.FromMinutes(60);
            });

            var tokenKey = Configuration.GetValue<string>("TokenKey");
            var key = Encoding.ASCII.GetBytes(tokenKey);
            services.AddAuthentication(x => { x.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme; })
                .AddJwtBearer(x =>
            {
                x.RequireHttpsMetadata = false;
                x.SaveToken = true;
                x.TokenValidationParameters = new TokenValidationParameters
                {
                    ValidateIssuerSigningKey = true,
                    IssuerSigningKey = new SymmetricSecurityKey(key),
                    ValidateIssuer = false,
                    ValidateAudience = false
                };
            });
            services.AddSingleton<IJWTAuthenticationManager>(new JWTAuthenticationManager(tokenKey));
        }

        // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
        public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
        {
            if (env.IsDevelopment())
            {
                app.UseDeveloperExceptionPage();
            }

            app.UseHttpsRedirection();

            app.UseRouting();

            app.UseCookiePolicy();
            app.UseSession();
            app.Use(async (context, next) =>
            {
                var JWToken = context.Session.GetString("JWToken");
                if (!string.IsNullOrEmpty(JWToken))
                {
                    context.Request.Headers.Add("Authorization", "Bearer " + JWToken);
                }
                await next();
            });

            app.UseAuthentication();
            app.UseAuthorization();

            app.UseEndpoints(endpoints =>
            {
                endpoints.MapControllers();
            });
        }
Run Code Online (Sandbox Code Playgroud)

问题是,即使这个控制器具有该Authorize属性,所有操作都会被调用,甚至Authorize属性,即使过滤器使授权无效,

另外,当我在方法中放置以下代码时OnAuthorization

context.Result = new StatusCodeResult(StatusCodes.Status401Unauthorized);
Run Code Online (Sandbox Code Playgroud)

它阻止了所有操作的访问,包括那些具有AllowAnnoynmous属性的操作。

请帮助我,我已经被困在这个问题上三个小时了。

MrV*_*ype 5

如果您确实想使用自定义AuthorizeAttribute,就可以了,这很有效。:)

您将看到一些波浪线,但 VS 将能够自动添加语句using

原来的代码有多个问题:

  1. 设置Reponse.StatusCode实际上并不导致返回响应。
  2. HttpContext.User首先不会填充,因为如果端点使用内置AuthorizeAttribute. 以下代码通过派生自 来解决此问题AuthorizeAttribute
  3. 在这种情况下,不需要额外的过滤器工厂类,因为您没有注入依赖项。不过,如果您必须注入,我认为您会运气不佳,因为您无法同时从TypeFilterAttribute和派生AuthorizeAttribute,并且声明列表将始终为空。

工作代码

public class MyAuthorizeAttribute : AuthorizeAttribute, IAuthorizationFilter
{
    readonly string[] _requiredClaims;

    public MyAuthorizeAttribute(params string[] claims)
    {
        _requiredClaims = claims;
    }

    public void OnAuthorization(AuthorizationFilterContext context)
    {
        var isAuthenticated = context.HttpContext.User.Identity.IsAuthenticated;
        if (!isAuthenticated)
        {
            context.Result = new UnauthorizedResult();
            return;
        }

        var hasAllRequredClaims = _requiredClaims.All(claim => context.HttpContext.User.HasClaim(x => x.Type == claim));
        if (!hasAllRequredClaims)
        {
            context.Result = new ForbidResult();
            return;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

您可能应该使用策略

其工作方式如此糟糕的原因是 ASP.NET Core 团队不希望您编写自定义 Authorize Attributes。请参阅有关该主题的答案。“正确”的方法是创建保单,并将您的索赔要求分配给这些保单。但我也认为授权如此不灵活且缺乏对基本场景的支持是愚蠢的。