# typed: strict # frozen_string_literal: true require "tapioca/compilers/sorbet" begin require "active_support" rescue LoadError return end return unless Tapioca::Compilers::Sorbet.supports?(:mixes_in_class_methods_multiple_args) module Tapioca module Compilers module Dsl # `Tapioca::Compilers::Dsl::ActiveSupportConcern` generates RBI files for classes that both `extend` # `ActiveSupport::Concern` and `include` another class that extends `ActiveSupport::Concern` # # For example for the following hierarchy: # # ~~~rb # # concern.rb # module Foo # extend ActiveSupport::Concern # module ClassMethods; end # end # # module Bar # extend ActiveSupport::Concern # module ClassMethods; end # include Foo # end # # class Baz # include Bar # end # ~~~ # # this generator will produce the RBI file `concern.rbi` with the following content: # # ~~~rbi # # typed: true # module Bar # mixes_in_class_methods(::Foo::ClassMethods) # end # ~~~ class ActiveSupportConcern < Base extend T::Sig sig { override.params(root: RBI::Tree, constant: Module).void } def decorate(root, constant) dependencies = linearized_dependencies_of(constant) mixed_in_class_methods = dependencies .uniq # Deduplicate .map do |concern| # Map to class methods module name, if exists "#{qualified_name_of(concern)}::ClassMethods" if concern.const_defined?(:ClassMethods) end .compact # Remove non-existent records return if mixed_in_class_methods.empty? root.create_path(constant) do |mod| mixed_in_class_methods.each do |mix| mod.create_mixes_in_class_methods(mix) end end end sig { override.returns(T::Enumerable[Module]) } def gather_constants # Find all Modules that are: all_modules.select do |mod| # named (i.e. not anonymous) name_of(mod) && # not singleton classes !mod.singleton_class? && # extend ActiveSupport::Concern, and mod.singleton_class < ActiveSupport::Concern && # have dependencies (i.e. include another concern) !dependencies_of(mod).empty? end end private sig { params(concern: Module).returns(T::Array[Module]) } def dependencies_of(concern) concern.instance_variable_get(:@_dependencies) end sig { params(concern: Module).returns(T::Array[Module]) } def linearized_dependencies_of(concern) # Grab all the dependencies of the concern dependencies = dependencies_of(concern) # Flatten this concern's dependencies and all of their dependencies dependencies.flat_map do |dependency| # Linearize dependencies of the current dependency, # which, itself, is a concern linearized_dependencies_of(dependency) << dependency end end end end end end