# encoding: utf-8 require "mongoid/commands/create" require "mongoid/commands/deletion" require "mongoid/commands/delete" require "mongoid/commands/delete_all" require "mongoid/commands/destroy" require "mongoid/commands/destroy_all" require "mongoid/commands/save" module Mongoid #:nodoc: # This module is included in the +Document+ to provide all the persistence # methods required on the +Document+ object and class. module Commands def self.included(base) base.class_eval do include InstanceMethods extend ClassMethods end end module InstanceMethods # Delete the +Document+ from the database. This method is an optimized # delete that does not force any callbacks. # # Example: # # document.delete # # Returns: true unless an error occurs. def delete Delete.execute(self) end # Destroy the +Document+. This will delete the document from the database # and run the before and after destroy callbacks. # # Example: # # document.destroy # # Returns: true unless an error occurs. def destroy Destroy.execute(self) end # Save the +Document+. If the document is new, then the before and after # create callbacks will get executed as well as the save callbacks. # Otherwise only the save callbacks will run. # # Options: # # validate: Run validations or not. Defaults to true. # # Example: # # document.save # save with validations # document.save(false) # save without validations # # Returns: true if validation passes, false if not. def save(validate = true) new = new_record? run_callbacks(:before_create) if new begin saved = Save.execute(self, validate) rescue Mongo::OperationFailure => e errors.add(:mongoid, e.message) end run_callbacks(:after_create) if new saved end # Save the +Document+, dangerously. Before and after save callbacks will # get run. If validation fails an error will get raised. # # Example: # # document.save! # # Returns: true if validation passes def save! return save(true) || (raise Errors::Validations.new(self.errors)) end # Update the document attributes and persist the document to the # database. Will delegate to save with all callbacks. # # Example: # # document.update_attributes(:title => "Test") def update_attributes(attrs = {}) set_attributes(attrs); save end # Update the document attributes and persist the document to the # database. Will delegate to save! # # Example: # # document.update_attributes!(:title => "Test") def update_attributes!(attrs = {}) set_attributes(attrs); save! end protected def set_attributes(attrs = {}) run_callbacks(:before_update) write_attributes(attrs) run_callbacks(:after_update) end end module ClassMethods # Create a new +Document+. This will instantiate a new document and save # it in a single call. Will always return the document whether save # passed or not. # # Example: # # Person.create(:title => "Mr") # # Returns: the +Document+. def create(attributes = {}) document = new(attributes) begin Create.execute(document) rescue Mongo::OperationFailure => e document.errors.add(:mongoid, e.message) end document end # Create a new +Document+. This will instantiate a new document and save # it in a single call. Will always return the document whether save # passed or not. Will raise an error if validation fails. # # Example: # # Person.create!(:title => "Mr") # # Returns: the +Document+. def create!(attributes = {}) document = Create.execute(new(attributes), true) raise Errors::Validations.new(document.errors) unless document.errors.empty? document end # Delete all documents given the supplied conditions. If no conditions # are passed, the entire collection will be dropped for performance # benefits. Does not fire any callbacks. # # Example: # # Person.delete_all(:conditions => { :title => "Sir" }) # Person.delete_all # # Returns: true or raises an error. def delete_all(conditions = {}) DeleteAll.execute(self, conditions) end # Delete all documents given the supplied conditions. If no conditions # are passed, the entire collection will be dropped for performance # benefits. Fires the destroy callbacks if conditions were passed. # # Example: # # Person.destroy_all(:conditions => { :title => "Sir" }) # Person.destroy_all # # Returns: true or raises an error. def destroy_all(conditions = {}) DestroyAll.execute(self, conditions) end end end end