jgaskins/perpetuity

View on GitHub
lib/perpetuity/retrieval.rb

Summary

Maintainability
A
0 mins
Test Coverage
require 'perpetuity/reference'
require 'perpetuity/identity_map'

module Perpetuity
  class Retrieval
    include Enumerable
    attr_accessor :sort_attribute, :sort_direction, :result_limit, :result_page, :result_offset, :result_cache
    attr_reader :identity_map

    def initialize mapper, query, options={}
      @mapper = mapper
      @collection_name = mapper.collection_name
      @query = query
      @identity_map = options.fetch(:identity_map) { IdentityMap.new }
      @data_source = mapper.data_source
    end
    
    def sort attribute=:name
      retrieval = clone
      retrieval.sort_attribute = attribute
      retrieval.sort_direction = :ascending
      retrieval.clear_cache

      retrieval
    end

    def reverse
      retrieval = clone
      retrieval.sort_direction = retrieval.sort_direction == :descending ? :ascending : :descending
      retrieval.clear_cache

      retrieval
    end

    def page page
      retrieval = clone
      retrieval.result_limit ||= 20
      retrieval.result_page = page
      retrieval.result_offset = (page - 1) * retrieval.result_limit
      retrieval.clear_cache
      retrieval
    end

    def per_page per
      retrieval = clone
      retrieval.result_limit = per
      retrieval.result_offset = (retrieval.result_page - 1) * per
      retrieval.clear_cache
      retrieval
    end

    def each &block
      to_a.each(&block)
    end

    def to_a
      @result_cache ||= @data_source.unserialize(@data_source.retrieve(@collection_name, @query, options), @mapper)

      @result_cache.map do |result|
        klass = result.class
        id = @mapper.id_for(result)

        if cached_result = identity_map[klass, id]
          cached_result
        else
          identity_map << result
          result
        end
      end
    end

    def count
      @data_source.count(@collection_name, @query)
    end

    def first
      limit(1).to_a.first
    end

    def sample
      sample_size = [count, result_limit].compact.max
      result = drop(rand(sample_size)).first
      result
    end

    def options
      {
        attribute: sort_attribute,
        direction: sort_direction,
        limit: result_limit,
        skip: result_offset
      }
    end

    def [] index
      to_a[index]
    end

    def empty?
      to_a.empty?
    end
    
    def limit lim
      retrieval = clone
      retrieval.result_limit = lim
      retrieval.clear_cache
      
      retrieval
    end
    alias_method :take, :limit

    def drop count
      retrieval = clone
      retrieval.result_offset = count
      retrieval.clear_cache

      retrieval
    end

    def clear_cache
      @result_cache = nil
    end
  end
end