浅谈ASP.NET MVC 防止跨站请求伪造(CSRF)攻击的实现方法
 更新时间:2018年01月05日 10:23:29   作者:叶长种  

下面小编就为大家分享一篇浅谈ASP.NET MVC 防止跨站请求伪造(CSRF)攻击的实现方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧

在HTTP POST请求中,我们多次在View和Controller中看下如下代码:
1.View中调用了Html.AntiForgeryToken()。
2.Controller中的方法添加了[ValidateAntiForgeryToken]注解。
这样看似一对的写法其实是为了避免引入跨站请求伪造(CSRF)攻击。
这种攻击形式大概在2001年才为人们所认知,2006年美国在线影片租赁网站Netflix爆出多个CSRF漏洞,2008年流行的视频网址YouTube受到CSRF攻击,同年墨西哥一家银行客户受到CSRF攻击,杀毒厂商McAfee也曾爆出CSRF攻击(引自wikipedia)。
之所以很多大型网址也遭遇CSRF攻击,是因为CSRF攻击本身的流程就比较长,很多开发人员可能在几年的时间都没遇到CSRF攻击,因此对CSRF的认知比较模糊,没有引起足够的重视。
CSRF攻击的模拟示例
我们这里将通过一个模拟的示例,讲解CSRF的攻击原理,然后再回过头来看下MVC提供的安全策略。
看似安全的银行转账页面
假设我们是银行的Web开发人员,现在需要编写一个转账页面,客户登录后在此输入对方的账号和转出的金额,即可实现转账:

[Authorize]
public ActionResult TransferMoney()
{
return View();
}
[HttpPost]
[Authorize]
public ActionResult TransferMoney(string ToAccount, int Money)
{
// 这里放置转账业务代码
ViewBag.ToAccount = ToAccount;
ViewBag.Money = Money;
return View();
}

由于这个过程需要身份验证,所以我们为TransferMoney的两个操作方法都加上了注解[Authorize],以阻止匿名用户的访问。
如果直接访问localhost:55654/Home/TransferMoney,会跳转到登录页面:

登录后,来到转账页面,我们看下转账的视图代码:

@{
ViewBag.Title = “Transfer Money”;
}

<h2>Transfer Money</h2>

@if (ViewBag.ToAccount == null)
{
using (Html.BeginForm())
{
<input type=”text” name=”ToAccount” />
<input type=”text” name=”Money” />
<input type=”submit” value=”转账” />
}
}
else
{
@:您已经向账号 [@ViewBag.ToAccount] 转入 [@ViewBag.Money] 元!
}

视图代码中有一个逻辑判断,根据ViewBag.ToAccount是否为空来显示不同内容:
1.ViewBag.ToAccount为空,则表明是页面访问。
2.ViewBag.ToAccount不为空,则为转账成功,需要显示转账成功的提示信息。

来看下页面运行效果:

功能完成!看起来没有任何问题,但是这里却又一个CSRF漏洞,隐蔽而难于发现。
我是Hacker,Show me the money
这里就有两个角色,银行的某个客户A,HackerB。
HackerB发现了银行的这个漏洞,就写了两个简单的页面,页面一(click_me_please.html):

<!DOCTYPE html>
<html>
<head>
<meta http-equiv=”Content-Type” content=”text/html;charset=utf-8″ />
</head>
<body>

哈哈,逗你玩的!

<iframe frameborder=”0″
style=”display:none;” src=”./click_me_please_iframe.html”></iframe>

</body>
</html>

第一个页面仅包含了一个隐藏的iframe标签,指向第二个页面(click_me_please_iframe.html):

<!DOCTYPE html>
<html>
<head>
<meta http-equiv=”Content-Type” content=”text/html;charset=utf-8″ />
</head>
<body onload=”document.getElementById(‘myform1’).submit();”>

<form method=”POST” id=”myform1″
action=”localhost:55654/Home/TransferMoney”>
<input type=”hidden” name=”ToAccount” value=”999999999″>
<input type=”hidden” name=”Money” value=”3000″>
</form>

</body>
</html>

第二个页面放置了一个form标签,并在里面放置了Hacker自己的银行账号和转账金额,在页面打开时提交表单(body的onload属性)。
现在Hacker把这两个页面放到公网:
fineui.com/demo_mvc/csrf/click_me_please.html
然后批量向用户发送带有攻击链接的邮件,而银行的客户A刚好登录了银行系统,并且手贱点击了这个链接:

然后你将看到这个页面:

你可能会在心里想,谁这么无聊,然后郁闷的关闭了这个页面。之后客户A会更加郁闷,因为HackerB的银行账号[999999999]已经成功多了3000块钱!
到底怎么转账的,不是有身份验证吗
是的。转账的确是需要身份验证,现在的问题是你登录了银行系统,已经完成了身份验证,并且在浏览器新的Tab中打开了Hacker的链接,我们来看下到底发生了什么:

这里有三个HTTP请求,第一个就是[逗你玩]页面,第二个是里面的IFrame页面,第三个是IFrame加载完毕后发起的POST请求,也就是具体的转账页面。因为IFrame是隐藏的,所以用户并不知道发生了什么。
我们来具体看下第三个请求:

明显这次转账是成功的,并且Cookie中带上了用户身份验证信息,所有后台根本不知道这次请求是来自Hacker的页面,转账成功的返回内容:

如何阻止CSRF攻击
从上面的实例我们可以看出,CSRF源于表单身份验证的实现机制。
由于HTTP本身是无状态的,也就是说每一次请求对于Web服务器来说都是全新的,服务器不知道之前请求的任何状态,而身份验证需要我们在第二次访问时知道是否登录的状态(不可能每次请求都验证账号密码),这本身就是一种矛盾!
解决这个矛盾的办法就是Cookie,Cookie可以在浏览器中保存少量信息,所以Forms Authentication就用Cookie来保存加密过的身份信息。而Cookie中保存的全部值在每次HTTP请求中(不管是GET还是POST,也不管是静态资源还是动态资源)都会被发送到服务器,这也就给CSRF以可乘之机。
所以,CSRF的根源在于服务器可以从Cookie中获知身份验证信息,而无法得知本次HTTP请求是否真的是用户发起的。
Referer验证
Referer是HTTP请求头信息中的一部分,每当浏览器向服务器发送请求时,都会附带上Referer信息,表明当前发起请求的页面地址。
一个正常的转账请求,我们可以看到Referer和浏览器地址栏是一致的:

我们再来看下刚才的Hacker页面:

可以看到Referer的内容和当前发起请求的页面地址一样,注意对比:
1.浏览器网址:click_me_please.html
2.HTTP请求地址:Home/TransferMoney
3.Referer:click_me_please_iframe.html,注意这个是发起请求的页面,而不一定就是浏览器地址栏显示的网址。
基于这个原理,我们可以简单的对转账的POST请求进行Referer验证:

[HttpPost]
[Authorize]
public ActionResult TransferMoney(string ToAccount, int Money)
{
if(Request.Url.Host != Request.UrlReferrer.Host)
{
throw new Exception(“Referrer validate fail!”);
}

// 这里放置转账业务代码

ViewBag.ToAccount = ToAccount;
ViewBag.Money = Money;
return View();
}

此时访问fineui.com/demo_mvc/csrf/click_me_please.html,恶意转账失败:

MVC默认支持的CSRF验证
MVC默认提供的CSRF验证方式更加彻底,它通过验证当前请求是否真的来自用户的操作。
在视图页面,表单内部增加对Html.AntiForgeryToken函数的调用:

@if (ViewBag.ToAccount == null)
{
using (Html.BeginForm())
{
@Html.AntiForgeryToken()
<input type=”text” name=”ToAccount” />
<input type=”text” name=”Money” />
<input type=”submit” value=”转账” />
}
}
else
{
@:您已经向账号 [@ViewBag.ToAccount] 转入 [@ViewBag.Money] 元!
}

这会在表单标签里面和Cookie中分别生成一个名为__RequestVerificationToken 的Token:

然后添加[ValidateAntiForgeryToken]注解到控制器方法中:

[HttpPost]
[Authorize]
[ValidateAntiForgeryToken]
public ActionResult TransferMoney(string ToAccount, int Money)
{
// 这里放置转账业务代码
ViewBag.ToAccount = ToAccount;
ViewBag.Money = Money;
return View();
}

在服务器端,会验证这两个Token是否一致(不是相等),如果不一致就会报错。
下面手工修改表单中这个隐藏字段的值,来看下错误提示:

类似的道理,运行Hacker页面fineui.com/demo_mvc/csrf/click_me_please.html,恶意转账失败:

此时,虽然Cookie中的__RequestVerificationToken提交到了后台,但是Hacker无法得知表单字段中的__RequestVerificationToken值,所以转账失败。
以上这篇浅谈ASP.NET MVC 防止跨站请求伪造(CSRF)攻击的实现方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持华域联盟。

您可能感兴趣的文章:CSRF在ASP.NET Core中的处理方法详解

ASP.NET
MVC
跨站请求
伪造
CSRF

相关文章
蛇年多屏图片切换(可添加图片链接以及编辑标题)朋友要求,做一个多屏图片切换效果可添加图片,链接以及标题,前台不必在每次更新时,去修改前台代码 2013-01-01
Image显示服务器上任意绝对路径下的图片(采用二进制流实现)有这样一个需求:数据库中存储的是照片所在的绝对路径(可以不在系统所在路径下),Image控件动态加载路径下的图片,另类实现方法,感兴趣的朋友可以参考下,或许本文对你学习二进制流有所帮助 2013-02-02
详解ASP.NET Core 中的框架级依赖注入本篇文章主要介绍了详解ASP.NET Core 中的框架级依赖注入,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
2017-10-10
详解在.net中读写config文件的各种方法本篇文章主要介绍了在.net中读写config文件的各种方法,详细的介绍各种配置文件的读写操作,具有一定的参考价值,有兴趣的可以了解一下。
2016-12-12
ASP.NET下上传图片到数据库,并且读出图片的代码(详细版)上传图片到数据库,从数据库的创建到数据库中图片的现实都给出了具体的代码,因为asp.net版本的问题,大家可能需要稍微修改下。 2010-07-07
Asp.net之数据过滤浅析在B/S开发中,安全性能应该是需要注意的,很多年前有位高人说过,凡是来自客户端的数据,都是不可信的,今天开发时,把数据过滤的方法重写了一下,所以忍不住想拿出来讨论下,欢迎大家指点 2011-11-11
如何在 .NET 中使用 Flurl 高效处理Http请求这篇文章主要介绍了如何在 .NET 中使用 Flurl 高效处理Http请求,帮助大家更好的理解和学习使用.net技术,感兴趣的朋友可以了解下 2021-05-05
C#下载文件(TransmitFile/WriteFile/流方式)实例介绍C#下载文件想必很多业内人士都不会陌生吧,C#下载文件方法很多,本文整理了一些,可供大家参考,感兴趣的你可以研究下,或许本文所提供的知识点对你有所帮助 2013-02-02
asp.net js模拟Button点击事件asp.net 中用js模拟Button点击事件,大家可以看下。
2009-08-08
ASP.NET Core 配置和使用环境变量的实现这篇文章主要介绍了ASP.NET Core 配置和使用环境变量的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧 2020-08-08

最新评论

声明:本站(华域联盟www.cnhackhy.com)所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。