module ActiveFedora::AutosaveAssociation::ClassMethods

def add_autosave_association_callbacks(reflection)

before actually defining them.
check if the save or validation methods have already been defined
the callbacks to get defined multiple times, there are guards that
called _after_ the association has been defined. Since we don't want
this can change, for instance, when using nested attributes, which is
get created when they are invoked for the very first time. However,
However the validation and callback methods are lazy and those methods
For performance reasons, we don't check whether to validate at runtime.

the +reflection+.
Adds validation and save callbacks for the association as specified by
def add_autosave_association_callbacks(reflection)
  save_method = :"autosave_associated_records_for_#{reflection.name}"
  validation_method = :"validate_associated_records_for_#{reflection.name}"
  collection = reflection.collection?
  unless method_defined?(save_method)
    if collection
      before_save :before_save_collection_association
      define_non_cyclic_method(save_method, reflection) { save_collection_association(reflection) }
      # Doesn't use after_save as that would save associations added in after_create/after_update twice
      after_create save_method
      after_update save_method
    else
      define_non_cyclic_method(save_method, reflection) { save_belongs_to_association(reflection) }
      before_save save_method
    end
  end
  if reflection.validate? && !method_defined?(validation_method)
    method = (collection ? :validate_collection_association : :validate_single_association)
    define_non_cyclic_method(validation_method, reflection) { send(method, reflection) }
    validate validation_method
  end
end

def define_non_cyclic_method(name, reflection, &block)

def define_non_cyclic_method(name, reflection, &block)
  define_method(name) do |*args|
    result = true; @_already_called ||= {}
    # Loop prevention for validation of associations
    unless @_already_called[[name, reflection.name]]
      begin
        @_already_called[[name, reflection.name]]=true
        result = instance_eval(&block)
      ensure
        @_already_called[[name, reflection.name]]=false
      end
    end
    result
  end
end