module RailsERD class Domain class Relationship class Cardinality extend Inspectable inspection_attributes :source_range, :destination_range N = Infinity = 1.0/0 # And beyond. CLASSES = { [1, 1] => :one_to_one, [1, N] => :one_to_many, [N, 1] => :many_to_one, [N, N] => :many_to_many } # @private :nodoc: # Returns a range that indicates the source (left) cardinality. attr_reader :source_range # Returns a range that indicates the destination (right) cardinality. attr_reader :destination_range # Create a new cardinality based on a source range and a destination # range. These ranges describe which number of values are valid. def initialize(source_range, destination_range) # @private :nodoc: @source_range = compose_range(source_range) @destination_range = compose_range(destination_range) end # Returns the name of this cardinality, based on its two cardinal # numbers (for source and destination). Can be any of # +:one_to_one:+, +:one_to_many+, or +:many_to_many+. The name # +:many_to_one+ also exists, but Rails ERD always normalises these # kinds of relationships by inverting them, so they become # +:one_to_many+ associations. # # You can also call the equivalent method with a question mark, which # will return true if the name corresponds to that method. For example: # # cardinality.one_to_one? # #=> true # cardinality.one_to_many? # #=> false def name CLASSES[cardinality_class] end # Returns +true+ if the source (left side) is not mandatory. def source_optional? source_range.first < 1 end # Returns +true+ if the destination (right side) is not mandatory. def destination_optional? destination_range.first < 1 end # Returns the inverse cardinality. Destination becomes source, source # becomes destination. def inverse self.class.new destination_range, source_range end CLASSES.each do |cardinality_class, name| class_eval <<-RUBY def #{name}? cardinality_class == #{cardinality_class.inspect} end RUBY end def ==(other) # @private :nodoc: source_range == other.source_range and destination_range == other.destination_range end def <=>(other) # @private :nodoc: (cardinality_class <=> other.cardinality_class).nonzero? or compare_with(other) { |x| x.source_range.first + x.destination_range.first }.nonzero? or compare_with(other) { |x| x.source_range.last + x.destination_range.last }.nonzero? or compare_with(other) { |x| x.source_range.last }.nonzero? or compare_with(other) { |x| x.destination_range.last } end # Returns an array with the cardinality classes for the source and # destination of this cardinality. Possible return values are: # [1, 1], [1, N], [N, N], and (in theory) # [N, 1]. def cardinality_class [source_cardinality_class, destination_cardinality_class] end protected # The cardinality class of the source (left side). Either +1+ or +Infinity+. def source_cardinality_class source_range.last == 1 ? 1 : N end # The cardinality class of the destination (right side). Either +1+ or +Infinity+. def destination_cardinality_class destination_range.last == 1 ? 1 : N end private def compose_range(r) return r..r if r.kind_of?(Integer) && r > 0 return (r.begin)..(r.end - 1) if r.exclude_end? r end def compare_with(other, &block) yield(self) <=> yield(other) end end end end end