Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Avoid loading ActionView::Base during initialization #1528

Merged
merged 7 commits into from
Oct 11, 2022

Conversation

jdelStrother
Copy link
Contributor

What are you trying to accomplish?

ActionView::Base shouldn't be loaded during the initialization process.
Rails does a poor job of making this explicit, IMO - there's a suggestion that you shouldn't load frameworks (ActiveRecord::Base, ActionView::Base etc) during boot here: https://edgeguides.rubyonrails.org/engines.html#load-and-configuration-hooks, since it slows the boot process.
More concretely, loading those frameworks too soon can mean that they ignore configuration - eg setting config.active_record.belongs_to_required_by_default in config/initializers/new_framework_defaults.rb is ignored if ActiveRecord::Base has already been loaded.
As it happens, ActionView::Base doesn't have the same problem because most of its settings are propagated to, eg, ActionView::Helpers::TagHelper in after_initialize, rather than being set directly on ActionView:Base, but that seems a fragile thing to rely on.

#1507

What approach did you choose and why?

I've started by making the test suite fail if Rails frameworks load too soon here: 50f0bf0 ... it's not great, I'm open to better suggestions. It's not an individual test, it just nukes the entire test process.

f10920a then fixes this by making the VC::Engine (rather than VC::Base) responsible for loading VC::Config, and the engine sets that config on VC::Base after initialization is complete (in ActiveSupport.on_load(:view_component) { ... })

I think it might be more rails-y to have something like this in the initializer:

modified   lib/view_component/engine.rb
@@ -36,7 +36,9 @@ module ViewComponent
 
     initializer "view_component.enable_instrumentation" do |app|
       ActiveSupport.on_load(:view_component) do
-        Base.config = app.config.view_component
+        app.config.view_component.each do |k, v|
+          send("#{k}=", v)
+        end
 
         if app.config.view_component.instrumentation_enabled.present?
           # :nocov:

which I mostly had working, although this integration test fails because it seems to expect changing Base.config to affect app.config.view_component and vice versa.
I'm not sure I agree with that (eg setting ActiveRecord::Base.verbose_query_logs = true doesn't change config.active_record.verbose_query_logs), but for the sake of not breaking backwards compatibility have left it as-is. I don't have a strong opinion either way.

Anything you want to highlight for special attention from reviewers?

I'm not quite sure what the original code was doing here -

%i[generate preview_controller preview_route show_previews_source].each do |config_option|
options[config_option] ||= ViewComponent::Base.public_send(config_option)
end
. options on line 18 is the exact same object as VC::Base.config, so it seems like that boils down to

  options = ViewComponent::Base.config
  options[:generate] ||= options.send(:generate)
  options[:preview_controller] ||= options.send(:preview_controller)
  options[:preview_route] ||= options.send(:preview_route)
  options[:show_previews_source] ||= options.send(:show_previews_source)

but let me know if I'm missing something dumb.

@jdelStrother
Copy link
Contributor Author

Bah, will try & take a look at the pvc failure later

Couldn't see a clean way of running this inside of an actual TestCase
test, so took the nuclear option & fail the entire suite.
@jdelStrother
Copy link
Contributor Author

I've added require "view_component/config" to the top of engine.rb so that it still works for people requiring view_component/engine directly. I think that could be removed in 3.0 in favour of autoloads.

Copy link
Member

@joelhawksley joelhawksley left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👏🏻 love that test. Thank you!

docs/CHANGELOG.md Outdated Show resolved Hide resolved
@joelhawksley joelhawksley merged commit 22dbe8f into ViewComponent:main Oct 11, 2022
jonrohan added a commit that referenced this pull request Jan 11, 2023
joelhawksley pushed a commit that referenced this pull request Jan 11, 2023
…#1626)

* Revert "Avoid loading ActionView::Base during initialization (#1528)"

This reverts commit 22dbe8f.

* Put back changelog

* Adding changelog for revert
claudiob pushed a commit to claudiob/view_component that referenced this pull request Dec 22, 2023
)

* Force tests to fail if Rails frameworks load during initialization

Couldn't see a clean way of running this inside of an actual TestCase
test, so took the nuclear option & fail the entire suite.

* Avoid loading ViewComponent::Base during initialization

It forces ActionView::Base to load, which is against Rails' engine
guidelines https://edgeguides.rubyonrails.org/engines.html#load-and-configuration-hooks

* Update docs/CHANGELOG.md

Co-authored-by: Joel Hawksley <joelhawksley@github.com>
Co-authored-by: Joel Hawksley <joel@hawksley.org>
claudiob pushed a commit to claudiob/view_component that referenced this pull request Dec 22, 2023
…ponent#1528)" (ViewComponent#1626)

* Revert "Avoid loading ActionView::Base during initialization (ViewComponent#1528)"

This reverts commit 22dbe8f.

* Put back changelog

* Adding changelog for revert
claudiob pushed a commit to claudiob/view_component that referenced this pull request Jan 3, 2024
)

* Force tests to fail if Rails frameworks load during initialization

Couldn't see a clean way of running this inside of an actual TestCase
test, so took the nuclear option & fail the entire suite.

* Avoid loading ViewComponent::Base during initialization

It forces ActionView::Base to load, which is against Rails' engine
guidelines https://edgeguides.rubyonrails.org/engines.html#load-and-configuration-hooks

* Update docs/CHANGELOG.md

Co-authored-by: Joel Hawksley <joelhawksley@github.com>
Co-authored-by: Joel Hawksley <joel@hawksley.org>
claudiob pushed a commit to claudiob/view_component that referenced this pull request Jan 3, 2024
…ponent#1528)" (ViewComponent#1626)

* Revert "Avoid loading ActionView::Base during initialization (ViewComponent#1528)"

This reverts commit 22dbe8f.

* Put back changelog

* Adding changelog for revert
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants