在 MVC 应用程序上自动执行防伪令牌验证
Automating Anti Forgery Token Validation on MVC Apps
有没有一种方法可以在控制器上的所有 HTTP post 方法上自动添加 [ValidateAntiForgeryToken]
注释,而无需显式定义它?
还有没有办法扩展 MVC Html.BeginForm()
助手以始终包含防伪标记?
最后这样做的目的是保持整个应用程序的一致性,在某些情况下是否有理由不这样做?
我自己正在研究这个主题并在下面编译了完整的解决方案 here。它由几个部分组成,包括 AntiForgeryTokenFilterProvider
和 BeginSecureForm
辅助方法。它还允许使用 DisableAntiForgeryCheckAttribute
.
跳过单个操作的验证
public class AntiForgeryTokenFilterProvider : IFilterProvider
{
public IEnumerable<Filter> GetFilters(ControllerContext controllerContext, ActionDescriptor actionDescriptor)
{
IEnumerable<FilterAttribute> filters = actionDescriptor.GetFilterAttributes(true);
bool disableAntiForgery = filters.Any(f => f is DisableAntiForgeryCheckAttribute);
string method = controllerContext.HttpContext.Request.HttpMethod;
if (!disableAntiForgery
&& String.Equals(method, "POST", StringComparison.OrdinalIgnoreCase))
{
yield return new Filter(new ValidateAntiForgeryTokenAttribute(), FilterScope.Global, null);
}
}
}
[AttributeUsage(AttributeTargets.Method)]
public sealed class DisableAntiForgeryCheckAttribute : FilterAttribute
{
}
// Usage:
public class FilterConfig
{
public static void RegisterGlobalFilters(GlobalFilterCollection filters)
{
//**//
FilterProviders.Providers.Add(new AntiForgeryTokenFilterProvider());
//**//
}
}
// Html Helper method
public static class HtmlExtensions
{
public static MvcForm BeginSecureForm(this HtmlHelper html, string action, string controller)
{
var form = html.BeginForm(action, controller);
html.ViewContext.Writer.Write(html.AntiForgeryToken().ToHtmlString());
return form;
}
}
有没有一种方法可以在控制器上的所有 HTTP post 方法上自动添加 [ValidateAntiForgeryToken]
注释,而无需显式定义它?
还有没有办法扩展 MVC Html.BeginForm()
助手以始终包含防伪标记?
最后这样做的目的是保持整个应用程序的一致性,在某些情况下是否有理由不这样做?
我自己正在研究这个主题并在下面编译了完整的解决方案 here。它由几个部分组成,包括 AntiForgeryTokenFilterProvider
和 BeginSecureForm
辅助方法。它还允许使用 DisableAntiForgeryCheckAttribute
.
public class AntiForgeryTokenFilterProvider : IFilterProvider
{
public IEnumerable<Filter> GetFilters(ControllerContext controllerContext, ActionDescriptor actionDescriptor)
{
IEnumerable<FilterAttribute> filters = actionDescriptor.GetFilterAttributes(true);
bool disableAntiForgery = filters.Any(f => f is DisableAntiForgeryCheckAttribute);
string method = controllerContext.HttpContext.Request.HttpMethod;
if (!disableAntiForgery
&& String.Equals(method, "POST", StringComparison.OrdinalIgnoreCase))
{
yield return new Filter(new ValidateAntiForgeryTokenAttribute(), FilterScope.Global, null);
}
}
}
[AttributeUsage(AttributeTargets.Method)]
public sealed class DisableAntiForgeryCheckAttribute : FilterAttribute
{
}
// Usage:
public class FilterConfig
{
public static void RegisterGlobalFilters(GlobalFilterCollection filters)
{
//**//
FilterProviders.Providers.Add(new AntiForgeryTokenFilterProvider());
//**//
}
}
// Html Helper method
public static class HtmlExtensions
{
public static MvcForm BeginSecureForm(this HtmlHelper html, string action, string controller)
{
var form = html.BeginForm(action, controller);
html.ViewContext.Writer.Write(html.AntiForgeryToken().ToHtmlString());
return form;
}
}