jQuery Ajax calls and the Html.AntiForgeryToken()

前端 未结 20 2395
鱼传尺愫
鱼传尺愫 2020-11-22 16:34

I have implemented in my app the mitigation to CSRF attacks following the informations that I have read on some blog post around the internet. In particular these post have

20条回答
  •  遇见更好的自我
    2020-11-22 17:29

    I feel like an advanced necromancer here, but this is still an issue 4 years later in MVC5.

    To handle ajax requests properly the anti-forgery token needs to be passed to the server on ajax calls. Integrating it into your post data and models is messy and unnecessary. Adding the token as a custom header is clean and reusable - and you can configure it so you don't have to remember to do it every time.

    There is an exception - Unobtrusive ajax does not need special treatment for ajax calls. The token is passed as usual in the regular hidden input field. Exactly the same as a regular POST.

    _Layout.cshtml

    In _layout.cshtml I have this JavaScript block. It doesn't write the token into the DOM, rather it uses jQuery to extract it from the hidden input literal that the MVC Helper generates. The Magic string that is the header name is defined as a constant in the attribute class.

    
    

    Note the use of single quotes in the beforeSend function - the input element that is rendered uses double quotes that would break the JavaScript literal.

    Client JavaScript

    When this executes the beforeSend function above is called and the AntiForgeryToken is automatically added to the request headers.

    $.ajax({
      type: "POST",
      url: "CSRFProtectedMethod",
      dataType: "json",
      contentType: "application/json; charset=utf-8",
      success: function (data) {
        //victory
      }
    });
    

    Server Library

    A custom attribute is required to process the non standard token. This builds on @viggity's solution, but handles unobtrusive ajax correctly. This code can be tucked away in your common library

    [AttributeUsage(AttributeTargets.Class | AttributeTargets.Method)]
    public class ValidateAntiForgeryTokenOnAllPosts : AuthorizeAttribute
    {
        public const string HTTP_HEADER_NAME = "x-RequestVerificationToken";
    
        public override void OnAuthorization(AuthorizationContext filterContext)
        {
            var request = filterContext.HttpContext.Request;
    
            //  Only validate POSTs
            if (request.HttpMethod == WebRequestMethods.Http.Post)
            {
    
                var headerTokenValue = request.Headers[HTTP_HEADER_NAME];
    
                // Ajax POSTs using jquery have a header set that defines the token.
                // However using unobtrusive ajax the token is still submitted normally in the form.
                // if the header is present then use it, else fall back to processing the form like normal
                if (headerTokenValue != null)
                {
                    var antiForgeryCookie = request.Cookies[AntiForgeryConfig.CookieName];
    
                    var cookieValue = antiForgeryCookie != null
                        ? antiForgeryCookie.Value
                        : null;
    
                    AntiForgery.Validate(cookieValue, headerTokenValue);
                }
                else
                {
                    new ValidateAntiForgeryTokenAttribute()
                        .OnAuthorization(filterContext);
                }
            }
        }
    }
    

    Server / Controller

    Now you just apply the attribute to your Action. Even better you can apply the attribute to your controller and all requests will be validated.

    [HttpPost]
    [ValidateAntiForgeryTokenOnAllPosts]
    public virtual ActionResult CSRFProtectedMethod()
    {
      return Json(true, JsonRequestBehavior.DenyGet);
    }
    

提交回复
热议问题