了解 Rails 真实性令牌

Understanding the Rails Authenticity Token

提问人:Ricardo Acras 提问时间:6/3/2009 最后编辑:Mateen UlhaqRicardo Acras 更新时间:9/21/2023 访问量:226326

问:

Rails 中的真实性令牌是什么?

Ruby-on-Rails Ruby 真实性令牌

评论

7赞 Chloe 2/20/2013
另请参阅:“为什么 Google 将 while(1) 预置到他们的 JSON 响应中?stackoverflow.com/questions/2669690/......
0赞 satchel 7/16/2021
我也把它作为对答案的编辑:一个指向 github 存储库的链接,允许点击引用:pix.realquadrant.com/authenticity-token

答:

150赞 Topher Fangio 6/3/2009 #1

真实性令牌的设计使您知道您的表单是从您的网站提交的。它是从运行它的计算机生成的,具有只有您的计算机才能知道的唯一标识符,从而有助于防止跨站点请求伪造攻击。

如果您只是在 rails 拒绝 AJAX 脚本访问时遇到困难,您可以使用

<%= form_authenticity_token %>

在创建表单时生成正确的令牌。

您可以在文档中阅读有关它的更多信息。

39赞 andi 6/3/2009 #2

是 rails 防止“跨站请求伪造(CSRF 或 XSRF)攻击”的方法。Authenticity Token

简单来说,它确保对 Web 应用的 PUT / POST / DELETE(可以修改内容的方法)请求是从客户端的浏览器发出的,而不是从有权访问在客户端创建的 cookie 的第三方(攻击者)发出的。

1535赞 Faisal 10/15/2009 #3

会发生什么

当用户查看表单以创建、更新或销毁资源时,Rails 应用程序会创建一个随机的,将此令牌存储在会话中,并将其放置在表单的隐藏字段中。当用户提交表单时,Rails 会查找 ,将其与会话中存储的表单进行比较,如果它们匹配,则允许请求继续。authenticity_tokenauthenticity_token

为什么会这样

由于真实性令牌存储在会话中,因此客户端无法知道其值。这样可以防止人们向 Rails 应用提交表单,而无需在该应用本身中查看表单。 想象一下,您正在使用服务 A,您登录了该服务,一切正常。现在想象一下,你去使用服务 B,你看到了一张你喜欢的图片,然后按下图片查看它的更大尺寸。现在,如果服务 B 上存在一些恶意代码,它可能会向服务 A(您已登录)发送请求,并通过向 发送请求来要求删除您的帐户。这就是所谓的 CSRF(跨站点请求伪造)。http://serviceA.example/close_account

如果服务 A 使用真实性令牌,则此攻击媒介不再适用,因为来自服务 B 的请求将不包含正确的真实性令牌,并且不允许继续。

API 文档描述了有关元标记的详细信息:

CSRF 保护是用该方法打开的, 检查令牌并在会话不匹配时重置会话 是意料之中的。为新的 Rails 生成对此方法的调用 默认情况下,应用程序。 token 参数是默认命名的。名称 并且必须将此令牌的值添加到呈现的每个布局中 表单,包含在 HTML 标题中。protect_from_forgeryauthenticity_tokencsrf_meta_tags

笔记

请记住,Rails 只验证不验证幂等方法(POST、PUT/PATCH 和 DELETE)。不检查 GET 请求的真实性令牌。为什么?因为 HTTP 规范规定 GET 请求是幂等的,不应在服务器上创建、更改或销毁资源,并且请求应该是幂等的(如果多次运行相同的命令,则每次都应得到相同的结果)。

此外,实际实现在开始时定义要复杂一些,以确保更好的安全性。Rails 不会对每个表单都发出相同的存储令牌。它也不会每次都生成和存储不同的令牌。它在会话中生成并存储加密哈希,并颁发新的加密令牌,每次呈现页面时,这些令牌都可以与存储的令牌进行匹配。请参阅 request_forgery_protection.rb

用于保护非幂等方法(POST、PUT/PATCH 和 DELETE)。此外,请确保不允许任何可能修改服务器上资源的 GET 请求。authenticity_token


查看@erturne关于 GET 请求幂等的注释。他比我在这里做的更好。

评论

29赞 marcamillion 10/26/2010
@Faisal,攻击者是否有可能简单地读取/捕获服务 A 表单的“隐藏”元素,并获取为用户生成的唯一令牌 - 前提是他们已经获得了用户为服务 A 启动的会话的访问权限?
14赞 Faisal 10/26/2010
@marcamillion:如果有人在服务 A 上劫持了您的会话,那么真实性令牌将无法保护您。劫机者将能够提交请求,并被允许继续。
14赞 Faisal 1/31/2011
@zabba:如果表单提交时没有正确的令牌,Rails 会引发 ActionController::InvalidAuthenticityToken 异常。您可以rescue_from异常并执行所需的任何处理。
5赞 Steven Soroka 4/26/2012
re “还要确保不要发出任何可能修改服务器上资源的 GET 请求”——这包括不要在路由中使用 match(),这可能会允许对仅接收 POST 的控制器操作发出 GET 请求
109赞 Eric Turner 8/19/2012
"...并且请求应该是幂等的(如果多次运行相同的命令,则每次都应该得到相同的结果)。这里只是一个微妙的澄清。安全意味着没有副作用。幂等意味着无论调用服务多少次,都会产生相同的副作用。所有安全服务本质上都是幂等的,因为没有副作用。对当前时间资源多次调用 GET 每次都会返回不同的结果,但它是安全的(因此是幂等的)。
27赞 jdp 8/31/2011 #4

请注意,如果来自同一客户端的多个并发请求,真实性令牌机制可能会导致争用条件。在这种情况下,您的服务器可以生成多个真实性令牌,而应该只有一个,并且以表单接收早期令牌的客户端将在下一个请求中失败,因为会话 cookie 令牌已被覆盖。 这里有一篇关于这个问题的文章和一个不完全平凡的解决方案:http://www.paulbutcher.com/2007/05/race-conditions-in-rails-sessions-and-how-to-fix-them/

99赞 Rose Perrone 6/13/2012 #5

什么是CSRF?

真实性令牌是跨站点请求伪造 (CSRF) 的对策。你问什么是CSRF?

攻击者可以在不知道会话令牌的情况下劫持会话。

场景

  • 访问您银行的网站,登录。
  • 然后访问攻击者的网站(例如,来自不受信任组织的赞助广告)。
  • 攻击者的页面包含与银行的“转账资金”表单具有相同字段的表单。
  • 攻击者知道您的帐户信息,并预先填写了表单字段,用于将资金从您的帐户转移到攻击者的帐户。
  • 攻击者的页面包含向银行提交表单的 Javascript。
  • 提交表单后,浏览器会包含银行网站的 cookie,包括会话令牌。
  • 银行将资金转入攻击者的账户。
  • 表单可以位于不可见的 iframe 中,因此您永远不知道攻击发生了。
  • 这称为跨站点请求伪造 (CSRF)。

CSRF解决方案

  • 服务器可以标记来自服务器本身的表单
  • 每个表单都必须包含一个额外的身份验证令牌作为隐藏字段。
  • 令牌必须是不可预测的(攻击者无法猜测它)。
  • 服务器在其页面的表单中提供有效的令牌。
  • 服务器在表单发布时检查令牌,拒绝没有正确令牌的表单。
  • 示例令牌:使用服务器密钥加密的会话标识符。
  • Rails 会自动生成这样的标记:查看每个表单中的authenticity_token输入字段。

评论

1赞 Lutz Prechelt 11/20/2015
这是同一解释的一个版本,它不那么精确,但也不那么抽象:stackoverflow.com/a/33829607/2810305
0赞 divideByZero 9/23/2016
我不确定,但是,现代浏览器是否允许向另一个域发送非幂等请求(POST/PUT/DELETE)?我想,浏览器本身必须有针对此类的保护措施
0赞 Peter Gerdes 6/29/2021
@divideByZero(哦,好名字!)有一些 CORS 标头形式的保护。一个站点可以指定它希望从哪些域接收请求(某些浏览器/API 甚至更严格),但我不确定这是什么时候采用的,或者是否真的旧浏览器都支持它,并且人们可能也希望有这种保护,以防域将其 CORS 设置保留为 *。developer.mozilla.org/en-US/docs/Web/HTTP/CORS
34赞 Yuan He 8/1/2012 #6

因为是如此重要,在 Rails 3.0+ 中你可以使用Authenticity Token

 <%= token_tag nil %>

创建

<input name="authenticity_token" type="hidden" value="token_value">

无论何处

评论

0赞 Michael 2/25/2014
这对我很有帮助。我实际上是在尝试在登录页面上这样做,不是出于恶意目的,而是使用预先填写的用户名创建一个新会话。现在我知道我可以使用.XSSvalue="token_value"
11赞 uma 2/26/2014 #7

需要authenticity_token的方法

authenticity_token在幂等方法(如 post、put 和 delete)的情况下是必需的,因为幂等方法会影响数据。

为什么需要它

这是防止邪恶行为所必需的。authenticity_token存储在会话中,每当在网页上创建表单以创建或更新资源时,真实性令牌就会存储在隐藏字段中,并与服务器上的表单一起发送。在执行操作之前,用户发送authenticity_token会与会话中存储的操作进行交叉检查。如果相同,则进程继续,否则不执行操作。authenticity_tokenauthenticity_token

评论

3赞 Jean-Théo 7/10/2014
其实,不是恰恰相反吗?GET 是幂等的,因为它的调用不应改变系统的状态,其中 PUT POST 和 DELETE 谓词不是幂等动词,因为它们会改变系统状态。I.E:在非幂等方法的情况下需要authenticity_token。
2赞 Ciro Santilli OurBigBook.com 11/13/2014
@Jean-Daube,uma:幂等意味着如果执行两次,操作只会发生一次。GET、PUT 和 DELETE 幂等的: w3.org/Protocols/rfc2616/rfc2616-sec9.html 这里的关键属性不是幂等性,而是方法是否更改数据,这称为“安全方法”与否。
49赞 Ciro Santilli OurBigBook.com 11/13/2014 #8

可防止的最小攻击示例:CSRF

在我的网站上,我说服您提交以下表格:evil.example

<form action="http://bank.com/transfer" method="post">
  <p><input type="hidden" name="to"      value="ciro"></p>
  <p><input type="hidden" name="ammount" value="100"></p>
  <p><button type="submit">CLICK TO GET PRIZE!!!</button></p>
</form>

如果您通过会话 cookie 登录您的银行,则 cookie 将被发送并在您不知情的情况下进行转账。

这就是 CSRF 代币发挥作用的原因:

  • 使用返回表单的 GET 响应,Rails 会发送一个非常长的随机隐藏参数
  • 当浏览器发出 POST 请求时,它将发送参数,并且服务器只会在匹配时接受它

因此,真实浏览器上的表单如下所示:

<form action="http://bank.com/transfer" method="post">
  <p><input type="hidden" name="authenticity_token" value="j/DcoJ2VZvr7vdf8CHKsvjdlDbmiizaOb5B8DMALg6s=" ></p>
  <p><input type="hidden" name="to"                 value="ciro"></p>
  <p><input type="hidden" name="ammount"            value="100"></p>
  <p><button type="submit">Send 100$ to Ciro.</button></p>
</form>

因此,我的攻击会失败,因为它没有发送参数,而且我无法猜到它,因为它是一个巨大的随机数。authenticity_token

这种预防技术称为同步器令牌模式

同源策略

但是,如果攻击者使用 JavaScript 发出了两个请求,一个请求读取令牌,另一个请求进行传输,该怎么办?

仅靠同步器令牌模式不足以防止这种情况发生!

这就是我在以下位置解释的同源策略的用武之地:https://security.stackexchange.com/questions/8264/why-is-the-same-origin-policy-so-important/72569#72569

Rails 如何发送令牌

涵盖: Rails:csrf_meta_tag是如何工作的?

基本上:

  • HTML 帮助程序,例如在表单中添加隐藏字段(如果它不是 GET 表单)form_tag

  • AJAX 由 jquery-ujs 自动处理,它从添加到标头的元素中读取令牌(存在于默认模板中),并将其添加到发出的任何请求中。metacsrf_meta_tags

    uJS 还会尝试在过时的缓存片段中的表单中更新令牌。

其他预防方法

评论

0赞 bjm88 2/28/2017
谢谢,但是您关于依赖同源策略而不能首先读取 CSRF 令牌的观点似乎是有缺陷的。所以首先你说你可以 POST 到不同的来源但不能从中读取,这似乎很奇怪,但我想这是正确的,但你可以注入一个带有 get 到页面的图像或脚本标签并链接一个处理程序来解析响应并得到它,是吗?
1赞 Ciro Santilli OurBigBook.com 2/28/2017
@bjm88将脚本注入到哪里?在您的网站上,还是在被攻击的网站上?如果网站受到攻击,允许脚本注入是一个众所周知的安全漏洞,并且实际上会典当网站。每个网站都必须通过输入卫生来对抗它。对于图像,我看不出它们如何用于攻击。在攻击站点上:您可以修改浏览器以允许读取,从而随意自动典当自己:-)但是体面的浏览器默认会阻止它,请尝试一下。
7赞 Pradeep Sapkota 7/22/2016 #9

什么是authentication_token?

这是 rails 应用程序使用的随机字符串,用于确保用户从应用程序页面请求或执行操作,而不是从其他应用程序或站点请求或执行操作。

为什么需要authentication_token?

保护您的应用或网站免遭跨站点请求伪造。

如何将authentication_token添加到表单中?

如果使用 form_for 标记生成表单,则会自动添加authentication_token,否则您可以使用 .<%= csrf_meta_tag %>

70赞 Adam Zerner 4/14/2017 #10

真实性令牌用于防止跨站点请求伪造攻击 (CSRF)。要了解真实性令牌,必须首先了解 CSRF 攻击。

CSRF的

假设您是 的作者。您的网站上有一个表单,用于通过 GET 请求将资金转入其他帐户:bank.example

在此处输入图像描述

黑客可以向服务器发送一个HTTP请求,说,对吧?GET /transfer?amount=$1000000&account-to=999999

在此处输入图像描述

错。黑客攻击是行不通的。服务器基本上会想吗?

哼?这个试图发起转移的人是谁。它不是帐户的所有者,这是肯定的。

服务器如何知道这一点?因为没有 cookie 对请求者进行身份验证。session_id

当您使用用户名和密码登录时,服务器会在您的浏览器上设置一个 cookie。这样,您就不必使用用户名和密码对每个请求进行身份验证。当您的浏览器发送 cookie 时,服务器会知道:session_idsession_id

哦,那是约翰·多伊。他在 2.5 分钟前成功登录。他很高兴。

黑客可能会想:

嗯。普通的HTTP请求是行不通的,但如果我能得到那个cookie,我就会成为金牌。session_id

用户浏览器为域设置了一堆 cookie。每次用户向域发出请求时,都会发送所有 cookie。包括 cookie。bank.examplebank.examplesession_id

因此,如果黑客可以让你发出 GET 请求,将钱转入他的账户,他就会成功。他怎么会骗你这样做? 与跨站点请求伪造。

实际上,这很简单。黑客可以让你访问他的网站。在他的网站上,他可以有以下图像标签:

<img src="http://bank.example/transfer?amount=$1000000&account-to=999999">

当用户浏览器遇到该图像标签时,它将向该 URL 发出 GET 请求。由于请求来自他的浏览器,它将发送与 .如果用户最近登录了...cookie将被设置,服务器将认为用户打算将1,000,000美元转移到帐户999999!bank.examplebank.examplesession_id

在此处输入图像描述

好吧,只要不要去危险的地方,你就会没事的。

这还不够。如果有人将该图片发布到 Facebook 并出现在您的墙上怎么办?如果它被注入到您正在访问的 XSS 攻击站点中怎么办?

还不错。只有 GET 请求容易受到攻击。

不对。可以动态生成发送 POST 请求的表单。以下是 Rails 安全指南中的示例:

<a href="http://www.harmless.example/" onclick="
  var f = document.createElement('form');
  f.style.display = 'none';
  this.parentNode.appendChild(f);
  f.method = 'POST';
  f.action = 'http://www.example.com/account/destroy';
  f.submit();
  return false;">To the harmless survey</a>

真实性令牌

当你有这个:ApplicationController

protect_from_forgery with: :exception

这:

<%= form_tag do %>
  Form contents
<% end %>

编译成这样:

<form accept-charset="UTF-8" action="/" method="post">
  <input name="utf8" type="hidden" value="&#x2713;" />
  <input name="authenticity_token" type="hidden" value="J7CBxfHalt49OSHp27hblqK20c9PgwJ108nDHX/8Cts=" />
  Form contents
</form>

具体而言,会生成以下内容:

<input name="authenticity_token" type="hidden" value="J7CBxfHalt49OSHp27hblqK20c9PgwJ108nDHX/8Cts=" />

为了防止 CSRF 攻击,如果 Rails 没有看到与请求一起发送的真实性令牌,它就不会认为请求是安全的。

攻击者应该如何知道这个令牌是什么?每次生成表单时,都会随机生成不同的值:

在此处输入图像描述

跨站点脚本 (XSS) 攻击 - 就是这样。但对于不同的一天来说,这是一个不同的漏洞。

-1赞 Mohsin Amjad 9/21/2023 #11

在 Ruby on Rails 中,真实性令牌(通常称为 CSRF(跨站点请求伪造)令牌)是一种安全功能,用于保护 Web 应用程序免受某些类型的攻击,尤其是 CSRF 攻击。