跨站请求伪造(CSRF/XSRF)

简介

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

 

场景

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

 

  

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

 

 

 

  大神God发现,这个网站没有做防止CSRF的措施,而且他自己也有一个有一定访问量的网站,于是,他计划在自己的网站上内嵌一个隐藏的Iframe伪造请求(每10s发送一次),来等待鱼儿Fish上钩,给自己转账。

  网站源码:


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


    伪造请求源码:


<html><head><title></title></head><body><form id="theForm" action="http://localhost:22699/Home/Transfer" method="post"><input class="form-control" id="TargetUser" name="TargetUser" placeholder="用户名" type="text" value="God" /><input class="form-control" id="Amount" name="Amount" placeholder="转账金额" type="text" value="100" /></form><script type="text/javascript">document.getElementById('theForm').submit();</script></body></html>


 

 

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

  

  因为鱼儿Fish没有登陆,所以,伪造请求一直无法执行,一直跳转回登录页面。

  然后鱼儿Fish想起了要登录在线银行Online Bank查询内容,于是他登录了Online Bank。

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

 

 

  鱼儿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代码:

$(function () {            var token = $('@Html.AntiForgeryToken()').val();$('#btnSubmit').click(function () {                var targetUser = $('#TargetUser').val();                var amount = $('#Amount').val();                var data = { 'targetUser': targetUser, 'amount': amount };                return $.ajax({url: '@Url.Action("Transfer2", "Home")',type: 'POST',data: JSON.stringify(data),contentType: 'application/json',dataType: 'json',traditional: 'true',beforeSend: function (xhr) {xhr.setRequestHeader('__RequestVerificationToken', token);},success:function() {window.location = '@Url.Action("Index", "Home")';}});});});


 

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

[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method)]    
public class AjaxValidateAntiForgeryTokenAttribute : FilterAttribute, IAuthorizationFilter{     public void OnAuthorization(AuthorizationContext filterContext){        
           
if (filterContext == null){              
               
throw new ArgumentNullException("filterContext");}      
      
var request = filterContext.HttpContext.Request;            var antiForgeryCookie = request.Cookies[AntiForgeryConfig.CookieName];          
      
var cookieValue = antiForgeryCookie != null ? antiForgeryCookie.Value : null;          
      
var formToken = request.Headers["__RequestVerificationToken"];AntiForgery.Validate(cookieValue, formToken);}}

 

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

  

  大功告成,好有成就感!

 

全局处理

  如果所有的操作请求都要加一个ValidateAntiForgeryToken或者AjaxValidateAntiForgeryToken,不是挺麻烦吗?可以在某个地方统一处理吗?答案是阔仪的。

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

  ExtendedAuthorizeAttribute:

public class ExtendedAuthorizeAttribute : AuthorizeAttribute{        

 public override void OnAuthorization(AuthorizationContext filterContext){PreventCsrf(filterContext);  
           
base.OnAuthorization(filterContext);GenerateUserContext(filterContext);}      
        
/// <summary>/// http://www.asp.net/mvc/overview/security/xsrfcsrf-prevention-in-aspnet-mvc-and-web-pages/// </summary>private static void PreventCsrf(AuthorizationContext filterContext){          
       
var request = filterContext.HttpContext.Request;            if (request.HttpMethod.ToUpper() != "POST"){            
                
return;}          
           
var allowAnonymous = HasAttribute(filterContext, typeof(AllowAnonymousAttribute));  
           
if (allowAnonymous){              
               
return;}          
           
var bypass = HasAttribute(filterContext, typeof(BypassCsrfValidationAttribute));  
           
if (bypass){                return;}          
           
if (filterContext.HttpContext.Request.IsAjaxRequest()){          
             
var antiForgeryCookie = request.Cookies[AntiForgeryConfig.CookieName];        
        
var cookieValue = antiForgeryCookie != null ? antiForgeryCookie.Value : null;              
           
var formToken = request.Headers["__RequestVerificationToken"];AntiForgery.Validate(cookieValue, formToken);}    
           
else{AntiForgery.Validate();}}        

private static bool HasAttribute(AuthorizationContext filterContext, Type attributeType){          
  
return filterContext.ActionDescriptor.IsDefined(attributeType, true) ||filterContext.ActionDescriptor.ControllerDescriptor.IsDefined(attributeType, true);}        

private static void GenerateUserContext(AuthorizationContext filterContext){          
  
var formsIdentity = filterContext.HttpContext.User.Identity as FormsIdentity;    
        
if (formsIdentity == null || string.IsNullOrWhiteSpace(formsIdentity.Name)){UserContext.Current = null;                return;}UserContext.Current = new WebUserContext(formsIdentity.Name);}}


   

  然后在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


.NET社区新闻,深度好文,微信中搜索dotNET跨平台或扫描二维码关注

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/329019.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java NIO系列教程(四) Scatter/Gather

转载自 Java NIO系列教程&#xff08;四&#xff09; Scatter/Gather译文地址 作者&#xff1a;Jakob Jenkov 译者&#xff1a;郭蕾 Java NIO开始支持scatter/gather&#xff0c;scatter/gather用于描述从Channel&#xff08;译者注&#xff1a;Channel在中文经常翻译为…

Linux获取本机hostname函数,Linux下获得主机与域名-gethostbyname和gethostbyaddr

1.数据结构hostent和servent:struct hostent{char *h_name;/* official domain name of host */char **h_aliases;/* null-terminated array of domain names */int h_addrtype;/* host address type (AF_INET) */int h_length;/* length of an address, in bytes */char **h_a…

ASP.NET Core Docker部署

前言 在前面文章中&#xff0c;介绍了 ASP.NET Core在 macOS&#xff0c;Linux 上基于Nginx和Jexus的发布和部署&#xff0c;本篇文章主要是如何在Docker容器中运行ASP.NET Core应用程序。 Asp.Net Core 发布和部署&#xff08; MacOS Linux Nginx &#xff09; Asp.Net Core…

Java NIO系列教程(五) 通道之间的数据传输

转载自 Java NIO系列教程&#xff08;五&#xff09; 通道之间的数据传输译文地址 作者&#xff1a;Jakob Jenkov 译者&#xff1a;郭蕾 校对&#xff1a;周泰 在Java NIO中&#xff0c;如果两个通道中有一个是FileChannel&#xff0c;那你可以直接将数据从一个chan…

linux下单点故障的软件,KeepAlived防止单点故障

负载均衡器里面&#xff0c;HAProxy比较出名了&#xff0c;但是如果只用一台HAProxy,则会出现单点故障。这个时候&#xff0c;该KeepAlived出马了。环境介绍OS: Ubuntu 12.04虚拟IP: 192.168.56.2KeepAlived HaProxy : 192.168.56.101 (master)KeepAlived HaProxy : 192.168.…

当当网上书店购物车——JS源码

$(function($){//根据您挑选的商品&#xff0c;当当为您推荐部分的显示和隐藏$("#shopping_commend_arrow").click(function(){if($("#shopping_commend_sort").css("display")"none"){$(this).attr("src","images/sho…

开放重定向(Open Redirection)

简介 那些通过请求&#xff08;如查询字符串和表单数据&#xff09;指定重定向URL的Web程序可能会被篡改&#xff0c;而把用户重定向到外部的恶意URL。这种篡改就被称为开发重定向攻击。 场景分析 假设有一个正规网站http://nerddinner.com/&#xff0c;还有一个恶意网站或钓鱼…

linux ssh密钥对,Mac使用ssh密钥登录Linux

ssh登录Linux通常有两种方法&#xff1a;用户名密码登录、用户名密钥登录&#xff1b;使用用户名密码登录每次都要输入密码&#xff0c;相当麻烦&#xff0c;而使用用户名密钥登录则可以避免这个问题。创建密钥对文件打开本地终端&#xff0c;执行 ssh-keygen 命令创建密钥对&a…

ASP.NET Core quot;完整发布,自带运行时quot; 到jexus

一、阅读前须知 1.使用 jexus整合asp.net core的优点&#xff1a; 1)支持多站点&#xff0c;同一端口可以同时支持任何多的asp.net core应用程序&#xff1b; 2)应用程序启动、停止、重启与站点的启动、停止、重启等操作一致&#xff0c;无需手工管理asp.net core应用程序…

当当网头部和尾部——源码

头部&#xff1a; <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns"http://www.w3.org/1999/xhtml"> <head><meta http-equiv&…

linux 文档操作,Linux学习之文档操作

8种机械键盘轴体对比本人程序员&#xff0c;要买一个写代码的键盘&#xff0c;请问红轴和茶轴怎么选&#xff1f;The Linux Command Line 学习翻译mkdirThe mkdir command is used to create directories.It works like this:mkdir命令是用来创建目录的&#xff0c;这样使用&am…

过多提交(Over Posting)

简介 过多提交的内容相对比较简单&#xff0c;因此&#xff0c;我只打算把原文中的一些关键信息翻译一下。原文链接如下&#xff1a; http://www.asp.net/mvc/overview/getting-started/getting-started-with-ef-using-mvc/implementing-basic-crud-functionality-with-the-ent…

关于ByteBuffer使用解释

转载自 关于ByteBuffer使用解释之前看过相关的ByteBuffer的使用&#xff0c;但是问题是那时还年轻&#xff0c;所以现在有点老了&#xff0c;因此&#xff0c;忘记了&#xff0c;所以决心看源代码了解一番----故作此篇文章。查看ByteBuffer的API&#xff0c;看的我是一头雾水&…

linux 命令 抛后台,在后台运行Linux命令的方法

通常情况下&#xff0c;当你在终端运行一个命令&#xff0c;在你开始输入另外一个命令之前&#xff0c;你必须等待当前命令运行结束。这被称为在前台运行命令&#xff0c;或者前台进程。当一个进程在前台运行时&#xff0c;它占用了你的 shell&#xff0c;并且你可以通过输入设…

.Net 分布式云平台基础服务建设说明概要

1&#xff09; 背景 建设云平台的基础框架&#xff0c;用于支持各类云服务的业务的构建及发展。 2&#xff09; 基础服务 根据目前对业务的理解和发展方向&#xff0c;总结抽象出以下几个基础服务&#xff0c;如图所示 3&#xff09; 概要说明 基础服务的发展会根据业务的发…

当当网头部和尾部——CSS源码

头部 charset "utf-8"; /* CSS Document *//*网页头部导航样式开始*/ .header_top,.header_middle,.header_search{margin-left:auto;margin-right:auto;width:955px;clear:both; } .header_top{border:solid 1px #999;background-image:url(../images/dd_header_bg.…

全球首发免费的MySql for Entity Framework Core

Background 一时兴起&#xff0c;想实现.NET Core下的MySQL ADO层和与Entity Framework Core对接&#xff0c;同时也是非常鄙视某厂商借着目前Oracle官方没有对.NET Core进行支持的时机捞金&#xff0c;然后也是为了学习一下Microsoft的Entity Framework原理&#xff0c;于是这…

麒麟linux百度云,百度网盘 Linux 版发布,搭配优麒麟运行更完美!

百度网盘 Linux 版发布&#xff0c;搭配优麒麟运行更完美!Linuxer 2019年6月21日 暂无评论 阅读 3,363 次近日&#xff0c;百度官宣推出了百度网盘 Linux 版(在此之前已发布 Windows、Android、iPhone、iPad、Windows Phone、Mac 等多个平台的版本)&#xff0c;版本号为 Linux …

学习笔记之ByteBuffer使用和实现以及文件内存映射

转载自 学习笔记之ByteBuffer使用和实现以及文件内存映射ByteBuffer和CharBuffer等其它Buffer的直接子类一样&#xff0c;顾名思义&#xff0c;就是存取字节的Buffer。很多数据最终在和底层交互上都是使用了字节&#xff0c;而更大的数据是由字节组合而成。谈到字节的组合&…

当当网上书店头部和尾部——JS源码

头部 $(function($){ // οnmοuseοver"myddang_show(dd_menu_top_down)" οnmοuseοut"myddang_hidden(dd_menu_top_down)"//下拉菜单$("#menu").mouseenter(function(){$("#dd_menu_top_down").slideDown(1000);}).mouseleave(…