# encoding: utf-8
module Mongoid #:nodoc:
module Document
def self.included(base)
base.class_eval do
include ActiveSupport::Callbacks
include Associations, Attributes, Commands, Observable, Validatable
include InstanceMethods
extend ClassMethods
extend Finders
# Set up the class attributes that must be available to all subclasses.
# These include defaults, fields
class_inheritable_accessor :defaults, :fields
# The same collection is used for the entire class hierarchy.
cattr_accessor :_collection, :collection_name, :embedded, :primary_key
# Set the initial values. Defaults and fields get set to a
# +HashWithIndifferentAccess+ while the collection name will get set to
# the demodulized class.
self.defaults = {}.with_indifferent_access
self.fields = {}.with_indifferent_access
self.collection_name ||= self.to_s.demodulize.tableize
attr_accessor :association_name, :_parent
attr_reader :attributes, :new_record
delegate :collection, :defaults, :embedded?, :fields, :primary_key, :to => :klass
# Define all the callbacks that are accepted by the document.
define_callbacks :before_create, :before_destroy, :before_save, :before_update, :before_validation
define_callbacks :after_create, :after_destroy, :after_save, :after_update, :after_validation
index :_type
end
end
module ClassMethods
# Returns the collection associated with this +Document+. If the
# document is embedded, there will be no collection associated
# with it.
#
# Returns: Mongo::Collection
def collection
raise Errors::InvalidCollection.new(self) if embedded?
self._collection ||= Mongoid.database.collection(self.collection_name)
end
# return true if the +Document+ is embedded in another +Documnet+.
def embedded?
self.embedded == true
end
# Defines all the fields that are accessable on the Document
# For each field that is defined, a getter and setter will be
# added as an instance method to the Document.
#
# Options:
#
# name: The name of the field, as a +Symbol+.
# options: A +Hash+ of options to supply to the +Field+.
#
# Example:
#
# field :score, :default => 0
def field(name, options = {})
set_field(name, options)
set_default(name, options)
end
# Returns a human readable version of the class.
def human_name
name.underscore.humanize
end
# Adds an index on the field specified. Options can be :unique => true or
# :unique => false. It will default to the latter.
def index(name, options = { :unique => false })
collection.create_index(name, options[:unique])
end
# Instantiate a new object, only when loaded from the database.
def instantiate(attrs = {}, allocating = false)
attributes = attrs.with_indifferent_access
if attributes[:_id] || allocating
document = allocate
document.instance_variable_set(:@attributes, attributes)
return document
else
return new(attributes)
end
end
# Defines the field that will be used for the id of this +Document+. This
# set the id of this +Document+ before save to a parameterized version of
# the field that was supplied. This is good for use for readable URLS in
# web applications and *MUST* be defined on documents that are embedded
# in order for proper updates in has_may associations.
def key(*fields)
self.primary_key = fields
before_save :generate_key
end
# Macro for setting the collection name to store in.
def store_in(name)
self.collection_name = name.to_s
end
# Returns all types to query for when using this class as the base.
def _types
@_type ||= (self.subclasses + [ self.name ])
end
protected
# Define a field attribute for the +Document+.
def set_field(name, options = {})
meth = options.delete(:as) || name
fields[name] = Field.new(name.to_s, options)
create_accessors(name, meth, options)
end
# Create the field accessors.
def create_accessors(name, meth, options = {})
define_method(meth) { read_attribute(name) }
define_method("#{meth}=") { |value| write_attribute(name, value) }
define_method("#{meth}?") { read_attribute(name) == true } if options[:type] == Boolean
end
# Set up a default value for a field.
def set_default(name, options = {})
value = options[:default]
defaults[name] = value if value
end
end
module InstanceMethods
# Performs equality checking on the attributes. For now we chack against
# all attributes excluding timestamps on the object.
def ==(other)
return false unless other.is_a?(Document)
@attributes.except(:modified_at).except(:created_at) ==
other.attributes.except(:modified_at).except(:created_at)
end
# Introduces a child object into the +Document+ object graph. This will
# set up the relationships between the parent and child and update the
# attributes of the parent +Document+.
#
# Options:
#
# parent: The +Document+ to assimilate with.
# options: The association +Options+ for the child.
#
# Example:
#
# name.assimilate(person, options)
#
# Returns: The child +Document+.
def assimilate(parent, options)
parentize(parent, options.name); notify; self
end
# Clone the current +Document+. This will return all attributes with the
# exception of the document's id and versions.
def clone
self.class.instantiate(@attributes.except(:_id).except(:versions).dup, true)
end
# Get the id associated with this object. This will pull the _id value out
# of the attributes +Hash+.
def id
@attributes[:_id]
end
# Set the id
def id=(new_id)
@attributes[:_id] = new_id
end
alias :_id :id
alias :_id= :id=
# Instantiate a new +Document+, setting the Document's attributes if
# given. If no attributes are provided, they will be initialized with
# an empty +Hash+.
#
# If a primary key is defined, the document's id will be set to that key,
# otherwise it will be set to a fresh +Mongo::ObjectID+ string.
#
# Options:
#
# attrs: The attributes +Hash+ to set up the document with.
#
# Example:
#
# Person.new(:title => "Mr", :age => 30)
def initialize(attrs = {})
@attributes = {}.with_indifferent_access
process(defaults.merge(attrs))
@new_record = true if id.nil?
document = yield self if block_given?
generate_key; generate_type; document
end
# Returns the class name plus its attributes.
def inspect
attrs = fields.map { |name, field| "#{name}: #{@attributes[name] || 'nil'}" } * ", "
"#<#{self.class.name} _id: #{id}, #{attrs}>"
end
# Returns true is the +Document+ has not been persisted to the database,
# false if it has. This is determined by the variable @new_record
# and NOT if the object has an id.
def new_record?
@new_record == true
end
# Sets the new_record boolean - used after document is saved.
def new_record=(saved)
@new_record = saved
end
# Set the changed state of the +Document+ then notify observers that it has changed.
#
# Example:
#
# person.notify
def notify
changed(true)
notify_observers(self)
end
# Sets up a child/parent association. This is used for newly created
# objects so they can be properly added to the graph and have the parent
# observers set up properly.
#
# Options:
#
# abject: The parent object that needs to be set for the child.
# association_name: The name of the association for the child.
#
# Example:
#
# address.parentize(person, :addresses)
def parentize(object, association_name)
self._parent = object
self.association_name = association_name
add_observer(object)
end
# Reloads the +Document+ attributes from the database.
def reload
@attributes = collection.find_one(:_id => id).with_indifferent_access
end
# Remove a child document from this parent +Document+. Will reset the
# memoized association and notify the parent of the change.
def remove(child)
name = child.association_name
@attributes.remove(name, child.attributes)
remove_instance_variable("@#{name}")
notify
end
# Return the root +Document+ in the object graph. If the current +Document+
# is the root object in the graph it will return self.
def _root
object = self
while (object._parent) do object = object._parent; end
object || self
end
# Return an array with this +Document+ only in it.
def to_a
[ self ]
end
# Returns the id of the Document, used in Rails compatibility.
def to_param
id
end
# Returns the object type.
def _type
@attributes[:_type]
end
# Set the type.
def _type=(new_type)
@attributes[:_type] = new_type
end
# Observe a notify call from a child +Document+. This will either update
# existing attributes on the +Document+ or clear them out for the child if
# the clear boolean is provided.
#
# Options:
#
# child: The child +Document+ that sent the notification.
# clear: Will clear out the child's attributes if set to true.
#
# Example:
#
# person.notify_observers(self) will cause this method to execute.
#
# This will also cause the observing +Document+ to notify it's parent if
# there is any.
def update(child, clear = false)
name = child.association_name
clear ? @attributes.delete(name) : @attributes.insert(name, child.attributes)
notify
end
# Needs to run the appropriate callbacks the delegate up to the validatable
# gem.
def valid?
run_callbacks(:before_validation)
result = super
run_callbacks(:after_validation)
result
end
protected
def generate_key
if primary_key
values = primary_key.collect { |key| @attributes[key] }
@attributes[:_id] = values.join(" ").parameterize.to_s
else
@attributes[:_id] = Mongo::ObjectID.new.to_s unless id
end
end
def generate_type
@attributes[:_type] ||= self.class.name
end
# Convenience method to get the document's class
def klass
self.class
end
end
end
end