5

我有一个使用 CORS 对另一个域进行身份验证的单页应用程序。所有请求都是 JSON 请求。

我的应用程序可以进行身份​​验证,并且可以使 GET 请求正常。身份验证使用 token_authenticable。即所有请求都附加 '?auth_token=whatever'

所以,我的实际问题是,当我尝试执行 PUT 请求时,我收到一条警告:无法在 rails 日志中验证 CSRF 令牌真实性消息以及CanCan::AccessDenied (您无权访问此页面。 )例外。

只需添加skip_before_filter :verify_authenticity_token到 rails 控制器即可解决问题。

因此我只能得出结论,我的 ajax 请求正在发送一个无效或空的 csrf_token。

我真的不明白这是怎么回事,因为我相信我在每个 ajax 请求中都正确地发送了 X-CSRF-Token 标头。

基本上,我的应用程序进行身份验证,然后 Devise 发回一个 auth_token 和一个 csrf_token:

render :status => 200, :json => {
  :auth_token => @user.authentication_token,
  :csrf_token => form_authenticity_token
}

然后我将这些令牌存储在我的 ajax 应用程序中,并ajaxSend在 jQuery 中使用,设置它以便 jQuery 将这些令牌与每个请求一起传递:

initialize: ->
  @bindTo $(document), 'ajaxSend', @appendTokensToRequest

appendTokensToRequest: (event, jqXHR, options) ->
  if not @authToken? then return

  if @csrfToken?
    jqXHR.setRequestHeader 'X-CSRF-Token', @csrfToken

  if options.type is 'POST'
    options.data = options.data + (if options.data.match(/\=/) then '&' else '') +
    $.param auth_token:@authToken
  else
    options.url = options.url + (if options.url.match(/\?/) then '&' else '?') +
    $.param auth_token:@authToken

然后我可以在 chrome 网络选项卡中看到,对于每个 GET 请求,auth_token正在发送参数以及X-CSRF-Token标头。

但是,在 PUT 请求上,它似乎不起作用。

我的理论是 CORS 把事情搞砸了。如果您发出 CORS 请求,您的浏览器实际上会首先发出一个额外的OPTIONS 请求,只是为了检查您是否有权访问此资源。

我怀疑是 OPTIONS 请求没有通过 X-CSRF-Token 标头,因此 rails 立即使 rails 端的 csrf_token 无效。然后,当 jQuery 发出实际的 PUT 请求时,它传递的 csrf_token 不再有效。

这可能是问题吗?

我能做些什么来证明这一点?Chrome 似乎没有在网络选项卡中向我显示 OPTIONS 请求来帮助我调试问题。

这不是一个大问题,因为我可以关闭 CSRF 的东西。但我想知道为什么它不起作用。

4

3 回答 3

1

我认为您需要处理 OPTIONS 请求,该请求应响应允许 CORS 请求的各种标头,IIRC 它们是 access-control-allow-method、access-control-allow-origin 和 access-control-允许标题。因为 OPTIONS 请求失败,所以 PUT 请求可能没有发生。

于 2013-01-04T03:17:30.867 回答
1

我刚刚遇到了同样的问题。问题是_session_id无法在 CORS 中发送 cookie。因此,当 Rails 尝试验证令牌时,session[:_csrf_token]isnull和 Rails 在比较之前会生成一个新的。

要解决此问题,您需要在 CORS 中启用 cookie 发送。这是Mozilla 开发者网络参考。服务器端和客户端都需要工作才能使其正常工作。

客户端 - 请参阅您的客户端技术文档。

服务器 -在对预检 (HTTP OPTIONS) 调用的响应中将标头设置Access-Control-Allow-Credentials为(string)。true

于 2015-06-02T21:43:55.600 回答
0

在 Rails 中,每个表单提交都需要 CSRF 令牌真实性。它用于安全地提交表单。当我们打开我们的应用程序时,CSRF 令牌(每次)将在 rails 中新创建。如果 CSRF 令牌没有在我们的控制器内部传递,则会显示此警告。我们需要在所有表单提交中传递这个令牌。

于 2012-10-24T05:38:06.247 回答