turbo-rails: Turbo stream partial with button_to tag throws ActionView::Template::Error
after_create_commit with broadcast_append_to throws this error:
ActionView::Template::Error (Request forgery protection requires a working session store but your application has sessions disabled. You need to either disable request forgery protection, or configure a working session store.) 
(seems to happen to all broadcasts with partials)
Log trace:
10:22:02 web.1  | ActionView::Template::Error (Request forgery protection requires a working session store but your application has sessions disabled. You need to either disable request forgery protection, or configure a working session store.):
10:22:02 web.1  |     13:     <div class="flex">
10:22:02 web.1  |     14: 
10:22:02 web.1  |     15:       <div class="up">
10:22:02 web.1  |     16:         <%= button_to increment_question_group_path(question_group), method: :patch do %>
10:22:02 web.1  |     17:           <svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5 text-gray-500 font-light" viewBox="0 0 20 20" fill="currentColor">
10:22:02 web.1  |     18:             <path fill-rule="evenodd" d="M5.293 9.707a1 1 0 010-1.414l4-4a1 1 0 011.414 0l4 4a1 1 0 01-1.414 1.414L11 7.414V15a1 1 0 11-2 0V7.414L6.707 9.707a1 1 0 01-1.414 0z" clip-rule="evenodd" />
10:22:02 web.1  |     19:           </svg>
10:22:02 web.1  |   
10:22:02 web.1  | app/views/question_groups/_question_group.html.erb:16
10:22:02 web.1  | app/models/question_group.rb:16:in `append_to_list'
Rails 7 alpha2, turbo-rails 0.7.14
About this issue
- Original URL
- State: closed
- Created 3 years ago
- Reactions: 4
- Comments: 46 (8 by maintainers)
Commits related to this issue
- Execute CI against `rails/rails` main In preparation for troubleshooting [hotwired/turbo-rails#243][], this commit expands the GitHub Actions CI matrix to account for both `rails@~>6.1` and `rails@ma... — committed to seanpdoyle/turbo-rails by seanpdoyle 3 years ago
- Execute CI against `rails/rails` main In preparation for troubleshooting [hotwired/turbo-rails#243][], this commit expands the GitHub Actions CI matrix to account for both `rails@~>6.1` and `rails@ma... — committed to seanpdoyle/turbo-rails by seanpdoyle 3 years ago
- Execute CI against `rails/rails` main In preparation for troubleshooting [hotwired/turbo-rails#243][], this commit expands the GitHub Actions CI matrix to account for both `rails@~>6.1` and `rails@ma... — committed to seanpdoyle/turbo-rails by seanpdoyle 3 years ago
- Execute CI against `rails/rails` main In preparation for troubleshooting [hotwired/turbo-rails#243][], this commit expands the GitHub Actions CI matrix to account for both `rails@~>6.1` and `rails@ma... — committed to seanpdoyle/turbo-rails by seanpdoyle 3 years ago
- Execute CI against `rails/rails` main In preparation for troubleshooting [hotwired/turbo-rails#243][], this commit expands the GitHub Actions CI matrix to account for both `rails@~>6.1` and `rails@ma... — committed to seanpdoyle/turbo-rails by seanpdoyle 3 years ago
- Execute CI against `rails/rails` main In preparation for troubleshooting [hotwired/turbo-rails#243][], this commit expands the GitHub Actions CI matrix to account for both `rails@~>6.1` and `rails@ma... — committed to seanpdoyle/turbo-rails by seanpdoyle 3 years ago
- Execute CI against `rails/rails` main In preparation for troubleshooting [hotwired/turbo-rails#243][], this commit expands the GitHub Actions CI matrix to account for both `rails@~>6.1` and `rails@ma... — committed to seanpdoyle/turbo-rails by seanpdoyle 3 years ago
- Disable CSRF when rendering from turbo Ref: https://github.com/hotwired/turbo-rails/issues/243 CSRF protection requires access to the user session, but it is only available during a regular request ... — committed to casperisfine/turbo-rails by byroot 3 years ago
- Stop failing GSRF token generation when session is disabled In theory this should have warned early that the CSRF check will fail, which would have been less puzzling for the developer. However ther... — committed to Shopify/rails by byroot 3 years ago
I believe this needs to be reopened.
Rails 7 alpha raises an error when sessions are disabled, which appears to break model broadcasts.
A minimal application reproducing the error can be found here: https://github.com/DavidColby/turbo_rails_7_bug
This app includes an
after_create_commitbroadcast:And it includes a
button_toin thepostpartial:Note that a
form_withwill fail too, rendering any form appears to cause the error.To reproduce on this app, boot up the app, head to /posts/new and see that
ActionController::RequestForgeryProtection::DisabledSessionError in Posts#createis raised when you attempt to submit the form.Uncomment this line and reboot the app and see that post submission works as expected.
The same is happening with
rich_text_fieldtags used in forms (just to be clear, the form needs to appear in the Turbo broadcast). I found this error in our application to be raised after the changes introduced in rails/rails #38957:Your application has sessions disabled. To write to the session you must first configure a session store.after_create_commitwithbroadcast_append_later_toenqueues aTurbo::Streams::ActionBroadcastJobfor the prepend. The newest changes for therich_text_area_taghelper now checks for a Rails session. The problem with that is a background job won’t have a rails session. In ourbroadcast_append_later_towe send a partial that includes a form and a rich text field.https://github.com/DmitryTsepelev/rails/blob/193289dbbe146c56ec16faf8dd1a2c88611feb83/actiontext/app/helpers/action_text/tag_helper.rb#L37
Here is an example of the stack trace,
Solutions for getting the ‘authenticity_token’ when a form is rendered via a broadcast from @aantix [1] and @mhenrixon [2] or [2] may work (I’m using the @aantix one, as of now) but IMHO the issue should be handled at a global level.
Hello, just another update on the problem @aantix seems the only solution through turbo atm when triggering the broadcast from a worker. But in the case of displaying a collection that relies on this broadcast you are then forced to setup an n+x situation. So it can be a solution, but you are quickly forced to use another system whenever the n+x is too much of trouble when you can’t do without authentication.
@tbcooney @AhmedNadar Broadcast from your background job a lazy-loaded turbo frame.
That lazy-loaded frame will then in turn load your form with a second request, using the current user’s session.
E.g.
My solution was a little different:
I use mrujs to set an authenticity token dynamically based on the one on the top level.
Are you sure you have a working session store, as the error mentions? Can you create a boiled down app that replicates this? Not seeing it on my end.
Stumbled upon this today. rails (7.0.4) turbo-rails (1.3.2)
a button_to is used in a partial that is broadcasted. Clicking on it results in “ActionController::InvalidAuthenticityToken (Can’t verify CSRF token authenticity.)”
It just seems surprising that it does not work out of the box
When I was using
rails v-7.0.0.alpha2withturbo-rails v-0.9.0I solved it withform_withBut now, using
rails v-7.0.0.rc3withturbo-rails v-1.0.0that it is working withbutton_toandmethod: :deleteThat’s what I’d expect yes.
It looks like this already happens. Using the example above, when a form is rendered via a broadcast the HTML looks like this:
The same form rendered during a normal page turn looks like this:
I suppose this means there must be some path that bypasses token generation when the session is unwriteable. Then when the form is submitted without a token, the token from the
<head>gets used instead?I was able to address this be changing from
button_totolink_tosincelink_towill get a CSRF token on the fly.Looking at this further, it may be more appropriate to raise this issue against Rails since the change to default to raising errors when the session is disabled causes rendering outside of the controller context to fail and turbo-rails isn’t the only library that expects to be able to safely render partials from models, background jobs, etc.
I’ll leave this here for now and see what others think. IMO, a default in Rails 7 that explicitly breaks rendering from models seems like an incorrect default given the reliance on this pattern in a number of libraries.