module CouchPotato class Database class ValidationsFailedError < ::StandardError; end def initialize(couchrest_database) @couchrest_database = couchrest_database end # executes a view and return the results. you pass in a view spec # which is usually a result of a SomePersistentClass.some_view call. # # Example: # # class User # include CouchPotato::Persistence # property :age # view :all, key: :age # end # db = CouchPotato.database # # db.view(User.all) # => [user1, user2] # # You can pass the usual parameters you can pass to a couchdb view to the view: # # db.view(User.all(limit: 5, startkey: 2, reduce: false)) # # For your convenience when passing a hash with only a key parameter you can just pass in the value # # db.view(User.all(key: 1)) == db.view(User.all(1)) # # Instead of passing a startkey and endkey you can pass in a key with a range: # # db.view(User.all(key: 1..20)) == db.view(startkey: 1, endkey: 20) == db.view(User.all(1..20)) # # You can also pass in multiple keys: # # db.view(User.all(keys: [1, 2, 3])) def view(spec) ActiveSupport::Notifications.instrument('couch_potato.view', name: "#{spec.design_document}/#{spec.view_name}") do results = CouchPotato::View::ViewQuery.new( couchrest_database, spec.design_document, {spec.view_name => { :map => spec.map_function, :reduce => spec.reduce_function } }, ({spec.list_name => spec.list_function} unless spec.list_name.nil?), spec.lib, spec.language ).query_view!(spec.view_parameters) processed_results = spec.process_results results processed_results.each do |document| document.database = self if document.respond_to?(:database=) end if processed_results.respond_to?(:each) processed_results end end # returns the first result from a #view query or nil def first(spec) spec.view_parameters = spec.view_parameters.merge({:limit => 1}) view(spec).first end # returns th first result from a #view or raises CouchPotato::NotFound def first!(spec) first(spec) || raise(CouchPotato::NotFound) end # saves a document. returns true on success, false on failure. # if passed a block will: # * yield the object to be saved to the block and run if once before saving # * on conflict: reload the document, run the block again and retry saving def save_document(document, validate = true, retries = 0, &block) begin block.call document if block save_document_without_conflict_handling(document, validate) rescue CouchRest::Conflict => e if block handle_write_conflict document, validate, retries, &block else raise CouchPotato::Conflict.new end end end alias_method :save, :save_document # saves a document, raises a CouchPotato::Database::ValidationsFailedError on failure def save_document!(document) save_document(document) || raise(ValidationsFailedError.new(document.errors.full_messages)) end alias_method :save!, :save_document! def destroy_document(document) begin destroy_document_without_conflict_handling document rescue CouchRest::Conflict retry if document = document.reload end end alias_method :destroy, :destroy_document # loads a document by its id(s) # id - either a single id or an array of ids # returns either a single document or an array of documents (if an array of ids was passed). # returns nil if the single document could not be found. when passing an array and some documents # could not be found these are omitted from the returned array def load_document(id) raise "Can't load a document without an id (got nil)" if id.nil? ActiveSupport::Notifications.instrument('couch_potato.load') do if id.is_a?(Array) bulk_load id else instance = couchrest_database.get(id) instance.database = self if instance instance end end end alias_method :load, :load_document # loads one or more documents by its id(s) # behaves like #load except it raises a CouchPotato::NotFound if any of the documents could not be found def load!(id) doc = load(id) if id.is_a?(Array) missing_docs = id - doc.map(&:id) end raise(CouchPotato::NotFound, missing_docs.try(:join, ', ')) if doc.nil? || missing_docs.try(:any?) doc end def inspect #:nodoc: "#" end # returns the underlying CouchRest::Database instance def couchrest_database @couchrest_database end private def handle_write_conflict(document, validate, retries, &block) document = document.reload if retries == 5 raise CouchPotato::Conflict.new else save_document document, validate, retries + 1, &block end end def destroy_document_without_conflict_handling(document) document.run_callbacks :destroy do document._deleted = true couchrest_database.delete_doc document.to_hash end document._id = nil document._rev = nil end def save_document_without_conflict_handling(document, validate = true) if document.new? create_document(document, validate) else update_document(document, validate) end end def bulk_load(ids) response = couchrest_database.bulk_load ids docs = response['rows'].map{|row| row["doc"]}.compact docs.each{|doc| doc.database = self if doc.respond_to?(:database=) } end def create_document(document, validate) document.database = self if validate document.errors.clear return false if false == document.run_callbacks(:validation_on_save) do return false if false == document.run_callbacks(:validation_on_create) do return false unless valid_document?(document) end end end return false if false == document.run_callbacks(:save) do return false if false == document.run_callbacks(:create) do res = couchrest_database.save_doc document.to_hash document._rev = res['rev'] document._id = res['id'] end end true end def update_document(document, validate) if validate document.errors.clear return false if false == document.run_callbacks(:validation_on_save) do return false if false == document.run_callbacks(:validation_on_update) do return false unless valid_document?(document) end end end return false if false == document.run_callbacks(:save) do return false if false == document.run_callbacks(:update) do if document.dirty? res = couchrest_database.save_doc document.to_hash document._rev = res['rev'] end end end true end def valid_document?(document) errors = document.errors.errors.dup errors.instance_variable_set("@messages", errors.messages.dup) if errors.respond_to?(:messages) document.valid? errors.each do |k, v| if v.respond_to?(:each) v.each {|message| document.errors.add(k, message)} else document.errors.add(k, v) end end document.errors.empty? end end end