module Eco module API module Common module ClassHelpers # Creates a class and instance object methods with name `name` to resolve `klass` name def class_resolver(name, klass) define_singleton_method(name) { resolve_class(klass) } define_method(name) { self.class.resolve_class(klass) } end # With given a `klass` name it resolves to an actual `Class` # @return [Class] the class that was being searched by name `klass`. def resolve_class(klass, exception: true) @resolved ||= {} @resolved[klass] ||= case klass when Class klass when String begin Kernel.const_get(klass) rescue NameError raise if exception end when Symbol resolve_class(self.send(klass)) else raise "Unknown class: #{klass}" if exception end end # Helper to normalize `key` into a correct `ruby` **constant name** # @param key [String, Symbol] to be normalized # @return [String] a correct constant name def to_constant(key) key.to_s.strip.split(/[\-\_ ]/i).compact.map do |str| str.slice(0).upcase + str.slice(1..-1).downcase end.join end # Helper to create an instance variable `name` # @param [String, Symbol] the name of the variable # @reutrn [String] the name of the created instance variable def instance_variable_name(name) str = name.to_s str = "@#{str}" unless str.start_with?("@") str end # If the class for `name` exists, it returns it. Otherwise it generates it. # @param name [String, Symbol] the name of the new class # @param inherits [Class] the parent class to _inherit_ from # @param parent_space [String] parent namespace of the generated class, if not given: `self` # @yield [child_class] configure the new class # @yieldparam child_class [Class] the new class # @return [Class] the new generated class def new_class(name, inherits:, parent_space: nil) name = name.to_sym.freeze class_name = to_constant(name) parent_space = parent_space ? resolve_class(parent_space) : self full_class_name = "#{parent_space}::#{class_name}" unless (target_class = resolve_class(full_class_name, exception: false)) target_class = Class.new(inherits) parent_space.const_set class_name, target_class end target_class.tap do |klass| yield(klass) if block_given? end end # Finds all child classes of the current class. # @param parent_class [Class] the parent class we want to find children of. # @param direct [Boolean] it will only include direct child classes. # @param scope [nil, Array] to only look for descendants among the ones in `scope`. # @return [Arrary] the child classes in hierarchy order. def descendants(parent_class: self, direct: false, scope: nil) scope ||= ObjectSpace.each_object(::Class) return [] if scope.empty? scope.select do |klass| klass < parent_class end.sort do |k_1, k_2| next -1 if k_2 < k_1 next 1 if k_1 < k_2 0 end.tap do |siblings| if direct siblings.reject! do |si| siblings.any? {|s| si < s} end end end end # @param parent_class [Class] the parent class we want to find children of. # @param direct [Boolean] it will only include direct child classes. # @return [Boolean] `true` if the current class has child classes, and `false` otherwise. def descendants?(parent_class: self, direct: false) descendants(parent_class: parent_class, direct: direct).length.positive? end # Keeps track on class instance variables that should be inherited by child classes. # @note # - subclasses will inherit the value as is at that moment # - any change afterwards will be only on the specific class (in line with class instance variables) # - adapted from https://stackoverflow.com/a/10729812/4352306 # TODO: this separates the logic of the method to the instance var. # Think if would be possible to join them somehow. def inheritable_class_vars(*vars) @inheritable_class_vars ||= [:inheritable_class_vars] @inheritable_class_vars += vars end # Builds the attr_reader and attr_writer of `attrs` and registers # the associated instance variable as inheritable. def inheritable_attrs(*attrs) attrs.each do |attr| class_eval %( class << self; attr_accessor :#{attr} end ), __FILE__, __LINE__ - 2 end inheritable_class_vars(*attrs) end # This callback method is called whenever a subclass of the current class is created. # @note # - values of the instance variables are copied as they are (no dups or clones) # - the above means: avoid methods that change the state of the mutable object on it # - mutating methods would reflect the changes on other classes as well # - therefore, `freeze` will be called on the values that are inherited. def inherited(subclass) super inheritable_class_vars.each do |var| instance_var = instance_variable_name(var) value = instance_variable_get(instance_var) subclass.instance_variable_set(instance_var, value.freeze) end end end end end end