Web安全相关(二):跨站请求伪造(CSRF/XSRF)

简介
  CSRF(Cross-site request forgery跨站请求伪造,也被称为“One Click Attack”或者Session Riding,一般缩写为CSRF或者XSRF,是一种对网站的恶意利用。尽管听起来像跨站脚本(XSS),但它与XSS很是不一样,而且攻击方式几乎相左。XSS利用站点内的信任用户,而CSRF则经过假装来自受信任用户的请求来利用受信任的网站。与XSS攻击相比,CSRF攻击每每不大流行(所以对其进行防范的资源也至关稀少)和难以防范,因此被认为比XSS更具危险性。javascript

场景html

      某程序员大神God在某在线银行Online Bank给他的朋友Friend转帐。java

  

  

  转帐后,出于好奇,大神God查看了网站的源文件,以及捕获到转帐的请求。git

   

  

  大神God发现,这个网站没有作防止CSRF的措施,并且他本身也有一个有必定访问量的网站,因而,他计划在本身的网站上内嵌一个隐藏的Iframe伪造请求(每10s发送一次),来等待鱼儿Fish上钩,给本身转帐。程序员

  网站源码:github

 1 <html>
 2 <head>
 3     <meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
 4     <title></title>
 5 </head>
 6 <body>
 7 <div>
 8  我是一个内容丰富的网站,你不会关闭我!  9 </div>
10 
11 <iframe name="frame" src="invalid.html" sandbox="allow-same-origin allow-scripts allow-forms" style="display: none; width: 800px; height: 1000px;"> </iframe> 
12 <script type="text/javascript">
13  setTimeout("self.location.reload();", 10000); 14 </script>
15 </body>
16 </html>

   伪造请求源码:web

 1 <html>
 2 <head>
 3     <title></title>
 4 </head>
 5 <body>
 6 <form id="theForm" action="http://localhost:22699/Home/Transfer" method="post">
 7     <input class="form-control" id="TargetUser" name="TargetUser" placeholder="用户名" type="text" value="God" />
 8     <input class="form-control" id="Amount" name="Amount" placeholder="转帐金额" type="text" value="100" />
 9 </form>
10 
11 <script type="text/javascript">
12  document.getElementById('theForm').submit(); 13 </script>
14 </body>
15 </html>

   

  鱼儿Fish打开了大神God的网站,在上面浏览丰富多彩的内容。此时伪造请求的结果是这样的(为了演示效果,去掉了隐藏):ajax

   

  由于鱼儿Fish没有登录,因此,伪造请求一直没法执行,一直跳转回登陆页面。数据库

  而后鱼儿Fish想起了要登陆在线银行Online Bank查询内容,因而他登陆了Online Bank。json

  此时伪造请求的结果是这样的(为了演示效果,去掉了隐藏):

   

  鱼儿Fish每10秒会给大神God转帐100元。

   

 

防止CSRF

  CSRF能成功是由于同一个浏览器会共享Cookies,也就是说,经过权限认证和验证是没法防止CSRF的。那么应该怎样防止CSRF呢?其实防止CSRF的方法很简单,只要确保请求是本身的站点发出的就能够了。那怎么确保请求是发自于本身的站点呢?ASP.NET以Token的形式来判断请求。

  咱们须要在咱们的页面生成一个Token,发请求的时候把Token带上。处理请求的时候须要验证Cookies+Token。

  

  

  此时伪造请求的结果是这样的(为了演示效果,去掉了隐藏):

  

$.ajax

  若是个人请求不是经过Form提交,而是经过Ajax来提交,会怎样呢?结果是验证不经过。

  

  为何会这样子?咱们回头看看加了@Html.AntiForgeryToken()后页面和请求的变化。

  1. 页面多了一个隐藏域,name为__RequestVerificationToken。

  

  2. 请求中也多了一个字段__RequestVerificationToken。

  

  原来要加这么个字段,我也加一个不就能够了!

    

  啊!为何仍是不行...逼我放大招,研究源码去!

  

  噢!原来token要从Form里面取。可是ajax中,Form里面并无东西。那token怎么办呢?我把token放到碗里,不对,是放到header里。

   js代码:

 1 $(function () {  2             var token = $('@Html.AntiForgeryToken()').val();  3 
 4             $('#btnSubmit').click(function () {  5                 var targetUser = $('#TargetUser').val();  6                 var amount = $('#Amount').val();  7                 var data = { 'targetUser': targetUser, 'amount': amount };  8                 return $.ajax({  9                     url: '@Url.Action("Transfer2", "Home")', 10                     type: 'POST', 11  data: JSON.stringify(data), 12                     contentType: 'application/json', 13                     dataType: 'json', 14                     traditional: 'true', 15                     beforeSend: function (xhr) { 16                         xhr.setRequestHeader('__RequestVerificationToken', token); 17  }, 18                     success:function() { 19                         window.location = '@Url.Action("Index", "Home")'; 20  } 21  }); 22  }); 23         });

   在服务端,参考ValidateAntiForgeryTokenAttribute,编写一个AjaxValidateAntiForgeryTokenAttribute:

 1 [AttributeUsage(AttributeTargets.Class | AttributeTargets.Method)]  2     public class AjaxValidateAntiForgeryTokenAttribute : FilterAttribute, IAuthorizationFilter  3  {  4         public void OnAuthorization(AuthorizationContext filterContext)  5  {  6             if (filterContext == null)  7  {  8                 throw new ArgumentNullException("filterContext");  9  } 10 
11             var request = filterContext.HttpContext.Request; 12 
13             var antiForgeryCookie = request.Cookies[AntiForgeryConfig.CookieName]; 14             var cookieValue = antiForgeryCookie != null ? antiForgeryCookie.Value : null; 15             var formToken = request.Headers["__RequestVerificationToken"]; 16  AntiForgery.Validate(cookieValue, formToken); 17  } 18     }

   而后调用时把ValidateAntiForgeryToken替换成AjaxValidateAntiForgeryToken。

   

  大功告成,好有成就感!

全局处理

  若是全部的操做请求都要加一个ValidateAntiForgeryToken或者AjaxValidateAntiForgeryToken,不是挺麻烦吗?能够在某个地方统一处理吗?答案是能够的。

  ValidateAntiForgeryTokenAttribute继承IAuthorizationFilter,那就在AuthorizeAttribute里作统一处理吧。

  ExtendedAuthorizeAttribute:

 1 public class ExtendedAuthorizeAttribute : AuthorizeAttribute  2  {  3         public override void OnAuthorization(AuthorizationContext filterContext)  4  {  5  PreventCsrf(filterContext);  6             base.OnAuthorization(filterContext);  7  GenerateUserContext(filterContext);  8  }  9 
10         /// <summary>
11         /// http://www.asp.net/mvc/overview/security/xsrfcsrf-prevention-in-aspnet-mvc-and-web-pages
12         /// </summary>
13         private static void PreventCsrf(AuthorizationContext filterContext) 14  { 15             var request = filterContext.HttpContext.Request; 16 
17             if (request.HttpMethod.ToUpper() != "POST") 18  { 19                 return; 20  } 21 
22             var allowAnonymous = HasAttribute(filterContext, typeof(AllowAnonymousAttribute)); 23 
24             if (allowAnonymous) 25  { 26                 return; 27  } 28 
29             var bypass = HasAttribute(filterContext, typeof(BypassCsrfValidationAttribute)); 30 
31             if (bypass) 32  { 33                 return; 34  } 35 
36             if (filterContext.HttpContext.Request.IsAjaxRequest()) 37  { 38                 var antiForgeryCookie = request.Cookies[AntiForgeryConfig.CookieName]; 39                 var cookieValue = antiForgeryCookie != null ? antiForgeryCookie.Value : null; 40                 var formToken = request.Headers["__RequestVerificationToken"]; 41  AntiForgery.Validate(cookieValue, formToken); 42  } 43             else
44  { 45  AntiForgery.Validate(); 46  } 47  } 48 
49         private static bool HasAttribute(AuthorizationContext filterContext, Type attributeType) 50  { 51             return filterContext.ActionDescriptor.IsDefined(attributeType, true) ||
52                    filterContext.ActionDescriptor.ControllerDescriptor.IsDefined(attributeType, true); 53  } 54 
55         private static void GenerateUserContext(AuthorizationContext filterContext) 56  { 57             var formsIdentity = filterContext.HttpContext.User.Identity as FormsIdentity; 58 
59             if (formsIdentity == null || string.IsNullOrWhiteSpace(formsIdentity.Name)) 60  { 61                 UserContext.Current = null; 62                 return; 63  } 64 
65             UserContext.Current = new WebUserContext(formsIdentity.Name); 66  } 67     }

   而后在FilterConfig注册一下。

    

  FAQ:

  1. BypassCsrfValidationAttribute是什么鬼?不是有个AllowAnonymousAttribute吗?

  若是有些操做你不须要作CSRF的处理,好比附件上传,你能够在对应的Controller或Action上添加BypassCsrfValidationAttribute。

  AllowAnonymousAttribute不只会绕过CSRF的处理,还会绕过认证和验证。BypassCsrfValidationAttribute绕过CSRF但不绕过认证和验证,

也就是BypassCsrfValidationAttribute做用于那些登陆或受权后的Action。

 

  2. 为何只处理POST请求?

  我开发的时候有一个原则,查询都用GET,操做用POST,而对于查询的请求没有必要作CSRF的处理。你们能够按本身的须要去安排!

  

  3. 我作了全局处理,而后还在Controller或Action上加了ValidateAntiForgeryToken或者AjaxValidateAntiForgeryToken,会冲突吗?

  不会冲突,只是验证会作两次。

源码下载

  为了方便使用,我没有使用任何数据库,而是用了一个文件来存储数据。代码下载后能够直接运行,无需配置。

  下载地址:https://github.com/ErikXu/CSRF

 

文章转载自:http://www.cnblogs.com/Erik_Xu/p/5481441.html

相关文章
相关标签/搜索