RUBY-180 Fix batch size

This commit is contained in:
Kyle Banker 2010-09-09 14:12:12 -04:00
parent b339871eb9
commit 36c71ef48e
3 changed files with 93 additions and 14 deletions

View File

@ -50,19 +50,20 @@ module Mongo
@explain = options[:explain]
@socket = options[:socket]
@tailable = options[:tailable] || false
@batch_size = options[:batch_size] || 0
batch_size(options[:batch_size] || 0)
@full_collection_name = "#{@collection.db.name}.#{@collection.name}"
@cache = []
@closed = false
@query_run = false
@returned = 0
end
# Get the next document specified the cursor options.
#
# @return [Hash, Nil] the next document or Nil if no documents remain.
def next_document
refresh if num_remaining == 0
refresh if @cache.length == 0#empty?# num_remaining == 0
doc = @cache.shift
if doc && doc['$err']
@ -179,6 +180,26 @@ module Mongo
self
end
# Set the batch size for server responses.
#
# Note that the batch size will take effect only on queries
# where the number to be returned is greater than 100.
#
# @param [Integer] size either 0 or some integer greater than 1. If 0,
# the server will determine the batch size.
#
# @return [Cursor]
def batch_size(size=0)
check_modifiable
if size < 0 || size == 1
raise ArgumentError, "Invalid value for batch_size #{size}; must be 0 or > 1."
else
@batch_size = size > @limit ? @limit : size
end
self
end
# Iterate over each document in this cursor, yielding it to the given
# block.
#
@ -191,10 +212,11 @@ module Mongo
# puts doc['user']
# end
def each
num_returned = 0
while has_next? && (@limit <= 0 || num_returned < @limit)
yield next_document
num_returned += 1
#num_returned = 0
#while has_next? && (@limit <= 0 || num_returned < @limit)
while doc = next_document
yield doc #next_document
#num_returned += 1
end
end
@ -220,7 +242,7 @@ module Mongo
#
# @core explain explain-instance_method
def explain
c = Cursor.new(@collection, query_options_hash.merge(:limit => -@limit.abs, :explain => true))
c = Cursor.new(@collection, query_opti/ns_hash.merge(:limit => -@limit.abs, :explain => true))
explanation = c.next_document
c.close
@ -327,7 +349,7 @@ module Mongo
end
end
# Return a number of documents remaining for this cursor.
# Return the number of documents remaining for this cursor.
def num_remaining
refresh if @cache.length == 0
@cache.length
@ -341,13 +363,22 @@ module Mongo
BSON::BSON_RUBY.serialize_cstr(message, "#{@db.name}.#{@collection.name}")
# Number of results to return.
if @limit
if (@returned + @batch_size) > @limit
message.put_int(@limit - @returned)
else
message.put_int(@batch_size)
end
else
message.put_int(@batch_size)
end
# Cursor id.
message.put_long(@cursor_id)
@logger.debug("MONGODB cursor.refresh() for cursor #{@cursor_id}") if @logger
results, @n_received, @cursor_id = @connection.receive_message(Mongo::Constants::OP_GET_MORE,
message, nil, @socket)
@returned += @n_received
@cache += results
close_cursor_if_query_complete
end
@ -360,6 +391,7 @@ module Mongo
message = construct_query_message
@logger.debug query_log_message if @logger
results, @n_received, @cursor_id = @connection.receive_message(Mongo::Constants::OP_QUERY, message, nil, @socket)
@returned += @n_received
@cache += results
@query_run = true
close_cursor_if_query_complete
@ -406,7 +438,9 @@ module Mongo
end
def close_cursor_if_query_complete
close if @limit > 0 && @n_received >= @limit
if @limit > 0 && @n_received >= @limit
close
end
end
def check_modifiable

View File

@ -0,0 +1,44 @@
require 'test/test_helper'
require 'logger'
class CursorTest < Test::Unit::TestCase
include Mongo
@@connection = Connection.new(ENV['MONGO_RUBY_DRIVER_HOST'] || 'localhost',
ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT)
@@db = @@connection.db(MONGO_TEST_DB)
@@coll = @@db.collection('test')
@@version = @@connection.server_version
def setup
@@coll.remove
@@coll.insert('a' => 1) # collection not created until it's used
@@coll_full_name = "#{MONGO_TEST_DB}.test"
end
def test_valid_batch_sizes
assert_raise ArgumentError do
@@coll.find({}, :batch_size => 1, :limit => 5)
end
assert_raise ArgumentError do
@@coll.find({}, :batch_size => -1, :limit => 5)
end
assert @@coll.find({}, :batch_size => 0, :limit => 5)
end
def test_batch_size
@@coll.remove
200.times do |n|
@@coll.insert({:a => n})
end
list = @@coll.find({}, :batch_size => 2, :limit => 6).to_a
assert_equal 6, list.length
list = @@coll.find({}, :batch_size => 100, :limit => 101).to_a
assert_equal 101, list.length
end
end

View File

@ -428,4 +428,5 @@ class CursorTest < Test::Unit::TestCase
cursor.rewind!
assert_equal 100, cursor.map {|doc| doc }.length
end
end