CSRF攻击

本文最后更新于 2025年8月25日 晚上

参考:

https://tech.meituan.com/2018/10/11/fe-security-csrf.html
https://juejin.cn/post/6844903638532358151

概念

CSRF(Cross-site request forgery)跨站请求伪造:攻击者利用浏览器在请求时自动带上用户凭证(如 Cookie、Session、HTTP 认证信息)的特性发起冒充用户执行未授权请求的攻击方式。

一个典型的 CSRF 攻击有着如下的流程:

  • 受害者登录 a.com,并保留了登录凭证(Cookie)
  • 攻击者引诱受害者访问了 b.com
  • b.com 向 a.com 发送了一个请求:a.com/act=xx,浏览器会默认携带 a.com 的 Cookie
  • a.com 接收到请求后对请求进行验证,确认是受害者的凭证,误以为是受害者发送的请求并执行
  • 攻击者在受害者不知情的情况下,冒充受害者,让 a.com 执行了攻击者定义的操作,攻击完成

分类

GET 型

  • 原理:利用浏览器自动加载资源的特性(如图片、CSS 等)
  • 特点:
    • 无需用户交互
    • 可嵌入在 <img><a><script> 等标签里
  • 例子:
    1
    <img src="https://bank.com/transfer?account=xiaoming&amount=10&to=hacker">
    在受害者访问含有这个 img 的页面后,浏览器会自动向 https://bank.com/transfer 发出一次 HTTP 请求,要求银行执行小明向黑客的账号转钱的操作。如果小明登录了 bank.com 而且该请求能直接转账就会中招。

POST 型

  • 原理:伪造不可见的表单,通过 JS 自动提交
  • 特点:
    • 无需用户交互
    • 构造隐藏 <form> 表单
  • 例子:
    1
    2
    3
    4
    5
    6
    7
    8
    <form action="http://bank.example/withdraw" method="POST">
    <input type="hidden" name="account" value="xiaoming" />
    <input type="hidden" name="amount" value="10" />
    <input type="hidden" name="to" value="hacker" />
    </form>
    <script>
    document.forms[0].submit();
    </script>
    访问该页面后,表单会自动提交,相当于模拟用户完成了一次 POST 操作。

POST 类型的攻击通常比 GET 要求更加严格一点,但仍并不复杂。任何个人网站、博客、被黑客上传页面的网站都有可能是发起攻击的来源,后端接口不能将安全寄托在仅允许 POST 上面。

点击诱导型

  • 原理:伪造正常交互元素(链接、按钮等),诱骗用户主动触发
  • 特点:
    • 需要用户交互
    • 伪造正常交互元素
  • 例子:
    1
    2
    3
    4
    <form action="https://weibo.com/follow" method="POST">
    <input type="hidden" name="target_id" value="attacker_id" />
    <button>点击抽奖</button>
    </form>
    恶意页面包含隐藏表单,用户点击“抽奖”按钮后自动提交关注请求,利用用户点击诱导链接自动关注攻击者账号

特点

  • 攻击一般发起在第三方网站,而不是被攻击的网站,被攻击的网站无法防止攻击发生
  • 攻击利用受害者在被攻击网站的登录凭证,冒充受害者提交操作,而不是直接窃取数据
  • 整个过程攻击者并不能获取到受害者的登录凭证,仅仅是“冒用”
  • 跨站请求可以用各种方式:图片 URL、超链接、CORS、Form 提交等等,部分请求方式可以直接嵌入在第三方论坛、文章中,难以进行追踪

CSRF 通常是跨域的,因为外域通常更容易被攻击者掌控。但如果本域下有容易被利用的功能,比如可以发图和链接的论坛和评论区,攻击可以直接在本域下进行,而且这种攻击更加危险。

防护

CSRF 通常从第三方网站发起,被攻击的网站无法防止攻击发生,只能通过增强自己网站针对 CSRF 的防护能力来提升安全性。而且攻击者不能获取 Cookie 等敏感信息,只是使用。针对这两点,可以专门制定防护策略,如下:

  • 阻止不明外域的访问
    • 同源检测
    • Cookie Samesite
  • 提交时要求附加本域才能获取的信息
    • CSRF Token
    • 双重 Cookie

同源检测

既然 CSRF 大多来自第三方网站,那么可以直接禁止外域(或者不受信任域名)对网站发起的请求。那么问题来了,如何判断请求是否来自外域?

在 HTTP 协议中,每一个异步请求都会携带两个 Header,用于标记来源域名:Origin Header 和 Referer Header。这两个 Header 在浏览器发起请求时,大多数情况会被自动带上,并且不能由前端自定义内容。 服务器可以通过解析这两个 Header 中的域名,确定请求的来源域。

Origin Header

在部分与 CSRF 有关的请求中,请求的 Header 会携带 Origin 字段。字段包含请求的源(协议 + 域名 + 端口,不包含路径和请求参数)。如果 Origin 存在,直接用 Origin 中的字段确认来源域名就可以。但是 Origin 在以下两种情况下并不存在:

  • IE11 同源策略: IE 11 不会在跨站 CORS 请求添加 Origin 标头,Referer 头仍然是唯一的标识。最根本原因是因为 IE 11 对同源的定义和其他浏览器有不同
  • 302 重定向: 在 302 重定向之后 Origin 不包含在重定向的请求中,因为 Origin 可能会被认为是其他来源的敏感信息。对于 302 重定向的情况来说都是定向到新的服务器上的 URL,因此浏览器不想将 Origin 泄漏到新的服务器上

Referer Header

根据 HTTP 协议,在 HTTP 头中有 Referer 字段 ,记录了 HTTP 请求的完整来源地址(协议 + 域名 + 端口 + 路径 + 请求参数)。对于 Ajax、图片、script 等资源请求,Referer 为发起请求的页面地址;对于页面跳转,Referer 为打开页面历史记录的前一个页面地址。因此使用 Referer 链接中的 Origin 部分可以得知请求的来源域名。

这种方法并非万无一失,Referer 的值是由浏览器提供的,虽然 HTTP 协议有明确的要求,但是每个浏览器对于 Referer 的具体实现可能有差别,并不能保证浏览器自身没有安全漏洞。使用验证 Referer 值的方法,就是把安全性都依赖于第三方(即浏览器)来保障,从理论上来讲,这样并不是很安全。在部分情况下,攻击者可以隐藏,甚至修改自己请求的 Referer。

2014 年,W3C 的 Web 应用安全工作组发布了 Referrer Policy 草案,对浏览器该如何发送 Referer 做了详细的规定。截止现在新版浏览器大部分已经支持了这份草案,终于可以灵活地控制自己网站的 Referer 策略。新版的 Referrer Policy 规定了五种 Referer 策略,之前就存在三种策略。他们的对应关系如下:

策略名称 属性值(新) 属性值(旧)
No Referrer no-Referrer never
No Referrer When Downgrade no-Referrer-when-downgrade default
Origin Only (same or strict) origin origin
Origin When Cross Origin (strict) origin-when-crossorigin -
Unsafe URL unsafe-url always

根据表格,需要把 Referrer Policy 的策略设置成 same-origin:对于同源的链接和引用,会发送 Referer,值为 Host 不带 Path;跨域访问则不携带 Referer。例如:aaa.com 引用 bbb.com 资源,不会发送 Referer。设置 Referrer Policy 的方法有三种:

  • 在 CSP 设置
  • 页面头部增加 meta 标签
  • a 标签增加 Referrer Policy 属性

上面说的比较多,但我们可以知道一个问题:攻击者可以在自己的请求中隐藏 Referer,可以让请求发起的攻击不携带 Referer。另外在以下情况下 Referer 没有或者不可信:

  • IE6、7 下使用 window.location.href=url 进行界面的跳转,会丢失 Referer
  • IE6、7 下使用 window.open,也会缺失 Referer
  • HTTPS 页面跳转到 HTTP 页面,所有浏览器 Referer 都丢失
  • 点击 Flash 上到达另外一个网站的时候,Referer 的情况就比较杂乱,不太可信

无法确认来源

如果 Origin 和 Referer 都不存在,则可以直接进行阻止,特别是如果没有使用随机 CSRF Token 或者双重 Cookie 作为第二次检查。

阻止外域请求

通过对 Header 验证,我们可以知道发起请求的来源域名,这些来源域名可能是网站本域、子域名、有授权的第三方域名,又或者来自不可信的未知域名。知道了请求是来自不可信的域名,直接阻止掉这些的请求,就能防御 CSRF 攻击了吗?答案是不一定!当一个请求是页面请求(比如网站的主页),而来源是搜索引擎的链接(如百度的搜索结果),也会被当成疑似 CSRF 攻击。所以在判断的时候需要过滤掉页面请求情况,通常这种情况下的 Header 符合以下情况:

1
2
Accept: text/html
Method: GET

但相应的,页面请求就暴露在了 CSRF 的攻击范围之中。如果网站中在页面的 GET 请求中对当前用户做了什么操作的话,防范就失效了。例如下面的页面请求:

1
GET https://example.com/addComment?comment=XXX&dest=orderId

这种严格来说并不一定存在 CSRF 攻击的风险,但仍然有很多网站经常把主文档 GET 请求挂上参数来实现产品功能,但是这样做对于自身来说是存在安全风险的。

而且 CSRF 大多情况下来自第三方域名,但并不能排除本域发起。如果攻击者有权限在本域发布评论(链接、图片等,统称 UGC),那么他可以直接在本域发起攻击,这种情况下同源策略无法防护。

综上所述,同源验证是一个相对简单的防范方法,能够防范绝大多数的 CSRF 攻击。但是并不是万无一失,对于安全性要求较高或者有较多用户输入内容的网站,需要对关键的接口做额外的防护措施。

CSRF Token

CSRF 的另一个特征是,攻击者无法直接窃取到用户的信息(Cookie,Header,网站内容等),仅是冒用 Cookie 中的信息。而 CSRF 攻击之所以能够成功,是因为服务器误把攻击者发送的请求当成了用户自己的请求。那么可以要求所有用户请求都携带一个 CSRF 攻击者无法获取到的 Token。服务器通过校验请求是否携带正确的 Token 来把正常的请求和攻击的请求区分开,也可以防范 CSRF 攻击。

原理

CSRF Token 的防护策略分为三个步骤:

  1. 将 CSRF Token 输出到页面中
    首先,用户打开页面的时候,服务器需要给这个用户生成一个 Token,该 Token 通过加密算法对数据进行加密,一般 Token 都包括随机字符串和时间戳的组合,显然在提交时 Token 不能再放在 Cookie 中了,否则又会被攻击者冒用。因此,为了安全起见 Token 最好存在服务器的 Session,之后在每次页面加载时,使用 JS 遍历整个 DOM 树,对于 DOM 中所有的 a 和 form 标签后加入 Token。这样可以解决大部分的请求,但是对于在页面加载后动态生成的 HTML 代码,这种方法就没有作用,需要程序员在编码时手动添加 Token。

  2. 页面提交的请求携带这个 Token
    对于 GET 请求,Token 将附在请求地址之后,这样 URL 就变成 http://url?csrftoken=tokenvalue。 对于 POST 请求,要在 form 的最后加上<input type=”hidden” name=”csrftoken” value=”tokenvalue”/>。这样就把 Token 以参数的形式加入到请求。

  3. 服务器验证 Token 是否正确
    当用户从客户端得到了 Token 再次提交给服务器时,服务器需要判断 Token 的有效性,验证过程是先解密 Token,对比加密字符串以及时间戳,如果加密字符串一致且时间未过期,那么这个 Token 就是有效的。这种方法要比之前检查 Referer 或者 Origin 要安全一些,Token 可以在产生并放于 Session 之中,然后在每次请求时把 Token 从 Session 中拿出,与请求中的 Token 进行比对,但这种方法的比较麻烦的在于如何把 Token 以参数的形式加入请求。

校验

在大型网站使用 Session 存储 CSRF Token 会带来很大的压力。访问不同服务器用不同的 Session。但是现在的大型网站中,我们的服务器通常不止一台,可能是几十台甚至几百台之多,甚至多个机房可能在不同的省份,用户发起的 HTTP 请求通常要经过像 Ngnix 之类的负载均衡器之后再路由到具体的服务器,由于 Session 默认存储在单机服务器内存中,因此在分布式环境下同一个用户发送的多次 HTTP 请求可能会先后落到不同的服务器上,导致后面发起的 HTTP 请求无法拿到之前的 HTTP 请求存储在服务器中的 Session 数据,从而使得 Session 机制在分布式环境下失效,因此在分布式集群中 CSRF Token 需要存储在 Redis 之类的公共存储空间。

由于使用 Session 存储,读取和验证 CSRF Token 会引起比较大的复杂度和性能问题,目前很多网站采用 Encrypted Token Pattern 方式。这种方法算的 Token 是一个计算出来的结果,而非随机生成的字符串。这样在校验时无需再去读取存储的 Token,只用再次计算一次即可。这种 Token 的值通常是使用 UserID、时间戳和随机数,通过加密的方法生成。这样既可以保证分布式服务的 Token 一致,又能保证 Token 不容易被破解。

在 Token 解密成功之后,服务器可以访问解析值,Token 中所包含的 UserID 和时间戳将会被拿来被验证有效性,将 UserID 与当前登录的 UserID 进行比较,并将时间戳与当前时间进行比较。

总结

Token 是一个比较有效的 CSRF 防护方法,只要页面没有 XSS 漏洞泄露 Token,那么接口的 CSRF 攻击就无法成功。但是此方法的实现比较复杂,需要给每个页面都写入 Token(前端无法使用纯静态页面),每一个 Form 及 Ajax 请求都携带这个 Token,后端对每一个接口都进行校验,并保证页面 Token 及请求 Token 一致。这就使得这个防护策略不能在通用的拦截上统一拦截处理,而需要每一个页面和接口都添加对应的输出和校验。这种方法工作量巨大,且有可能遗漏。

验证码和密码其实也可以起到 CSRF Token 的作用哦,而且更安全。所以很多银行等网站会要求已经登录的用户在转账时再次输入密码。

原理

在会话中存储 CSRF Token 比较繁琐,而且不能在通用的拦截上统一处理所有的接口。另外一种防御措施是使用双重提交 Cookie。利用 CSRF 攻击不能获取到用户 Cookie 的特点,我们可以要求 Ajax 和表单请求携带一个 Cookie 中的值。双重 Cookie 采用以下流程:

  • 在用户访问网站页面时,向请求域名注入一个 Cookie,内容为随机字符串(例如 csrfCookie=v8g9e4ksfhw
  • 在前端向后端发起请求时,取出 Cookie,并添加到 URL 的参数中(例如POST https://www.a.com/comment?csrfCookie=v8g9e4ksfhw
  • 后端接口验证 Cookie 中的字段与 URL 参数中的字段是否一致,不一致则拒绝请求

此方法相对于 CSRF Token 简单了许多,可以直接通过前后端拦截的的方法自动化实现。后端校验也更加方便,只需进行请求中字段的对比,不需要再进行查询和存储 Token。但是此方法并没有大规模应用,其在大型网站上的安全性没有 CSRF Token 高。这是因为跨域可能会使前端无法获取 Cookie,可能发生如下情况:

  • 如果用户访问的网站为 www.a.com,但是后端的 API 为 api.a.com。那么在 www.a.com 下,前端拿不到 api.a.com 的 Cookie,也就无法完成双重 Cookie 认证
  • 于是 api.a.com 的认证 Cookie 必须被种在 a.com 下,这样 a.com 的每个子域都可以访问和修改 Cookie
  • 如果某个子域存在漏洞被 XSS 攻击,即使这个子域并没有什么别的信息值得窃取,但攻击者可以获取或者修改该 Cookie,然后再向 www.a.com 发起 CSRF 攻击

总结

优点:

  • 无需使用 Session 和 Token,适用面更广,易于实施
  • Cookie 储存于客户端中,不会给服务器带来压力
  • 相对于 Token,实施成本更低,可以在前后端统一拦截校验,不需要一个个接口和页面添加

缺点:

  • Cookie 中增加了额外的字段
  • 如果有其他漏洞(例如 XSS),攻击者可以窃取 Cookie,那么该防御方式失效
  • 难以做到子域名的隔离
  • 为了确保 Cookie 传输安全,采用这种防御方式的最好确保用整站 HTTPS 的方式,否则也有风险

攻击时必须使用用户已有的登录凭证,被攻击的网站可以限制在非同站发起请求时禁止携带本站的 Cookie。需要在登录设置本站 Cookie 时设置 Samesite 相关属性。详细见:站点匹配 SameSite

如果 SamesiteCookie 被设置为 Strict,那么浏览器在任何跨域请求都不会携带 Cookie,新标签重新打开也不携带,所以 CSRF 攻击基本没有机会。但跳转子域名或者是新标签重新打开刚登陆的网站,之前的 Cookie 都不会存在,尤其是有登录的网站。那么新打开一个标签进入,或者跳转到子域名的网站,都需要重新登录,对于用户来讲,可能体验不会很好。

如果 SamesiteCookie 被设置为 Lax,那么其他网站通过页面跳转过来的时候可以使用 Cookie,可以保障外域连接打开页面时用户的登录状态。但相应的,其安全性也比较低。

防止被利用

前面四个都是被攻击的网站如何做好防护。而非防止攻击的发生,CSRF 的攻击可以来自:

  • 攻击者自己的网站
  • 有文件上传漏洞的网站
  • 第三方论坛等用户内容
  • 被攻击网站自己的评论功能

对于来自攻击者的网站我们无法防护。但对其他情况,有防止自己的网站被利用成为攻击的源头:

  • 严格管理所有的上传接口,防止任何预期之外的上传内容(例如 HTML)
  • 对于用户上传的图片,进行转存或者校验,不直接使用用户填写的图片链接
  • 添加 Header X-Content-Type-Options: nosniff 防止黑客上传 HTML 内容的资源(如图片)被解析为网页
  • 当前用户打开其他用户填的链接时告知风险(这也是很多论坛不允许直接在内容中发布外域链接的原因之一,不仅仅是为了用户留存,也有安全考虑)

其他措施

对于一线的程序员,可以通过各种防护策略来防御 CSRF。对于 QA、SRE、安全负责人,需要更多的措施来提升安全性。

CSRF 测试

CSRFTester 是一款 CSRF 漏洞的测试工具,CSRFTester 工具的测试原理大概是,使用代理抓取我们在浏览器中访问过的所有连接以及所有表单等信息,通过在 CSRFTester 中修改相应的表单等信息,重新提交,相当于一次伪造前端请求,如果修改后的测试请求成功(被网站服务器接受),则说明存在漏洞,此款工具也可以被用来进行 CSRF 攻击。 CSRFTester 使用方法大致分下面几个步骤:

  • 步骤 1:设置浏览器代理
    CSRFTester 默认使用 Localhost 上的端口 8008 作为其代理,如果代理配置成功,CSRFTester 将为浏览器生成的所有后续 HTTP 请求生成调试消息。

  • 步骤 2:使用合法账户访问网站开始测试
    找到一个想要测试的特定业务 Web 页面。找到此页面后,选择 CSRFTester 中的“开始录制”按钮并执行业务功能;完成后,点击 CSRFTester 中的“停止录制”按钮;正常情况下,该软件会全部遍历一遍当前页面的所有请求。

  • 步骤 3:通过 CSRF 修改并伪造请求
    之后会发现软件上有一系列跑出来的记录请求,这些都是浏览器在执行业务功能时生成的所有 GET 或者 POST 请求。通过选择列表中的某一行,可以修改用于执行业务功能的参数,可以通过点击对应的请求修改 query 和 form 等参数。当修改完所有希望诱导用户 form 最终的提交值后,可以选择开始生成 HTML 报告。

  • 步骤 4:拿到结果如有漏洞进行修复
    首先必须选择“报告类型”。报告类型决定了希望受害者浏览器如何提交先前记录的请求。目前有 5 种可能的报告:表单、iFrame、IMG、XHR 和链接。一旦选择了报告类型,可以选择在浏览器启动新生成的报告,最后根据报告的情况进行对应的排查和修复。

CSRF 监控

对于一个比较复杂的网站系统,某些项目、页面、接口漏掉了 CSRF 防护措施是很可能的。一旦发生了 CSRF 攻击,如何及时发现这些攻击呢?CSRF 攻击有着比较明显的特征:

  • 跨域请求
  • GET 类型请求 Header 的 MIME 类型大概率为图片,而实际返回 Header 的 MIME 类型为 Text、JSON、HTML

可以在网站的代理层监控所有的接口请求,如果请求符合上面的特征,就可以认为请求有 CSRF 攻击嫌疑。可以提醒对应的页面和项目负责人,检查或者 Review 其 CSRF 防护策略。

个人措施

经常上网的个人用户,可以采用以下方法来保护自己:

  • 使用网页版邮件来浏览邮件或新闻也会带来额外的风险,因为查看邮件或者新闻消息有可能导致恶意代码的攻击
  • 尽量不要打开可疑的链接,一定要打开时,使用不常用的浏览器

总结

简单总结防护策略:

  • CSRF 自动防御策略:同源检测(Origin 和 Referer 验证)
  • CSRF 主动防御措施:Token 验证 或者 双重 Cookie 验证以及配合 Samesite Cookie
  • 保证页面的幂等性,后端接口不要在 GET 请求中做用户操作

为了更好的防御 CSRF,最佳实践应该是结合上面总结的防御措施方式中的优缺点来综合考虑,结合当前 Web 应用程序自身的情况做合适的选择,才能更好的预防 CSRF 的发生。


CSRF攻击
https://xuekeven.github.io/2025/08/11/CSRF攻击/
作者
Keven
发布于
2025年8月11日
更新于
2025年8月25日
许可协议