From 50402466cd85f27a8515b3865797d12ac741c26d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Fern=C3=A1ndez-Capel?= Date: Tue, 19 Jul 2022 16:58:00 +0200 Subject: [PATCH 1/2] Give html: rendering param the same treatment as content: (#362) broadcast_action_to accepts a content: argument to set the turbo-stream template content directly, bypassing any rendering. Other params, like partial: or locals: are delegated to render. Since the arguments are passed directly to render, you might expect you can also use html: as an argument here. However, there's a catch: when using a queue system to broadcast the action later, the html_safe flag of the html param might be lost in the serialization process. Specifically, the flag remains with queue backends that work in process (like `inline` or `async`) but it's lost in other that serialize the payload to JSON (like `resque` or `sidekiq`). The resulting effect is that the broadcast action works fine in development, but then fails when it's deployed to a production system. This problem seems to have tripped a few people: https://github.com/hotwired/turbo-rails/issues/284 Treating html: the same as content: solves the issue. --- app/channels/turbo/streams/broadcasts.rb | 2 +- test/streams/streams_channel_test.rb | 8 ++++++++ 2 files changed, 9 insertions(+), 1 deletion(-) diff --git a/app/channels/turbo/streams/broadcasts.rb b/app/channels/turbo/streams/broadcasts.rb index b15ae2b9..f3aabd39 100644 --- a/app/channels/turbo/streams/broadcasts.rb +++ b/app/channels/turbo/streams/broadcasts.rb @@ -35,7 +35,7 @@ def broadcast_prepend_to(*streamables, **opts) def broadcast_action_to(*streamables, action:, target: nil, targets: nil, **rendering) broadcast_stream_to(*streamables, content: turbo_stream_action_tag(action, target: target, targets: targets, template: - rendering.delete(:content) || (rendering.any? ? render_format(:html, **rendering) : nil) + rendering.delete(:content) || rendering.delete(:html) || (rendering.any? ? render_format(:html, **rendering) : nil) )) end diff --git a/test/streams/streams_channel_test.rb b/test/streams/streams_channel_test.rb index 43e0c42f..35c990c8 100644 --- a/test/streams/streams_channel_test.rb +++ b/test/streams/streams_channel_test.rb @@ -89,6 +89,14 @@ class Turbo::StreamsChannelTest < ActionCable::Channel::TestCase assert_broadcast_on "stream", turbo_stream_action_tag("prepend", targets: ".message", template: render(options)) do Turbo::StreamsChannel.broadcast_action_to "stream", action: "prepend", targets: ".message", **options end + + assert_broadcast_on "stream", turbo_stream_action_tag("prepend", targets: ".message", template: "test") do + Turbo::StreamsChannel.broadcast_action_to "stream", action: "prepend", targets: ".message", content: "test" + end + + assert_broadcast_on "stream", turbo_stream_action_tag("prepend", targets: ".message", template: "test") do + Turbo::StreamsChannel.broadcast_action_to "stream", action: "prepend", targets: ".message", html: "test" + end end test "broadcasting replace later" do From f9872c35fdcdbd216141dc02a654d914bd25482c Mon Sep 17 00:00:00 2001 From: Sean Doyle Date: Tue, 2 Aug 2022 14:24:27 -0400 Subject: [PATCH 2/2] Improve upon test suite flakiness (#327) * Make `turbo_test.rb` with Rails' generated `test_helper.rb` > Something in the test suite configuration is preventing the database > from being wiped between test runs. This results in state leaking > between tests. As a result, our Continuous Integration tests are flaky. > > - [hotwired/turbo-rails#248][] As a follow-up to the [short-term solution][] shipped in [hotwired/turbo-rails#248][], this commit attempts to make the `test/turbo_test.rb` file's setup consistent with the test harness setup generated by Rails' [engine generator][] code. To that end, this commit: * renames the `test/turbo_test.rb` file to `test/test_helper.rb` * omits one-off `require` calls for particular dependencies * re-orders the require calls so that the `../test/dummy/config/environment` file is required ahead of the `rails/test_help` file [engine generator]: https://github.com/rails/rails/blob/3c48b4030adbded21bebaa0d78254216cca48a6e/railties/lib/rails/generators/rails/plugin/templates/test/test_helper.rb.tt [hotwired/turbo-rails#248]: https://github.com/hotwired/turbo-rails/pull/248 [short-term solution]: https://github.com/hotwired/turbo-rails/commit/c2dc5b15acaf3cb0aebb581b7dfa0b792ec50bc1 * Use Ruby 2.7 argument forwarding Switching to argument forwarding addresses deprecation warnings like: ``` hotwired/turbo-rails/test/streams/streams_channel_test.rb:140: warning: Using the last argument as keyword parameters is deprecated; maybe ** should be added to the call hotwired/turbo-rails/test/turbo_test.rb:14: warning: The called method `render' is defined here ``` * Tests: Load 6.1 defaults in Dummy Application Resolve deprecation warnings like: ``` Preparing test database DEPRECATION WARNING: Non-URL-safe CSRF tokens are deprecated. Use 6.1 defaults or above. (called from at /home/runner/work/turbo-rails/turbo-rails/test/dummy/config/initializers/inspect_helpers.rb:1) DEPRECATION WARNING: Using legacy connection handling is deprecated. Please set `legacy_connection_handling` to `false` in your application. The new connection handling does not support `connection_handlers` getter and setter. Read more about how to migrate at: https://guides.rubyonrails.org/active_record_multiple_databases.html#migrate-to-the-new-connection-handling (called from at /home/runner/work/turbo-rails/turbo-rails/test/test_helper.rb:6) ``` Since our GitHub CI matrix includes `6.1`, `7.0`, and `main`, CI's tests should run with at least the `6.1` defaults. * CI: Continue other executions on error Remove the `continue-on-error` configuration and instead allow other jobs complete in spite of failures. * Improve Flaky Test: Clear fields before filling in Resolve a flaky System Test by ensuring that an input is clear before filling in a new value. * Improve flaky tests: Broadcasts First, don't render HTML with the `` element. Instead, append the element when clicking a ` diff --git a/test/dummy/app/views/messages/echo.html.erb b/test/dummy/app/views/messages/echo.html.erb index ceace5e9..b73139a3 100644 --- a/test/dummy/app/views/messages/echo.html.erb +++ b/test/dummy/app/views/messages/echo.html.erb @@ -1,5 +1,8 @@

Echo Messages

-<%= turbo_stream_from "messages", channel: EchoChannel, data: {message: "Hello, world!"} %> +<%= render "template_button", content: "Start listening for broadcasts" do %> + <%= turbo_stream_from "messages", channel: EchoChannel, data: {message: "Hello, world!"} %> +<% end %> +
diff --git a/test/dummy/app/views/messages/index.html.erb b/test/dummy/app/views/messages/index.html.erb index 10ee7e96..4eb53daf 100644 --- a/test/dummy/app/views/messages/index.html.erb +++ b/test/dummy/app/views/messages/index.html.erb @@ -1,9 +1,8 @@

Messages

- - <%= @messages.count %> messages sent - +<%= render "template_button", content: "Start listening for broadcasts" do %> + <%= turbo_stream_from "messages" %> +<% end %> -<%= turbo_stream_from "messages" %>
diff --git a/test/dummy/app/views/users/profiles/index.html.erb b/test/dummy/app/views/users/profiles/index.html.erb index a3c19a5b..e5fa3551 100644 --- a/test/dummy/app/views/users/profiles/index.html.erb +++ b/test/dummy/app/views/users/profiles/index.html.erb @@ -1,4 +1,8 @@

Users::Profiles

-<%= turbo_stream_from "users_profiles" %> -
+<%= render "template_button", content: "Start listening for broadcasts" do %> + <%= turbo_stream_from "users_profiles" %> +<% end %> + +
+
diff --git a/test/dummy/config/application.rb b/test/dummy/config/application.rb index fefb2358..9d1ed2f3 100644 --- a/test/dummy/config/application.rb +++ b/test/dummy/config/application.rb @@ -12,7 +12,7 @@ module Dummy class Application < Rails::Application # Initialize configuration defaults for originally generated Rails version. - config.load_defaults 6.0 + config.load_defaults 6.1 # Settings in config/environments/* take precedence over those specified here. # Application configuration can go into files in config/initializers diff --git a/test/dummy/config/cable.yml b/test/dummy/config/cable.yml index 1cd0f836..663e871e 100644 --- a/test/dummy/config/cable.yml +++ b/test/dummy/config/cable.yml @@ -2,7 +2,7 @@ development: adapter: async test: - adapter: async + adapter: inline production: adapter: redis diff --git a/test/frames/frame_request_controller_test.rb b/test/frames/frame_request_controller_test.rb index e43cd60f..3ba6fa83 100644 --- a/test/frames/frame_request_controller_test.rb +++ b/test/frames/frame_request_controller_test.rb @@ -1,4 +1,4 @@ -require "turbo_test" +require "test_helper" class Turbo::FrameRequestControllerTest < ActionDispatch::IntegrationTest test "frame requests are rendered without a layout" do diff --git a/test/frames/frames_helper_test.rb b/test/frames/frames_helper_test.rb index 473b42be..678b5409 100644 --- a/test/frames/frames_helper_test.rb +++ b/test/frames/frames_helper_test.rb @@ -1,8 +1,6 @@ -require "turbo_test" +require "test_helper" class Turbo::FramesHelperTest < ActionView::TestCase - setup { Message.delete_all } - test "frame with src" do assert_dom_equal %(), turbo_frame_tag("tray", src: "/trays/1") end diff --git a/test/initializers/helpers_test.rb b/test/initializers/helpers_test.rb index 4f3c56a2..ce5ac036 100644 --- a/test/initializers/helpers_test.rb +++ b/test/initializers/helpers_test.rb @@ -1,4 +1,4 @@ -require "turbo_test" +require "test_helper" class Turbo::HelpersInInitializersTest < ActionDispatch::IntegrationTest test "AC::Base has the helpers in place when initializers run" do diff --git a/test/native/navigation_controller_test.rb b/test/native/navigation_controller_test.rb index da7a2f4b..92e1b012 100644 --- a/test/native/navigation_controller_test.rb +++ b/test/native/navigation_controller_test.rb @@ -1,4 +1,4 @@ -require "turbo_test" +require "test_helper" class Turbo::Native::NavigationControllerTest < ActionDispatch::IntegrationTest test "recede, resume, or refresh when native or redirect when not" do diff --git a/test/streams/broadcastable_test.rb b/test/streams/broadcastable_test.rb index daf1e29f..7dd757c2 100644 --- a/test/streams/broadcastable_test.rb +++ b/test/streams/broadcastable_test.rb @@ -1,4 +1,4 @@ -require "turbo_test" +require "test_helper" require "action_cable" class Turbo::BroadcastableTest < ActionCable::Channel::TestCase diff --git a/test/streams/streams_channel_test.rb b/test/streams/streams_channel_test.rb index 35c990c8..dea35c6f 100644 --- a/test/streams/streams_channel_test.rb +++ b/test/streams/streams_channel_test.rb @@ -1,4 +1,4 @@ -require "turbo_test" +require "test_helper" require "action_cable" class Turbo::StreamsChannelTest < ActionCable::Channel::TestCase diff --git a/test/streams/streams_controller_test.rb b/test/streams/streams_controller_test.rb index 733042b6..b1f1d3ba 100644 --- a/test/streams/streams_controller_test.rb +++ b/test/streams/streams_controller_test.rb @@ -1,4 +1,4 @@ -require "turbo_test" +require "test_helper" class Turbo::StreamsControllerTest < ActionDispatch::IntegrationTest test "create with respond to" do diff --git a/test/streams/streams_helper_test.rb b/test/streams/streams_helper_test.rb index 8efb7648..6ce36438 100644 --- a/test/streams/streams_helper_test.rb +++ b/test/streams/streams_helper_test.rb @@ -1,4 +1,4 @@ -require "turbo_test" +require "test_helper" class TestChannel < ApplicationCable::Channel; end diff --git a/test/system/broadcasts_test.rb b/test/system/broadcasts_test.rb index fab062bb..6a2ca3e2 100644 --- a/test/system/broadcasts_test.rb +++ b/test/system/broadcasts_test.rb @@ -1,61 +1,62 @@ require "application_system_test_case" class BroadcastsTest < ApplicationSystemTestCase - setup { Message.delete_all } - - include ActionCable::TestHelper - test "Message broadcasts Turbo Streams" do visit messages_path - wait_for_subscriber + subscribe_to_broadcasts - assert_text "Messages" - assert_broadcasts_text "Message 1" do |text| - Message.create(content: text).broadcast_append_to(:messages) + assert_broadcasts_text "Message 1", to: :messages do |text, target| + Message.create(content: text).broadcast_append_to(target) end end - test "New messages update the message count with html" do + test "Message broadcasts with html: render option" do visit messages_path - wait_for_subscriber - - assert_text "Messages" - message = Message.create(content: "A new message") + subscribe_to_broadcasts - message.broadcast_update_to(:messages, target: "message-count", - html: "#{Message.count} messages sent") - assert_selector("#message-count", text: Message.count, wait: 10) + assert_broadcasts_text "Hello, with html: option", to: :messages do |text, target| + Message.create(content: "Ignored").broadcast_append_to(target, html: text) + end end test "Users::Profile broadcasts Turbo Streams" do visit users_profiles_path - wait_for_subscriber + subscribe_to_broadcasts - assert_text "Users::Profiles" - assert_broadcasts_text "Profile 1" do |text| - Users::Profile.new(id: 1, name: text).broadcast_append_to(:users_profiles) + assert_broadcasts_text "Profile 1", to: :users_profiles do |text, target| + Users::Profile.new(id: 1, name: text).broadcast_append_to(target) end end test "passing extra parameters to channel" do visit echo_messages_path - wait_for_subscriber - assert_text "Hello, world!", wait: 100 + assert_broadcasts_text "Hello, world!", to: :messages do + subscribe_to_broadcasts + end end private - def assert_broadcasts_text(text, wait: 5, &block) - assert_no_text text - perform_enqueued_jobs { block.call(text) } - assert_text text, wait: wait + def subscribe_to_broadcasts + click_on "Start listening for broadcasts" + + assert_no_button "Start listening for broadcasts" + + Timeout.timeout(Capybara.default_max_wait_time) { wait_for_subscriber } + end + + def assert_broadcasts_text(text, to:, &block) + within(:element, id: to) { assert_no_text text } + + [text, to].yield_self(&block) + + within(:element, id: to) { assert_text text } end - def wait_for_subscriber(timeout: 10) - time = Time.now + def wait_for_subscriber loop do - subscriber_map = pubsub_adapter.instance_variable_get(:@subscriber_map) + subscriber_map = ActionCable.server.pubsub.instance_variable_get(:@subscriber_map) if subscriber_map.is_a?(ActionCable::SubscriptionAdapter::SubscriberMap) subscribers = subscriber_map.instance_variable_get(:@subscribers) sync = subscriber_map.instance_variable_get(:@sync) @@ -63,9 +64,7 @@ def wait_for_subscriber(timeout: 10) return unless subscribers.empty? end end - assert_operator(Time.now - time, :<, timeout, "subscriber waiting timed out") sleep 0.1 end end - end diff --git a/test/system/form_submissions_test.rb b/test/system/form_submissions_test.rb index f0745711..4fa80ecc 100644 --- a/test/system/form_submissions_test.rb +++ b/test/system/form_submissions_test.rb @@ -15,7 +15,8 @@ class FormSubmissionsTest < ApplicationSystemTestCase article = Article.create! body: "My article" visit edit_article_path(article.id) - fill_in("Body", with: "My edit").then { click_on "Submit as PATCH" } + fill_in "Body", with: "My edit", fill_options: { clear: :backspace } + click_on "Submit as PATCH" assert_text "Articles" assert_text "My edit", count: 1 diff --git a/test/turbo_test.rb b/test/test_helper.rb similarity index 69% rename from test/turbo_test.rb rename to test/test_helper.rb index a3c070aa..bc2878b8 100644 --- a/test/turbo_test.rb +++ b/test/test_helper.rb @@ -1,18 +1,15 @@ +# Configure Rails Environment ENV["RAILS_ENV"] = "test" -require "minitest/autorun" -require "rails" +require_relative "../test/dummy/config/environment" +ActiveRecord::Migrator.migrations_paths = [File.expand_path("../test/dummy/db/migrate", __dir__)] require "rails/test_help" -require "byebug" - -require_relative "dummy/config/environment" -ActiveRecord::Migrator.migrations_paths = [File.expand_path("../test/dummy/db/migrate", __dir__)] ActionCable.server.config.logger = Logger.new(STDOUT) if ENV["VERBOSE"] module ActionViewTestCaseExtensions - def render(*arguments, **options, &block) - ApplicationController.renderer.render(*arguments, **options, &block) + def render(...) + ApplicationController.renderer.render(...) end end