aggregations.rb 9.3 KB
Newer Older
D
Initial  
David Heinemeier Hansson 已提交
1 2
module ActiveRecord
  module Aggregations # :nodoc:
3
    def self.included(base)
4 5 6
      base.extend(ClassMethods)
    end

7 8 9
    def clear_aggregation_cache #:nodoc:
      self.class.reflect_on_all_aggregations.to_a.each do |assoc|
        instance_variable_set "@#{assoc.name}", nil
10
      end unless self.new_record?
11 12
    end

D
Initial  
David Heinemeier Hansson 已提交
13 14
    # Active Record implements aggregation through a macro-like class method called +composed_of+ for representing attributes 
    # as value objects. It expresses relationships like "Account [is] composed of Money [among other things]" or "Person [is]
D
David Heinemeier Hansson 已提交
15 16
    # composed of [an] address". Each call to the macro adds a description of how the value objects are created from the 
    # attributes of the entity object (when the entity is initialized either as a new object or from finding an existing object) 
D
Initial  
David Heinemeier Hansson 已提交
17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
    # and how it can be turned back into attributes (when the entity is saved to the database). Example:
    #
    #   class Customer < ActiveRecord::Base
    #     composed_of :balance, :class_name => "Money", :mapping => %w(balance amount)
    #     composed_of :address, :mapping => [ %w(address_street street), %w(address_city city) ]
    #   end
    #
    # The customer class now has the following methods to manipulate the value objects:
    # * <tt>Customer#balance, Customer#balance=(money)</tt>
    # * <tt>Customer#address, Customer#address=(address)</tt>
    #
    # These methods will operate with value objects like the ones described below:
    #
    #  class Money
    #    include Comparable
    #    attr_reader :amount, :currency
    #    EXCHANGE_RATES = { "USD_TO_DKK" => 6 }  
    # 
    #    def initialize(amount, currency = "USD") 
    #      @amount, @currency = amount, currency 
    #    end
    #
    #    def exchange_to(other_currency)
    #      exchanged_amount = (amount * EXCHANGE_RATES["#{currency}_TO_#{other_currency}"]).floor
    #      Money.new(exchanged_amount, other_currency)
    #    end
    #
    #    def ==(other_money)
    #      amount == other_money.amount && currency == other_money.currency
    #    end
    #
    #    def <=>(other_money)
    #      if currency == other_money.currency
    #        amount <=> amount
    #      else
    #        amount <=> other_money.exchange_to(currency).amount
    #      end
    #    end
    #  end
    #
    #  class Address
    #    attr_reader :street, :city
    #    def initialize(street, city) 
    #      @street, @city = street, city 
    #    end
    #
    #    def close_to?(other_address) 
    #      city == other_address.city 
    #    end
    #
    #    def ==(other_address)
    #      city == other_address.city && street == other_address.street
    #    end
    #  end
    #  
    # Now it's possible to access attributes from the database through the value objects instead. If you choose to name the
73
    # composition the same as the attribute's name, it will be the only way to access that attribute. That's the case with our
D
Initial  
David Heinemeier Hansson 已提交
74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
    # +balance+ attribute. You interact with the value objects just like you would any other attribute, though:
    #
    #   customer.balance = Money.new(20)     # sets the Money value object and the attribute
    #   customer.balance                     # => Money value object
    #   customer.balance.exchanged_to("DKK") # => Money.new(120, "DKK")
    #   customer.balance > Money.new(10)     # => true
    #   customer.balance == Money.new(20)    # => true
    #   customer.balance < Money.new(5)      # => false
    #
    # Value objects can also be composed of multiple attributes, such as the case of Address. The order of the mappings will
    # determine the order of the parameters. Example:
    #
    #   customer.address_street = "Hyancintvej"
    #   customer.address_city   = "Copenhagen"
    #   customer.address        # => Address.new("Hyancintvej", "Copenhagen")
    #   customer.address = Address.new("May Street", "Chicago")
    #   customer.address_street # => "May Street" 
    #   customer.address_city   # => "Chicago" 
    #
    # == Writing value objects
    #
95 96 97
    # Value objects are immutable and interchangeable objects that represent a given value, such as a +Money+ object representing
    # $5. Two +Money+ objects both representing $5 should be equal (through methods such as == and <=> from +Comparable+ if ranking
    # makes sense). This is unlike entity objects where equality is determined by identity. An entity class such as +Customer+ can
D
Initial  
David Heinemeier Hansson 已提交
98
    # easily have two different objects that both have an address on Hyancintvej. Entity identity is determined by object or
99
    # relational unique identifiers (such as primary keys). Normal <tt>ActiveRecord::Base</tt> classes are entity objects.
D
Initial  
David Heinemeier Hansson 已提交
100
    #
101 102
    # It's also important to treat the value objects as immutable. Don't allow the +Money+ object to have its amount changed after
    # creation. Create a new +Money+ object with the new value instead. This is exemplified by the <tt>Money#exchanged_to</tt> method that
D
Initial  
David Heinemeier Hansson 已提交
103
    # returns a new value object instead of changing its own values. Active Record won't persist value objects that have been
104
    # changed through means other than the writer method.
D
Initial  
David Heinemeier Hansson 已提交
105 106
    #
    # The immutable requirement is enforced by Active Record by freezing any object assigned as a value object. Attempting to 
107
    # change it afterwards will result in a <tt>ActiveSupport::FrozenObjectError</tt>.
D
Initial  
David Heinemeier Hansson 已提交
108 109 110
    # 
    # Read more about value objects on http://c2.com/cgi/wiki?ValueObject and on the dangers of not keeping value objects
    # immutable on http://c2.com/cgi/wiki?ValueObjectsShouldBeImmutable
111 112 113 114 115 116 117 118 119
    #
    # == Finding records by a value object
    #
    # Once a +composed_of+ relationship is specified for a model, records can be loaded from the database by specifying an instance
    # of the value object in the conditions hash. The following example finds all customers with +balance_amount+ equal to 20 and
    # +balance_currency+ equal to "USD":
    #
    #   Customer.find(:all, :conditions => {:balance => Money.new(20, "USD")})
    #
D
Initial  
David Heinemeier Hansson 已提交
120
    module ClassMethods
121 122
      # Adds reader and writer methods for manipulating a value object:
      # <tt>composed_of :address</tt> adds <tt>address</tt> and <tt>address=(new_address)</tt> methods.
D
Initial  
David Heinemeier Hansson 已提交
123 124
      #
      # Options are:
125
      # * <tt>:class_name</tt>  - specify the class name of the association. Use it only if that name can't be inferred
D
Initial  
David Heinemeier Hansson 已提交
126 127 128 129
      #   from the part id. So <tt>composed_of :address</tt> will by default be linked to the +Address+ class, but
      #   if the real class name is +CompanyAddress+, you'll have to specify it with this option.
      # * <tt>:mapping</tt> - specifies a number of mapping arrays (attribute, parameter) that bind an attribute name
      #   to a constructor parameter on the value class.
130
      # * <tt>:allow_nil</tt> - specifies that the aggregate object will not be instantiated when all mapped
131 132
      #   attributes are +nil+.  Setting the aggregate class to +nil+ has the effect of writing +nil+ to all mapped attributes.
      #   This defaults to +false+.
D
Initial  
David Heinemeier Hansson 已提交
133
      #
134
      # An optional block can be passed to convert the argument that is passed to the writer method into an instance of
135
      # <tt>:class_name</tt>. The block will only be called if the argument is not already an instance of <tt>:class_name</tt>.
136
      #
D
Initial  
David Heinemeier Hansson 已提交
137 138
      # Option examples:
      #   composed_of :temperature, :mapping => %w(reading celsius)
139
      #   composed_of(:balance, :class_name => "Money", :mapping => %w(balance amount)) {|balance| balance.to_money }
D
Initial  
David Heinemeier Hansson 已提交
140
      #   composed_of :address, :mapping => [ %w(address_street street), %w(address_city city) ]
141
      #   composed_of :gps_location
142 143
      #   composed_of :gps_location, :allow_nil => true
      #
144
      def composed_of(part_id, options = {}, &block)
145
        options.assert_valid_keys(:class_name, :mapping, :allow_nil)
D
Initial  
David Heinemeier Hansson 已提交
146 147

        name        = part_id.id2name
148
        class_name  = options[:class_name] || name.camelize
149
        mapping     = options[:mapping]    || [ name, name ]
150
        mapping     = [ mapping ] unless mapping.first.is_a?(Array)
151
        allow_nil   = options[:allow_nil]  || false
D
Initial  
David Heinemeier Hansson 已提交
152

153
        reader_method(name, class_name, mapping, allow_nil)
154
        writer_method(name, class_name, mapping, allow_nil, block)
155 156
        
        create_reflection(:composed_of, part_id, options, self)
D
Initial  
David Heinemeier Hansson 已提交
157 158 159
      end

      private
160
        def reader_method(name, class_name, mapping, allow_nil)
161 162 163 164 165
          module_eval do
            define_method(name) do |*args|
              force_reload = args.first || false
              if (instance_variable_get("@#{name}").nil? || force_reload) && (!allow_nil || mapping.any? {|pair| !read_attribute(pair.first).nil? })
                instance_variable_set("@#{name}", class_name.constantize.new(*mapping.collect {|pair| read_attribute(pair.first)}))
D
Initial  
David Heinemeier Hansson 已提交
166
              end
167
              instance_variable_get("@#{name}")
D
Initial  
David Heinemeier Hansson 已提交
168
            end
169
          end
J
Jeremy Kemper 已提交
170

171
        end
172

173 174 175 176
        def writer_method(name, class_name, mapping, allow_nil, conversion)
          module_eval do
            define_method("#{name}=") do |part|
              if part.nil? && allow_nil
177
                mapping.each { |pair| self[pair.first] = nil }
178 179 180
                instance_variable_set("@#{name}", nil)
              else
                part = conversion.call(part) unless part.is_a?(class_name.constantize) || conversion.nil?
181
                mapping.each { |pair| self[pair.first] = part.send(pair.last) }
182
                instance_variable_set("@#{name}", part.freeze)
183
              end
184
            end
185
          end
D
Initial  
David Heinemeier Hansson 已提交
186 187 188 189
        end
    end
  end
end