2004-11-23 20:04:44 -05:00
|
|
|
module ActiveRecord
|
|
|
|
module Aggregations # :nodoc:
|
2005-12-02 23:29:55 -05:00
|
|
|
def self.included(base)
|
2005-04-02 04:29:43 -05:00
|
|
|
base.extend(ClassMethods)
|
|
|
|
end
|
|
|
|
|
2005-12-07 23:46:57 -05:00
|
|
|
def clear_aggregation_cache #:nodoc:
|
|
|
|
self.class.reflect_on_all_aggregations.to_a.each do |assoc|
|
|
|
|
instance_variable_set "@#{assoc.name}", nil
|
2006-09-05 14:54:24 -04:00
|
|
|
end unless self.new_record?
|
2005-12-07 23:46:57 -05:00
|
|
|
end
|
|
|
|
|
2004-11-23 20:04:44 -05:00
|
|
|
# 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]
|
2005-10-26 09:05:48 -04:00
|
|
|
# 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)
|
2004-11-23 20:04:44 -05:00
|
|
|
# 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
|
2007-08-28 19:18:57 -04:00
|
|
|
# composition the same as the attribute's name, it will be the only way to access that attribute. That's the case with our
|
2004-11-23 20:04:44 -05:00
|
|
|
# +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
|
|
|
|
#
|
2008-05-25 07:29:00 -04:00
|
|
|
# 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 <tt>==</tt> and <tt><=></tt> from Comparable if ranking
|
|
|
|
# makes sense). This is unlike entity objects where equality is determined by identity. An entity class such as Customer can
|
2004-11-23 20:04:44 -05:00
|
|
|
# easily have two different objects that both have an address on Hyancintvej. Entity identity is determined by object or
|
2008-05-25 07:29:00 -04:00
|
|
|
# relational unique identifiers (such as primary keys). Normal ActiveRecord::Base classes are entity objects.
|
2004-11-23 20:04:44 -05:00
|
|
|
#
|
2008-05-25 07:29:00 -04:00
|
|
|
# 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 Money#exchanged_to method that
|
2004-11-23 20:04:44 -05:00
|
|
|
# returns a new value object instead of changing its own values. Active Record won't persist value objects that have been
|
2007-08-28 19:18:57 -04:00
|
|
|
# changed through means other than the writer method.
|
2004-11-23 20:04:44 -05:00
|
|
|
#
|
|
|
|
# The immutable requirement is enforced by Active Record by freezing any object assigned as a value object. Attempting to
|
2008-05-25 07:29:00 -04:00
|
|
|
# change it afterwards will result in a ActiveSupport::FrozenObjectError.
|
2004-11-23 20:04:44 -05:00
|
|
|
#
|
|
|
|
# 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
|
2008-01-18 22:45:24 -05:00
|
|
|
#
|
|
|
|
# == 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")})
|
|
|
|
#
|
2004-11-23 20:04:44 -05:00
|
|
|
module ClassMethods
|
2006-07-05 22:05:09 -04:00
|
|
|
# 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.
|
2004-11-23 20:04:44 -05:00
|
|
|
#
|
|
|
|
# Options are:
|
2005-02-07 09:15:53 -05:00
|
|
|
# * <tt>:class_name</tt> - specify the class name of the association. Use it only if that name can't be inferred
|
2008-05-25 07:29:00 -04:00
|
|
|
# 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.
|
2004-11-23 20:04:44 -05:00
|
|
|
# * <tt>:mapping</tt> - specifies a number of mapping arrays (attribute, parameter) that bind an attribute name
|
|
|
|
# to a constructor parameter on the value class.
|
2006-05-21 13:32:37 -04:00
|
|
|
# * <tt>:allow_nil</tt> - specifies that the aggregate object will not be instantiated when all mapped
|
2007-08-28 19:18:57 -04:00
|
|
|
# attributes are +nil+. Setting the aggregate class to +nil+ has the effect of writing +nil+ to all mapped attributes.
|
|
|
|
# This defaults to +false+.
|
2004-11-23 20:04:44 -05:00
|
|
|
#
|
2007-10-23 13:39:35 -04:00
|
|
|
# An optional block can be passed to convert the argument that is passed to the writer method into an instance of
|
2007-12-05 13:54:41 -05:00
|
|
|
# <tt>:class_name</tt>. The block will only be called if the argument is not already an instance of <tt>:class_name</tt>.
|
2007-10-23 13:39:35 -04:00
|
|
|
#
|
2004-11-23 20:04:44 -05:00
|
|
|
# Option examples:
|
|
|
|
# composed_of :temperature, :mapping => %w(reading celsius)
|
2007-10-23 13:39:35 -04:00
|
|
|
# composed_of(:balance, :class_name => "Money", :mapping => %w(balance amount)) {|balance| balance.to_money }
|
2004-11-23 20:04:44 -05:00
|
|
|
# composed_of :address, :mapping => [ %w(address_street street), %w(address_city city) ]
|
2005-03-01 18:52:36 -05:00
|
|
|
# composed_of :gps_location
|
2006-05-21 13:32:37 -04:00
|
|
|
# composed_of :gps_location, :allow_nil => true
|
|
|
|
#
|
2007-10-23 13:39:35 -04:00
|
|
|
def composed_of(part_id, options = {}, &block)
|
2006-05-21 13:32:37 -04:00
|
|
|
options.assert_valid_keys(:class_name, :mapping, :allow_nil)
|
2004-11-23 20:04:44 -05:00
|
|
|
|
|
|
|
name = part_id.id2name
|
2006-04-29 16:34:31 -04:00
|
|
|
class_name = options[:class_name] || name.camelize
|
2006-05-21 13:32:37 -04:00
|
|
|
mapping = options[:mapping] || [ name, name ]
|
2007-10-23 13:39:35 -04:00
|
|
|
mapping = [ mapping ] unless mapping.first.is_a?(Array)
|
2006-05-21 13:32:37 -04:00
|
|
|
allow_nil = options[:allow_nil] || false
|
2004-11-23 20:04:44 -05:00
|
|
|
|
2006-05-21 13:32:37 -04:00
|
|
|
reader_method(name, class_name, mapping, allow_nil)
|
2007-10-23 13:39:35 -04:00
|
|
|
writer_method(name, class_name, mapping, allow_nil, block)
|
2005-12-02 23:29:55 -05:00
|
|
|
|
|
|
|
create_reflection(:composed_of, part_id, options, self)
|
2004-11-23 20:04:44 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
2006-05-21 13:32:37 -04:00
|
|
|
def reader_method(name, class_name, mapping, allow_nil)
|
2007-10-23 13:39:35 -04:00
|
|
|
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)}))
|
2004-11-23 20:04:44 -05:00
|
|
|
end
|
2007-12-29 00:06:06 -05:00
|
|
|
instance_variable_get("@#{name}")
|
2004-11-23 20:04:44 -05:00
|
|
|
end
|
2007-10-23 13:39:35 -04:00
|
|
|
end
|
2007-10-10 19:01:18 -04:00
|
|
|
|
2007-10-23 13:39:35 -04:00
|
|
|
end
|
2006-05-21 13:32:37 -04:00
|
|
|
|
2007-10-23 13:39:35 -04:00
|
|
|
def writer_method(name, class_name, mapping, allow_nil, conversion)
|
|
|
|
module_eval do
|
|
|
|
define_method("#{name}=") do |part|
|
|
|
|
if part.nil? && allow_nil
|
2008-03-30 21:10:04 -04:00
|
|
|
mapping.each { |pair| self[pair.first] = nil }
|
2007-10-23 13:39:35 -04:00
|
|
|
instance_variable_set("@#{name}", nil)
|
|
|
|
else
|
|
|
|
part = conversion.call(part) unless part.is_a?(class_name.constantize) || conversion.nil?
|
2008-03-30 21:10:04 -04:00
|
|
|
mapping.each { |pair| self[pair.first] = part.send(pair.last) }
|
2007-10-23 13:39:35 -04:00
|
|
|
instance_variable_set("@#{name}", part.freeze)
|
2006-05-21 13:32:37 -04:00
|
|
|
end
|
2007-10-23 13:39:35 -04:00
|
|
|
end
|
2006-05-21 13:32:37 -04:00
|
|
|
end
|
2004-11-23 20:04:44 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|