# Copyright 2010 Sean Cribbs, Sonian Inc., and Basho Technologies, Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
require 'ripple'
module Ripple
# Raised by find! when a document cannot be found with the given key.
# begin
# Example.find!('badkey')
# rescue Ripple::DocumentNotFound
# puts 'No Document here!'
# end
class DocumentNotFound < StandardError
include Translation
def initialize(keys, found)
if keys.empty?
super(t("document_not_found.no_key"))
elsif keys.one?
super(t("document_not_found.one_key", :key => keys.first))
else
missing = keys - found.compact.map(&:key)
super(t("document_not_found.many_keys", :keys => missing.join(', ')))
end
end
end
module Document
module Finders
extend ActiveSupport::Concern
module ClassMethods
# Retrieve single or multiple documents from Riak.
# @overload find(key)
# Find a single document.
# @param [String] key the key of a document to find
# @return [Document] the found document, or nil
# @overload find(key1, key2, ...)
# Find a list of documents.
# @param [String] key1 the key of a document to find
# @param [String] key2 the key of a document to find
# @return [Array] a list of found documents, including nil for missing documents
# @overload find(keylist)
# Find a list of documents.
# @param [Array] keylist an array of keys to find
# @return [Array] a list of found documents, including nil for missing documents
def find(*args)
args.flatten!
return nil if args.empty? || args.all?(&:blank?)
return find_one(args.first) if args.one?
args.map {|key| find_one(key) }
end
# Retrieve single or multiple documents from Riak
# but raise Ripple::DocumentNotFound if a key can
# not be found in the bucket.
def find!(*args)
found = find(*args)
raise DocumentNotFound.new(args, found) if !found || Array(found).include?(nil)
found
end
# Find the first object using the first key in the
# bucket's keys using find. You should not expect to
# actually get the first object you added to the bucket.
# This is just a convenience method.
def first
find(bucket.keys.first)
end
# Find the first object using the first key in the
# bucket's keys using find!
def first!
find!(bucket.keys.first)
end
# Find all documents in the Document's bucket and return them.
# @overload all()
# Get all documents and return them in an array.
# @return [Array] all found documents in the bucket
# @overload all() {|doc| ... }
# Stream all documents in the bucket through the block.
# @yield [Document] doc a found document
def all
if block_given?
bucket.keys do |keys|
keys.each do |key|
obj = find_one(key)
yield obj if obj
end
end
[]
else
bucket.keys.inject([]) do |acc, k|
obj = find_one(k)
obj ? acc << obj : acc
end
end
end
private
def find_one(key)
instantiate(bucket.get(key, quorums.slice(:r)))
rescue Riak::FailedRequest => fr
return nil if fr.code.to_i == 404
raise fr
end
def instantiate(robject)
klass = robject.data['_type'].constantize rescue self
data = {'key' => robject.key}
data.reverse_merge!(robject.data) rescue data
klass.new(data).tap do |doc|
doc.key = data['key']
doc.instance_variable_set(:@new, false)
doc.instance_variable_set(:@robject, robject)
end
end
end
end
end
end