[转]浅谈CSRF攻击方式

在CSDN中看到对CSRF攻击的原理及防护文章,讲解浅显易懂,特转之:

来源:http://blog.csdn.net/fationyyk/article/details/50833620

 

一.CSRF是什么?

  CSRF(Cross-site request forgery),中文名称:跨站请求伪造,也被称为:one click attack/session riding,缩写为:CSRF/XSRF。

二.CSRF可以做什么?

  你这可以这么理解CSRF攻击:攻击者盗用了你的身份,以你的名义发送恶意请求。CSRF能够做的事情包括:以你名义发送邮件,发消息,盗取你的账号,甚至于购买商品,虚拟货币转账……造成的问题包括:个人隐私泄露以及财产安全。

三.CSRF漏洞现状

  CSRF这种攻击方式在2000年已经被国外的安全人员提出,但在国内,直到06年才开始被关注,08年,国内外的多个大型社区和交互网站分别 爆出CSRF漏洞,如:NYTimes.com(纽约时报)、Metafilter(一个大型的BLOG网站),YouTube和百度HI……而现在,互 联网上的许多站点仍对此毫无防备,以至于安全业界称CSRF为“沉睡的巨人”。

四.CSRF的原理

下图简单阐述了CSRF攻击的思想:
这里写图片描述
从上图可以看出,要完成一次CSRF攻击,受害者必须依次完成两个步骤:

  1.登录受信任网站A,并在本地生成Cookie。

  2.在不登出A的情况下,访问危险网站B。

  看到这里,你也许会说:“如果我不满足以上两个条件中的一个,我就不会受到CSRF的攻击”。是的,确实如此,但你不能保证以下情况不会发生:

  1.你不能保证你登录了一个网站后,不再打开一个tab页面并访问另外的网站。

  2.你不能保证你关闭浏览器了后,你本地的Cookie立刻过期,你上次的会话已经结束。(事实上,关闭浏览器不能结束一个会话,但大多数人都会错误的认为关闭浏览器就等于退出登录/结束会话了……)

  3.上图中所谓的攻击网站,可能是一个存在其他漏洞的可信任的经常被人访问的网站。
  
  上面大概地讲了一下CSRF攻击的思想,下面我将用几个例子详细说说具体的CSRF攻击,这里我以一个银行转账的操作作为例子(仅仅是例子,真实的银行网站没这么傻:>)

  示例1:

  银行网站A,它以GET请求来完成银行转账的操作,如:http://www.mybank.com/Transfer.php?toBankId=11&money=1000
危险网站B,它里面有一段HTML的代码如下:

<code class="hljs xml has-numbering"><span class="hljs-tag"><<span class="hljs-title">img <span class="hljs-attribute">src=<span class="hljs-value">http://<span class="hljs-attribute">www.mybank.com/<span class="hljs-attribute">Transfer.php?<span class="hljs-attribute">toBankId=<span class="hljs-value">11&money=1000></span></span></span></span></span></span></span></span></code>

首先,你登录了银行网站A,然后访问危险网站B,噢,这时你会发现你的银行账户少了1000块……

  为什么会这样呢?原因是银行网站A违反了HTTP规范,使用GET请求更新资源。在访问危险网站B的之前,你已经登录了银行网站A,而B中的以GET的方式请求第三方资源(这里的第三方就是指银行网站了,原本这是一个合法的请求,但这里被不法分子利用了),所以你的浏览器会带上你的银行网站A的Cookie发出Get请求,去获取资源“http://www.mybank.com/Transfer.php?toBankId=11&money=1000”,结果银行网站服务器收到请求后,认为这是一个更新资源操作(转账操作),所以就立刻进行转账操作……

  示例2:

  为了杜绝上面的问题,银行决定改用POST请求完成转账操作。

  银行网站A的WEB表单如下:  

<span style="background-color: #888888; color: #ff0000;"><code class="hljs xml has-numbering">  <span class="hljs-tag"><<span class="hljs-title">form <span class="hljs-attribute">action=<span class="hljs-value">"Transfer.php" <span class="hljs-attribute">method=<span class="hljs-value">"POST">
    <span class="hljs-tag"><<span class="hljs-title">p>ToBankId: <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">"text" <span class="hljs-attribute">name=<span class="hljs-value">"toBankId" /><span class="hljs-tag"></<span class="hljs-title">p>
    <span class="hljs-tag"><<span class="hljs-title">p>Money: <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">"text" <span class="hljs-attribute">name=<span class="hljs-value">"money" /><span class="hljs-tag"></<span class="hljs-title">p>
    <span class="hljs-tag"><<span class="hljs-title">p><span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">"submit" <span class="hljs-attribute">value=<span class="hljs-value">"Transfer" /><span class="hljs-tag"></<span class="hljs-title">p>
  <span class="hljs-tag"></<span class="hljs-title">form></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></span>

  后台处理页面Transfer.php如下:

<code class="hljs xml has-numbering">   <span class="php"><span class="hljs-preprocessor"><?php
    session_start();
    <span class="hljs-keyword">if (<span class="hljs-keyword">isset(<span class="hljs-variable">$_REQUEST[<span class="hljs-string">'toBankId'] && <span class="hljs-keyword">isset(<span class="hljs-variable">$_REQUEST[<span class="hljs-string">'money']))
    {
        buy_stocks(<span class="hljs-variable">$_REQUEST[<span class="hljs-string">'toBankId'], <span class="hljs-variable">$_REQUEST[<span class="hljs-string">'money']);
    }
  <span class="hljs-preprocessor">?></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code>

危险网站B,仍然只是包含那句HTML代码:
  

<code class="hljs xml has-numbering"><span class="hljs-tag"><<span class="hljs-title">img <span class="hljs-attribute">src=<span class="hljs-value">http://<span class="hljs-attribute">www.mybank.com/<span class="hljs-attribute">Transfer.php?<span class="hljs-attribute">toBankId=<span class="hljs-value">11&money=1000></span></span></span></span></span></span></span></span></code>

  和示例1中的操作一样,你首先登录了银行网站A,然后访问危险网站B,结果…..和示例1一样,你再次没了1000块~T_T,这次事故的原因是:银行后台使用了REQUEST_REQUEST既可以获取GET请求的数据,也可以获取POST请求的数据,这就造成了在后台处理程序无法区分这到底是GET请求的数据还是POST请求的数据。在PHP中,可以使用GET_POST分别获取GET请求和POST请求的数据。在JAVA中,用于获取请求数据request一样存在不能区分GET请求数据和POST数据的问题。

  示例3:

  经过前面2个惨痛的教训,银行决定把获取请求数据的方法也改了,改用$_POST,只获取POST请求的数据,后台处理页面Transfer.php代码如下:

<code class="hljs xml has-numbering"> <span class="php"><span class="hljs-preprocessor"><?php
    session_start();
    <span class="hljs-keyword">if (<span class="hljs-keyword">isset(<span class="hljs-variable">$_POST[<span class="hljs-string">'toBankId'] && <span class="hljs-keyword">isset(<span class="hljs-variable">$_POST[<span class="hljs-string">'money']))
    {
        buy_stocks(<span class="hljs-variable">$_POST[<span class="hljs-string">'toBankId'], <span class="hljs-variable">$_POST[<span class="hljs-string">'money']);
    }
  <span class="hljs-preprocessor">?></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code>

然而,危险网站B与时俱进,它改了一下代码:

<code class="hljs xml has-numbering"><span class="hljs-tag"><<span class="hljs-title">html>
  <span class="hljs-tag"><<span class="hljs-title">head>
    <span class="hljs-tag"><<span class="hljs-title">script <span class="hljs-attribute">type=<span class="hljs-value">"text/javascript"><span class="javascript">
      <span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">steal<span class="hljs-params">()
      {
               iframe = document.frames[<span class="hljs-string">"steal"];
               iframe.document.Submit(<span class="hljs-string">"transfer");
      }
    <span class="hljs-tag"></<span class="hljs-title">script>
  <span class="hljs-tag"></<span class="hljs-title">head>

  <span class="hljs-tag"><<span class="hljs-title">body <span class="hljs-attribute">οnlοad=<span class="hljs-value">"steal()">
    <span class="hljs-tag"><<span class="hljs-title">iframe <span class="hljs-attribute">name=<span class="hljs-value">"steal" <span class="hljs-attribute">display=<span class="hljs-value">"none">
      <span class="hljs-tag"><<span class="hljs-title">form <span class="hljs-attribute">method=<span class="hljs-value">"POST" <span class="hljs-attribute">name=<span class="hljs-value">"transfer" <span class="hljs-attribute">action=<span class="hljs-value">"http://www.myBank.com/Transfer.php">
        <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">"hidden" <span class="hljs-attribute">name=<span class="hljs-value">"toBankId" <span class="hljs-attribute">value=<span class="hljs-value">"11">
        <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">"hidden" <span class="hljs-attribute">name=<span class="hljs-value">"money" <span class="hljs-attribute">value=<span class="hljs-value">"1000">
      <span class="hljs-tag"></<span class="hljs-title">form>
    <span class="hljs-tag"></<span class="hljs-title">iframe>
  <span class="hljs-tag"></<span class="hljs-title">body>
<span class="hljs-tag"></<span class="hljs-title">html></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code>

如果用户仍是继续上面的操作,很不幸,结果将会是再次不见1000块……因为这里危险网站B暗地里发送了POST请求到银行!

  总结一下上面3个例子,CSRF主要的攻击模式基本上是以上的3种,其中以第1,2种最为严重,因为触发条件很简单,一个就可以了,而第3种比较麻烦,需要使用JavaScript,所以使用的机会会比前面的少很多,但无论是哪种情况,只要触发了CSRF攻击,后果都有可能很严重。
  理解上面的3种攻击模式,其实可以看出,CSRF攻击是源于WEB的隐式身份验证机制!WEB的身份验证机制虽然可以保证一个请求是来自于某个用户的浏览器,但却无法保证该请求是用户批准发送的!

五.CSRF的防御

  我总结了一下看到的资料,CSRF的防御可以从服务端和客户端两方面着手,防御效果是从服务端着手效果比较好,现在一般的CSRF防御也都在服务端进行。

  1.服务端进行CSRF防御

  服务端的CSRF方式方法很多样,但总的思想都是一致的,就是在客户端页面增加伪随机数。

<code class="hljs xml has-numbering"> <span class="php"><span class="hljs-preprocessor"><?php
    <span class="hljs-comment">//构造加密的Cookie信息
    <span class="hljs-variable">$value = “DefenseSCRF”;
    setcookie(”cookie”, <span class="hljs-variable">$value, time()+<span class="hljs-number">3600);
  <span class="hljs-preprocessor">?></span></span></span></span></span></span></span></code>

在表单里增加Hash值,以认证这确实是用户发送的请求。

<code class="hljs xml has-numbering"> <span class="php"><span class="hljs-preprocessor"><?php
    <span class="hljs-variable">$hash = md5(<span class="hljs-variable">$_COOKIE[<span class="hljs-string">'cookie']);
  <span class="hljs-preprocessor">?>
  <span class="hljs-tag"><<span class="hljs-title">form <span class="hljs-attribute">method=<span class="hljs-value">”POST” <span class="hljs-attribute">action=<span class="hljs-value">”transfer.php”>
    <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">”text” <span class="hljs-attribute">name=<span class="hljs-value">”toBankId”>
    <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">”text” <span class="hljs-attribute">name=<span class="hljs-value">”money”>
    <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">”hidden” <span class="hljs-attribute">name=<span class="hljs-value">”hash” <span class="hljs-attribute">value=<span class="hljs-value">”<?=$hash;?>”>
    <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">”submit” <span class="hljs-attribute">name=<span class="hljs-value">”submit” <span class="hljs-attribute">value=<span class="hljs-value">”Submit”>
  <span class="hljs-tag"></<span class="hljs-title">form></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code>

然后在服务器端进行Hash值验证

<code class="hljs r has-numbering">    <?php
        <span class="hljs-keyword">if(isset($_POST[<span class="hljs-string">'check'])) {
             $hash = md5($_COOKIE[<span class="hljs-string">'cookie']);
             <span class="hljs-keyword">if($_POST[<span class="hljs-string">'check'] == $hash) {
                  doJob();
             } <span class="hljs-keyword">else {
        //<span class="hljs-keyword">...
             }
        } <span class="hljs-keyword">else {
      //<span class="hljs-keyword">...
        }
      ?></span></span></span></span></span></span></span></span></span></code>

这个方法个人觉得已经可以杜绝99%的CSRF攻击了,那还有1%呢….由于用户的Cookie很容易由于网站的XSS漏洞而被盗取,这就另外的1%。一般的攻击者看到有需要算Hash值,基本都会放弃了,某些除外,所以如果需要100%的杜绝,这个不是最好的方法。
  (2).验证码

  这个方案的思路是:每次的用户提交都需要用户在表单中填写一个图片上的随机字符串,厄….这个方案可以完全解决CSRF,但个人觉得在易用性方面似乎不是太好,还有听闻是验证码图片的使用涉及了一个被称为MHTML的Bug,可能在某些版本的微软IE中受影响。

  (3).One-Time Tokens(不同的表单包含一个不同的伪随机值)

  在实现One-Time Tokens时,需要注意一点:就是“并行会话的兼容”。如果用户在一个站点上同时打开了两个不同的表单,CSRF保护措施不应该影响到他对任何表单的提交。考虑一下如果每次表单被装入时站点生成一个伪随机值来覆盖以前的伪随机值将会发生什么情况:用户只能成功地提交他最后打开的表单,因为所有其他的表单都含有非法的伪随机值。必须小心操作以确保CSRF保护措施不会影响选项卡式的浏览或者利用多个浏览器窗口浏览一个站点。

  以下我的实现:

  1).先是令牌生成函数(gen_token()):

<code class="hljs xml has-numbering"><span class="php"><span class="hljs-preprocessor"><?php
     <span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">gen_token<span class="hljs-params">() {
     <span class="hljs-comment">//这里我是贪方便,实际上单使用Rand()得出的随机数作为令牌,也是不安全的。
    <span class="hljs-comment">//这个可以参考我写的Findbugs笔记中的《Random object created and used only once》
          <span class="hljs-variable">$token = md5(uniqid(rand(), <span class="hljs-keyword">true));
          <span class="hljs-keyword">return <span class="hljs-variable">$token;
     }</span></span></span></span></span></span></span></span></span></span></span></span></code>

 2).然后是Session令牌生成函数(gen_stoken()):

<code class="hljs xml has-numbering"> <span class="php"><span class="hljs-preprocessor"><?php
       <span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">gen_stoken<span class="hljs-params">() {
      <span class="hljs-variable">$pToken = <span class="hljs-string">"";
      <span class="hljs-keyword">if(<span class="hljs-variable">$_SESSION[STOKEN_NAME]  == <span class="hljs-variable">$pToken){
        <span class="hljs-comment">//没有值,赋新值
        <span class="hljs-variable">$_SESSION[STOKEN_NAME] = gen_token();
      }   
      <span class="hljs-keyword">else{
        <span class="hljs-comment">//继续使用旧的值
      }
       }
     <span class="hljs-preprocessor">?></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code>

3).WEB表单生成隐藏输入域的函数: 

<code class="hljs xml has-numbering"><span class="php"><span class="hljs-preprocessor"><?php
       <span class="hljs-function"><span class="hljs-keyword">function <span class="hljs-title">gen_input<span class="hljs-params">() {
            gen_stoken();
            <span class="hljs-keyword">echo “<input type=\”hidden\” name=\”<span class="hljs-string">" . FTOKEN_NAME . “\”
                 value=\”" . <span class="hljs-variable">$_SESSION[STOKEN_NAME] . “\”> “;
       }
     <span class="hljs-preprocessor">?></span></span></span></span></span></span></span></span></span></span></code>

4).WEB表单结构:

<code class="hljs xml has-numbering"> <span class="php"><span class="hljs-preprocessor"><?php
          session_start();
          <span class="hljs-keyword">include(”functions.php”);
     <span class="hljs-preprocessor">?>
     <span class="hljs-tag"><<span class="hljs-title">form <span class="hljs-attribute">method=<span class="hljs-value">”POST” <span class="hljs-attribute">action=<span class="hljs-value">”transfer.php”>
          <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">”text” <span class="hljs-attribute">name=<span class="hljs-value">”toBankId”>
          <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">”text” <span class="hljs-attribute">name=<span class="hljs-value">”money”>
          <span class="php"><span class="hljs-preprocessor"><? gen_input(); <span class="hljs-preprocessor">?>
          <span class="hljs-tag"><<span class="hljs-title">input <span class="hljs-attribute">type=<span class="hljs-value">”submit” <span class="hljs-attribute">name=<span class="hljs-value">”submit” <span class="hljs-attribute">value=<span class="hljs-value">”Submit”>
     <span class="hljs-tag"></<span class="hljs-title">FORM>   </span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code>

5).服务端核对令牌:

  这个很简单,这里就不再啰嗦了。

  上面这个其实不完全符合“并行会话的兼容”的规则,大家可以在此基础上修改。

 

 

-------------------

资料结语:

1、对于.net WebForm中防御CSRF攻击是较麻烦的,具体的解决方法可以查看:保护ASP.NET 应用免受 CSRF 攻击

2、如果是.net MVC项目,防止CSRF攻击是非常容易的事,因为在Html帮助类有就提供了:<%=Html.AntiForgeryToken() %>,具体可以查看:【MVC整理】4.Asp.net MVC 如何防止CSRF攻击

 

 

>>小广告: 八爪鱼招标网 大家多给意见哦

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值