module CouchRest
module Model
module Designs
#
# A proxy class that allows view queries to be created using
# chained method calls. After each call a new instance of the method
# is created based on the original in a similar fashion to ruby's Sequel
# library, or Rails 3's Arel.
#
# CouchDB views have inherent limitations, so joins and filters as used in
# a normal relational database are not possible.
#
class View
include Enumerable
attr_accessor :owner, :model, :design_doc, :name, :query, :result
# Initialize a new View object. This method should not be called from
# outside CouchRest Model.
def initialize(design_doc, parent, new_query = {}, name = nil)
self.design_doc = design_doc
# Extract options from query
proxy = new_query.delete(:proxy)
delete = new_query.delete(:delete)
if parent.is_a?(Class) && parent < CouchRest::Model::Base
raise "Name must be provided for view to be initialized" if name.nil?
self.owner = (proxy || parent)
self.model = parent
self.name = name.to_s
# Default options:
self.query = { }
elsif parent.is_a?(self.class)
self.owner = (proxy || parent.owner)
self.model = parent.model
self.name = parent.name
self.query = parent.query.dup
else
raise "View cannot be initialized without a parent Model or View"
end
# Update the local query hash
query.update(new_query)
delete.each{|k| query.delete(k)} if delete
super()
end
# == View Execution Methods
#
# Request to the CouchDB database using the current query values.
# Return each row wrapped in a ViewRow object. Unlike the raw
# CouchDB request, this will provide an empty array if there
# are no results.
def rows
return @rows if @rows
if block_given?
execute do |row|
yield ViewRow.new(row, owner)
end
else
if execute && result['rows']
@rows ||= result['rows'].map{|v| ViewRow.new(v, owner)}
else
[ ]
end
end
end
# Fetch all the documents the view can access. If the view has
# not already been prepared for including documents in the query,
# it will be added automatically and reset any previously cached
# results.
def all(&block)
include_docs!
docs(&block)
end
# Provide all the documents from the view. If the view has not been
# prepared with the +include_docs+ option, each document will be
# loaded individually.
def docs
if block_given?
rows do |row|
yield row.doc
end
else
@docs ||= rows.map{|r| r.doc}
end
end
# If another request has been made on the view, this will return
# the first document in the set. If not, a new query object will be
# generated with a limit of 1 so that only the first document is
# loaded.
def first
result ? all.first : limit(1).all.first
end
# Same as first but will order the view in descending order. This
# does not however reverse the search keys or the offset, so if you
# are using a +startkey+ and +endkey+ you might end up with
# unexpected results.
#
# If in doubt, don't use this method!
#
def last
result ? all.last : limit(1).descending.all.last
end
# Return the number of documents in the currently defined result set.
# Use #count for the total number of documents regardless
# of the current limit defined.
def length
docs.length
end
# Perform a count operation based on the current view. If the view
# can be reduced, the reduce will be performed and return the first
# value. This is okay for most simple queries, but may provide
# unexpected results if your reduce method does not calculate
# the total number of documents in a result set.
#
# Trying to use this method with the group option will raise an error.
#
# If no reduce function is defined, a query will be performed
# to return the total number of rows, this is the equivalant of:
#
# view.limit(0).total_rows
#
def count
raise "View#count cannot be used with group options" if query[:group]
if can_reduce?
row = reduce.skip(0).limit(1).rows.first
row.nil? ? 0 : row.value
else
limit(0).total_rows
end
end
# Check to see if the array of documents is empty. This *will*
# perform the query and return all documents ready to use, if you don't
# want to load anything, use +#total_rows+ or +#count+ instead.
def empty?
all.empty?
end
# Run through each document provided by the +#all+ method.
# This is also used by the Enumerator mixin to provide all the standard
# ruby collection directly on the view.
def each(&block)
all.each(&block)
end
# Wrapper for the results offset. As per the CouchDB API,
# this may be nil if groups are used.
def offset
execute['offset']
end
# Wrapper for the total_rows value provided by the query. As per the
# CouchDB API, this may be nil if groups are used.
def total_rows
execute['total_rows']
end
# Convenience wrapper to provide all the values from the route
# set without having to go through +rows+.
def values
rows.map{|r| r.value}
end
# Accept requests as if the view was an array. Used for backwards compatibity
# with older queries:
#
# Model.all(:raw => true, :limit => 0)['total_rows']
#
# In this example, the raw option will be ignored, and the total rows
# will still be accessible.
#
def [](value)
execute[value]
end
# No yet implemented. Eventually this will provide a raw hash
# of the information CouchDB holds about the view.
def info
raise "Not yet implemented"
end
# == View Filter Methods
#
# View filters return a copy of the view instance with the query
# modified appropriatly. Errors will be raised if the methods
# are combined in an incorrect fashion.
#
# Find all entries in the index whose key matches the value provided.
#
# Cannot be used when the +#startkey+ or +#endkey+ have been set.
def key(value)
raise "View#key cannot be used when startkey or endkey have been set" unless query[:keys].nil? && query[:startkey].nil? && query[:endkey].nil?
update_query(:key => value)
end
# Find all index keys that start with the value provided. May or may
# not be used in conjunction with the +endkey+ option.
#
# When the +#descending+ option is used (not the default), the start
# and end keys should be reversed, as per the CouchDB API.
#
# Cannot be used if the key has been set.
def startkey(value)
raise "View#startkey cannot be used when key has been set" unless query[:key].nil? && query[:keys].nil?
update_query(:startkey => value)
end
# The result set should start from the position of the provided document.
# The value may be provided as an object that responds to the +#id+ call
# or a string.
def startkey_doc(value)
update_query(:startkey_docid => value.is_a?(String) ? value : value.id)
end
# The opposite of +#startkey+, finds all index entries whose key is before
# the value specified.
#
# See the +#startkey+ method for more details and the +#inclusive_end+
# option.
def endkey(value)
raise "View#endkey cannot be used when key has been set" unless query[:key].nil? && query[:keys].nil?
update_query(:endkey => value)
end
# The result set should end at the position of the provided document.
# The value may be provided as an object that responds to the +#id+
# call or a string.
def endkey_doc(value)
update_query(:endkey_docid => value.is_a?(String) ? value : value.id)
end
# Keys is a special CouchDB option that will cause the view request to be POSTed
# including an array of keys. Only documents with the matching keys will be
# returned. This is much faster than sending multiple requests for a set
# non-consecutive documents.
#
# If no values are provided, this method will act as a wrapper around
# the rows result set, providing an array of keys.
def keys(*keys)
if keys.empty?
rows.map{|r| r.key}
else
raise "View#keys cannot by used when key or startkey/endkey have been set" unless query[:key].nil? && query[:startkey].nil? && query[:endkey].nil?
update_query(:keys => keys.first)
end
end
# The results should be provided in descending order. If the startkey or
# endkey query options have already been seen set, calling this method
# will automatically swap the options around. If you don't want this,
# simply set descending before any other option.
#
# Descending is false by default, and this method cannot
# be undone once used, it has no inverse option.
def descending
if query[:startkey] || query[:endkey]
query[:startkey], query[:endkey] = query[:endkey], query[:startkey]
elsif query[:startkey_docid] || query[:endkey_docid]
query[:startkey_docid], query[:endkey_docid] = query[:endkey_docid], query[:startkey_docid]
end
update_query(:descending => true)
end
# Limit the result set to the value supplied.
def limit(value)
update_query(:limit => value)
end
# Skip the number of entries in the index specified by value. This would be
# the equivilent of an offset in SQL.
#
# The CouchDB documentation states that the skip option should not be used
# with large data sets as it is inefficient. Use the +startkey_doc+ method
# instead to skip ranges efficiently.
def skip(value = 0)
update_query(:skip => value)
end
# Use the reduce function on the view. If none is available this method
# will fail.
def reduce
raise "Cannot reduce a view without a reduce method" unless can_reduce?
update_query(:reduce => true, :delete => [:include_docs])
end
# Control whether the reduce function reduces to a set of distinct keys
# or to a single result row.
#
# By default the value is false, and can only be set when the view's
# +#reduce+ option has been set.
def group
raise "View#reduce must have been set before grouping is permitted" unless query[:reduce]
update_query(:group => true)
end
# Will set the level the grouping should be performed to. As per the
# CouchDB API, it only makes sense when the index key is an array.
#
# This will automatically set the group option.
def group_level(value)
group.update_query(:group_level => value.to_i)
end
def include_docs
update_query.include_docs!
end
### Special View Filter Methods
# Allow the results of a query to be provided "stale". Setting to 'ok'
# will disable all view updates for the query.
# When 'update_after' is provided the index will be update after the
# result has been returned.
def stale(value)
unless (['ok', 'update_after'].include?(value.to_s))
raise "View#stale can only be set with 'ok' or 'update_after'."
end
update_query(:stale => value.to_s)
end
# Set the view's proxy that will be used instead of the model
# for any future searches. As soon as this enters the
# new view's initializer it will be removed and set as the model
# object.
#
# See the Proxyable mixin for more details.
#
def proxy(value)
update_query(:proxy => value)
end
# Return any cached values to their nil state so that any queries
# requested later will have a fresh set of data.
def reset!
self.result = nil
@rows = nil
@docs = nil
end
# == Kaminari compatible pagination support
#
# Based on the really simple support for scoped pagination in the
# the Kaminari gem, we provide compatible methods here to perform
# the same actions you'd expect.
#
def page(page)
limit(model.default_per_page).skip(model.default_per_page * ([page.to_i, 1].max - 1))
end
def per(num)
raise "View#page must be called before #per!" if limit_value.nil? || offset_value.nil?
if (n = num.to_i) <= 0
self
else
limit(num).skip(offset_value / limit_value * n)
end
end
def total_count
@total_count ||= limit(nil).skip(nil).count
end
def offset_value
query[:skip]
end
def limit_value
query[:limit]
end
def total_pages
(total_count.to_f / limit_value).ceil
end
alias num_pages total_pages
def current_page
(offset_value / limit_value) + 1
end
# == ActiveRecord compatibility support
# Return the model name for #model just as User::ActiveRecord_Relation
# would for better ActiveRecord interoperability
def model_name
ActiveModel::Name.new(model)
end
def database
owner.database
end
protected
def include_docs!
raise "Cannot include documents in view that has been reduced!" if query[:reduce]
reset! if result && !include_docs?
query[:include_docs] = true
self
end
def include_docs?
!!query[:include_docs]
end
def update_query(new_query = {})
self.class.new(design_doc, self, new_query)
end
def can_reduce?
!design_doc['views'][name]['reduce'].blank?
end
def execute(&block)
return self.result if result
raise CouchRest::Model::DatabaseNotDefined if database.nil?
# Remove the reduce value if its not needed to prevent CouchDB errors
query.delete(:reduce) unless can_reduce?
design_doc.sync(database)
self.result = design_doc.view_on(database, name, query, &block)
end
# Class Methods
class << self
def define_and_create(design_doc, name, opts = {})
define(design_doc, name, opts)
create_model_methods(design_doc, name, opts)
end
# Simplified view definition. A new view will be added to the
# provided design document using the name and options.
#
# If the view name starts with "by_" and +:by+ is not provided in
# the options, the new view's map method will be interpreted and
# generated automatically. For example:
#
# View.define(Meeting, design, "by_date_and_name")
#
# Will create a view that searches by the date and name properties.
# Explicity setting the attributes to use is possible using the
# +:by+ option. For example:
#
# View.define(Meeting, design, "by_date_and_name", :by => [:date, :firstname, :lastname])
#
# The view name is the same, but three keys would be used in the
# subsecuent index.
#
# By default, a check is made on each of the view's keys to ensure they
# do not contain a nil value ('null' in javascript). This is probably what
# you want in most cases but sometimes in can be useful to create an
# index where nil is permited. Set the :allow_nil option to true to
# remove this check.
#
# Conversely, keys are not checked to see if they are empty or blank. If you'd
# like to enable this, set the :allow_blank option to false. The default
# is true, empty strings are permited in the indexes.
#
def define(design_doc, name, opts = {})
model = design_doc.model
# Is this an all view?
if name.to_s == 'all'
opts[:map] = <<-EOF
function(doc) {
if (doc['#{model.model_type_key}'] == '#{model.model_type_value}') {
emit(doc._id, null);
}
}
EOF
elsif !opts[:map]
if opts[:by].nil? && name.to_s =~ /^by_(.+)/
opts[:by] = $1.split(/_and_/)
end
raise "View cannot be created without recognised name, :map or :by options" if opts[:by].nil?
# convert emit symbols to properties
emits, emit_guards = if opts[:emit].is_a? Symbol
[["doc['#{opts[:emit]}']"], ["doc['#{opts[:emit]}']"]]
elsif opts[:emit].is_a? Array
[
opts[:emit].map { |i| i.is_a?(Symbol) ? "doc['#{i}']" : i },
opts[:emit].map { |i| i.is_a?(Symbol) ? "doc['#{i}']" : nil}.compact
]
else
[[opts[:emit] || 1], nil]
end
opts[:allow_blank] = opts[:allow_blank].nil? ? true : opts[:allow_blank]
opts[:guards] ||= []
opts[:guards].push "(doc['#{model.model_type_key}'] == '#{model.model_type_value}')"
keys = opts[:by].map{|o| "doc['#{o}']"}
emit_keys = keys.length == 1 ? keys.first : "[#{keys.join(', ')}]"
emit_value = emits.length == 1 ? emits.first : "[#{emits.join(', ')}]"
opts[:guards] += keys.map{|k| "(#{k} != null)"} unless opts[:allow_nil]
opts[:guards] += keys.map{|k| "(#{k} != '')"} unless opts[:allow_blank]
opts[:guards] += emit_guards.map{|k| "(#{k} != null)"} unless emit_guards.nil? || opts[:allow_nil]
opts[:guards] += emit_guards.map{|k| "(#{k} != '')"} unless emit_guards.nil? || opts[:allow_blank]
opts[:map] = <<-EOF
function(doc) {
if (#{opts[:guards].join(' && ')}) {
emit(#{emit_keys}, #{emit_value});
}
}
EOF
if opts[:reduce].nil? && emit_guards.nil?
# Use built-in sum function by default
opts[:reduce] = "_sum"
end
end
if opts[:reduce].is_a?(Symbol)
# Assume calling a built in method, convert to a string
opts[:reduce] = "_#{opts[:reduce]}"
end
design_doc['views'] ||= {}
view = design_doc['views'][name.to_s] = { }
view['map'] = opts[:map]
view['reduce'] = opts[:reduce] if opts[:reduce]
view
end
def create_model_methods(design_doc, name, opts = {})
method = design_doc.method_name
design_doc.model.instance_eval <<-EOS, __FILE__, __LINE__ + 1
def #{name}(opts = {})
#{method}.view('#{name}', opts)
end
def find_#{name}(*key)
#{name}.key(*key).first()
end
def find_#{name}!(*key)
find_#{name}(*key) || raise(CouchRest::Model::DocumentNotFound)
end
EOS
end
end
end
# A special wrapper class that provides easy access to the key
# fields in a result row.
class ViewRow < Hash
# Owner will either be the original model, or a proxy and must respond
# to `#build_from_database` and `#get` calls.
# The owner is responsible for setting the correct database on instantiation,
# if required.
attr_reader :owner
def initialize(hash, owner)
@owner = owner
replace(hash)
end
def id
self["id"]
end
def key
self["key"]
end
def value
self['value']
end
def raw_doc
self['doc']
end
# Send a request for the linked document either using the "id" field's
# value, or the ["value"]["_id"] used for linked documents.
def doc
@doc ||= begin
if self['doc']
owner.build_from_database(self['doc'])
else
doc_id = (value.is_a?(Hash) && value['_id']) ? value['_id'] : self.id
doc_id ? owner.get(doc_id) : nil
end
end
end
end
end
end
end