【问题标题】:Writing a filter that applies to both MVC and Web API requests编写适用于 MVC 和 Web API 请求的过滤器
【发布时间】:2013-09-17 05:07:27
【问题描述】:

我的 ASP.NET 应用程序使用 ASP.NET MVC 为网页提供服务,并使用 ASP.NET Web API 为源自这些网页的 AJAX 请求提供服务。

我想要某些事情,例如检查请求是否经过身份验证,在 HttpContext 中设置适当的事情,无论我正在处理哪种请求。

我现在要写两个类

1) 一个继承 - System.Web.Mvc.ActionFilterAttribute 用于 MVC 请求 2)一个继承 - System.Web.Http.Filters.ActionFilterAttribute for Web API请求

有没有办法为网站应用同时针对 MVC 和 API 请求运行的过滤器?或者通过旧的http模块是这种用例的推荐方式?

【问题讨论】:

    标签: c# asp.net .net asp.net-mvc-4 asp.net-web-api


    【解决方案1】:

    我没有看到将 MVC 和 Web API 中的身份验证/授权过滤器放入单个过滤器的好方法,因为它们的行为非常不同。对于当用户授权失败时的 MVC 请求,您希望将他们重定向到另一个页面以以另一个用户身份登录,或者只是让他们知道他们无权访问该页面。对于用户授权失败时的 Web API 请求,您希望发送指示授权失败的 HTTP 状态代码并让客户端处理它。例如,如果您 add basic authentication to your filter as described in this article,他们甚至可以在行为和功能上更多地离题。这仅对 Web API 请求有意义,对 MVC 请求无效。我认为拥有两个不同的过滤器是一种更清洁的分离方式。

    【讨论】:

    • 当然,我认为身份验证是一个不好的例子。假设我们想根据请求在 HttpContext 中设置某些东西 - 我想对 MVC 和 Web API 请求都这样做。
    【解决方案2】:

    这可以通过使用代理模式来实现。下面是一个简化的EnableCorsAttribute 示例,用于展示如何:

    using System.Web.Http.Controllers;
    using System.Web.Http.Filters;
    using System.Web.Mvc;
    using System.Net.Http;
    using System.Threading.Tasks;
    using System.Threading;
    using System.Collections.Generic;
    using System.Linq;
    using System;
    
    public class EnableCorsAttribute:System.Web.Mvc.ActionFilterAttribute, System.Web.Http.Filters.IActionFilter {
        class HttpActionFilter:System.Web.Http.Filters.ActionFilterAttribute {
            public override void OnActionExecuted(HttpActionExecutedContext filterContext) {
                var res = filterContext.Response;
                res.Headers.Add("Access-Control-Allow-Origin", String.Join(",\x20", m_httpFilter.Origins));
                res.Headers.Add("Access-Control-Allow-Headers", String.Join(",\x20", m_httpFilter.Headers));
                res.Headers.Add("Access-Control-Allow-Credentials", "true");
                base.OnActionExecuted(filterContext);
            }
    
            public HttpActionFilter(EnableCorsAttribute filter) {
                m_httpFilter=filter;
            }
    
            EnableCorsAttribute m_httpFilter;
        }
    
        public override void OnActionExecuted(ActionExecutedContext filterContext) {
            var res = filterContext.HttpContext.Response;
            res.Headers.Add("Access-Control-Allow-Origin", String.Join(",\x20", this.Origins));
            res.Headers.Add("Access-Control-Allow-Headers", String.Join(",\x20", this.Headers));
            res.Headers.Add("Access-Control-Allow-Credentials", "true");
            base.OnActionExecuted(filterContext);
        }
    
        public Task<HttpResponseMessage> ExecuteActionFilterAsync(HttpActionContext actionContext, CancellationToken cancellationToken, Func<Task<HttpResponseMessage>> continuation) {
            return m_mvcFilter.ExecuteActionFilterAsync(actionContext, cancellationToken, continuation);
        }
    
        public EnableCorsAttribute(String origins, String headers, String methods) {
            m_mvcFilter=new HttpActionFilter(this);
    
            var separator = new[] { ',' };
            var options = StringSplitOptions.RemoveEmptyEntries;
            this.Origins=origins.Split(separator, options);
            this.Headers=headers.Split(separator, options);
            this.Methods=methods.Split(separator, options);
        }
    
        public IList<String> Origins {
            get; private set;
        }
    
        public IList<String> Headers {
            get; private set;
        }
    
        public IList<String> Methods {
            get; private set;
        }
    
        System.Web.Http.Filters.IActionFilter m_mvcFilter;
    }
    

    【讨论】:

      猜你喜欢
      • 2020-12-05
      • 1970-01-01
      • 2018-03-14
      • 2016-04-07
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2021-08-18
      相关资源
      最近更新 更多