module Foobara module TypeDeclarations module Handlers class ExtendModelTypeDeclaration < ExtendRegisteredTypeDeclaration class ModelClassDesugarizer < TypeDeclarations::Desugarizer def applicable?(sugary_type_declaration) sugary_type_declaration[:type] == expected_type_symbol end def expected_type_symbol :model end def default_model_base_class Foobara::Model end def desugarize(strictish_type_declaration) klass = strictish_type_declaration[:model_class] model_module = if strictish_type_declaration.key?(:model_module) mod = strictish_type_declaration[:model_module] mod = case mod when ::Module mod when ::String, ::Symbol Object.const_get(mod) when nil Object else # :nocov: raise ArgumentError, "expected #{mod} to be a module or module name" # :nocov: end if mod == GlobalDomain Object else mod end else Object end model_class = if klass && Object.const_defined?(klass) && Object.const_get(klass).is_a?(::Class) Object.const_get(klass) else model_base_class = strictish_type_declaration[:model_base_class] || default_model_base_class if model_base_class.is_a?(::String) || model_base_class.is_a?(::Symbol) model_base_class = Object.const_get(model_base_class) end # TODO: why not call this domain_module instead???? model_name = strictish_type_declaration[:name] existing_class = if model_module.const_defined?(model_name) model_module.const_get(model_name) end # TODO: This is a pretty crazy situation and hacky. Should come up with a better solution. # Here's the situation: if models are nested, like A::B and A and A::B are modules, then # if A::B is imported first, A will be created as a module via make_module_p. # But then when we are here creating A, A is already in use incorrectly as a module. # We need to move A out of the way but set all of its constants on the newly created model # A. if existing_class.is_a?(::Module) && !existing_class.is_a?(::Class) if existing_class.instance_variable_get( :@foobara_created_via_make_class ) existing_module_to_copy_over = existing_class parent_mod = Util.module_for(existing_class) parent_mod.send(:remove_const, Util.non_full_name(existing_class)) existing_class = nil end end # TODO: feels too destructive to be in a desugarizer. Technically probably should be in the # to type transformer instead. existing_class || model_base_class.subclass( **create_model_class_args(model_module:, type_declaration: strictish_type_declaration) ).tap do |model_class| if existing_module_to_copy_over Foobara::Domain::DomainModuleExtension._copy_constants( existing_module_to_copy_over, model_class ) end end end strictish_type_declaration[:model_class] = model_class.name model_module ||= Util.module_for(model_class) model_module = nil if model_module == Object strictish_type_declaration[:model_module] = model_module&.name strictish_type_declaration[:model_base_class] = model_class.superclass.name strictish_type_declaration end def create_model_class_args(model_module:, type_declaration:) { name: type_declaration[:name], model_module: } end def priority Priority::LOW end end end end end end