From 70d7fb5491fe655715da1e7f52dbd8ac1085418e Mon Sep 17 00:00:00 2001 From: Alexander Momchilov Date: Wed, 19 Jun 2024 16:54:31 -0400 Subject: [PATCH 1/2] Optimize `URLHelpers.gather_constants` --- lib/tapioca/dsl/compilers/url_helpers.rb | 39 ++++++++++++++++-------- 1 file changed, 26 insertions(+), 13 deletions(-) diff --git a/lib/tapioca/dsl/compilers/url_helpers.rb b/lib/tapioca/dsl/compilers/url_helpers.rb index 0e36b157b..053d35558 100644 --- a/lib/tapioca/dsl/compilers/url_helpers.rb +++ b/lib/tapioca/dsl/compilers/url_helpers.rb @@ -106,19 +106,28 @@ def gather_constants routes_reloader = Rails.application.routes_reloader routes_reloader.execute_unless_loaded if routes_reloader&.respond_to?(:execute_unless_loaded) - Object.const_set(:GeneratedUrlHelpersModule, Rails.application.routes.named_routes.url_helpers_module) - Object.const_set(:GeneratedPathHelpersModule, Rails.application.routes.named_routes.path_helpers_module) + url_helpers_module = Rails.application.routes.named_routes.url_helpers_module + path_helpers_module = Rails.application.routes.named_routes.path_helpers_module + + Object.const_set(:GeneratedUrlHelpersModule, url_helpers_module) + Object.const_set(:GeneratedPathHelpersModule, path_helpers_module) constants = all_modules.select do |mod| next unless name_of(mod) - includes_helper?(mod, GeneratedUrlHelpersModule) || - includes_helper?(mod, GeneratedPathHelpersModule) || - includes_helper?(mod.singleton_class, GeneratedUrlHelpersModule) || - includes_helper?(mod.singleton_class, GeneratedPathHelpersModule) + # Fast-path to quickly disqualify most classes + next false unless mod < url_helpers_module || # rubocop:disable Style/InvertibleUnlessCondition + mod < path_helpers_module || + mod.singleton_class < url_helpers_module || + mod.singleton_class < path_helpers_module + + includes_helper?(mod, url_helpers_module) || + includes_helper?(mod, path_helpers_module) || + includes_helper?(mod.singleton_class, url_helpers_module) || + includes_helper?(mod.singleton_class, path_helpers_module) end - constants.concat(NON_DISCOVERABLE_INCLUDERS) + constants.concat(NON_DISCOVERABLE_INCLUDERS).push(GeneratedUrlHelpersModule, GeneratedPathHelpersModule) end sig { returns(T::Array[Module]) } @@ -134,17 +143,21 @@ def gather_non_discoverable_includers end.freeze end + # Returns `true` if `mod` "directly" includes `helper`. + # For classes, this method will return false if the `helper` is included only by a superclass sig { params(mod: Module, helper: Module).returns(T::Boolean) } private def includes_helper?(mod, helper) - superclass_ancestors = [] + ancestors = ancestors_of(mod) - if Class === mod - superclass = superclass_of(mod) - superclass_ancestors = ancestors_of(superclass) if superclass + own_ancestors = if Class === mod && (superclass = superclass_of(mod)) + # These ancestors are unique to `mod`, and exclude those in common with `superclass`. + ancestors.take(ancestors.count - ancestors_of(superclass).size) + else + ancestors end - ancestors = Set.new.compare_by_identity.merge(ancestors_of(mod)).subtract(superclass_ancestors) - ancestors.any? { |ancestor| helper == ancestor } + # Can't use a simple `include?(helper)` because of edge cases like `XPath`, whose `#==` operator misbehaves. + own_ancestors.any? { |a| helper == a } end end From ef5e6c9674fb77f59cd2f8a4f1003f83a442d73e Mon Sep 17 00:00:00 2001 From: Alexander Momchilov Date: Fri, 12 Jul 2024 14:55:39 -0400 Subject: [PATCH 2/2] Flip subtype checks to remove edge case `mod < url_helpers_module` gave false positives for weird classesl like `XPath`. They could survive through that first layer of filtering, and then require special handling later, by forcing us to use `.any? { |a| helper == a}` instead of the simpler (and faster) `.include?(helper)` check. By flipping this around, the receiver is always one of these known module types, which will always be hitting the usual `Module#>` implementation, which will filter out `XPath` early on. That frees us up to switch back to the faster `.include?(helper)` variant. --- lib/tapioca/dsl/compilers/url_helpers.rb | 13 ++++++------- 1 file changed, 6 insertions(+), 7 deletions(-) diff --git a/lib/tapioca/dsl/compilers/url_helpers.rb b/lib/tapioca/dsl/compilers/url_helpers.rb index 053d35558..8798c8ca4 100644 --- a/lib/tapioca/dsl/compilers/url_helpers.rb +++ b/lib/tapioca/dsl/compilers/url_helpers.rb @@ -115,11 +115,11 @@ def gather_constants constants = all_modules.select do |mod| next unless name_of(mod) - # Fast-path to quickly disqualify most classes - next false unless mod < url_helpers_module || # rubocop:disable Style/InvertibleUnlessCondition - mod < path_helpers_module || - mod.singleton_class < url_helpers_module || - mod.singleton_class < path_helpers_module + # Fast-path to quickly disqualify most cases + next false unless url_helpers_module > mod || # rubocop:disable Style/InvertibleUnlessCondition + path_helpers_module > mod || + url_helpers_module > mod.singleton_class || + path_helpers_module > mod.singleton_class includes_helper?(mod, url_helpers_module) || includes_helper?(mod, path_helpers_module) || @@ -156,8 +156,7 @@ def gather_non_discoverable_includers ancestors end - # Can't use a simple `include?(helper)` because of edge cases like `XPath`, whose `#==` operator misbehaves. - own_ancestors.any? { |a| helper == a } + own_ancestors.include?(helper) end end