# encoding: utf-8
module Mongoid #:nodoc:
module Contexts #:nodoc:
class Enumerable
include Ids, Paging
attr_reader :criteria
delegate :blank?, :empty?, :first, :last, :to => :execute
delegate :documents, :options, :selector, :to => :criteria
# Return aggregation counts of the grouped documents. This will count by
# the first field provided in the fields array.
#
# Returns:
#
# A +Hash+ with field values as keys, count as values
def aggregate
counts = {}
group.each_pair { |key, value| counts[key] = value.size }
counts
end
# Get the average value for the supplied field.
#
# Example:
#
# context.avg(:age)
#
# Returns:
#
# A numeric value that is the average.
def avg(field)
total = sum(field)
total ? (total.to_f / count) : nil
end
# Gets the number of documents in the array. Delegates to size.
def count
@count ||= documents.size
end
# Gets an array of distinct values for the supplied field across the
# entire array or the susbset given the criteria.
#
# Example:
#
# context.distinct(:title)
def distinct(field)
execute.collect { |doc| doc.send(field) }.uniq
end
# Enumerable implementation of execute. Returns matching documents for
# the selector, and adds options if supplied.
#
# Returns:
#
# An +Array+ of documents that matched the selector.
def execute(paginating = false)
limit(documents.select { |document| document.matches?(selector) })
end
# Groups the documents by the first field supplied in the field options.
#
# Returns:
#
# A +Hash+ with field values as keys, arrays of documents as values.
def group
field = options[:fields].first
documents.group_by { |doc| doc.send(field) }
end
# Create the new enumerable context. This will need the selector and
# options from a +Criteria+ and a documents array that is the underlying
# array of embedded documents from a has many association.
#
# Example:
#
# Mongoid::Contexts::Enumerable.new(criteria)
def initialize(criteria)
@criteria = criteria
end
# Iterate over each +Document+ in the results. This can take an optional
# block to pass to each argument in the results.
#
# Example:
#
# context.iterate { |doc| p doc }
def iterate(&block)
execute.each(&block)
end
# Get the largest value for the field in all the documents.
#
# Returns:
#
# The numerical largest value.
def max(field)
determine(field, :>=)
end
# Get the smallest value for the field in all the documents.
#
# Returns:
#
# The numerical smallest value.
def min(field)
determine(field, :<=)
end
# Get one document.
#
# Returns:
#
# The first document in the +Array+
alias :one :first
# Get the sum of the field values for all the documents.
#
# Returns:
#
# The numerical sum of all the document field values.
def sum(field)
sum = documents.inject(nil) do |memo, doc|
value = doc.send(field)
memo ? memo += value : value
end
end
protected
# If the field exists, perform the comparison and set if true.
def determine(field, operator)
matching = documents.inject(nil) do |memo, doc|
value = doc.send(field)
(memo && memo.send(operator, value)) ? memo : value
end
end
# Limits the result set if skip and limit options.
def limit(documents)
skip, limit = options[:skip], options[:limit]
if skip && limit
return documents.slice(skip, limit)
elsif limit
return documents.first(limit)
end
documents
end
end
end
end