RUBY-242 check BSON size on a per-connection basis.
This commit is contained in:
parent
9b42265d10
commit
3c127984a3
@ -20,11 +20,13 @@
|
|||||||
#include "bson_buffer.h"
|
#include "bson_buffer.h"
|
||||||
|
|
||||||
#define INITIAL_BUFFER_SIZE 256
|
#define INITIAL_BUFFER_SIZE 256
|
||||||
|
#define DEFAULT_MAX_SIZE 4 * 1024 * 1024
|
||||||
|
|
||||||
struct bson_buffer {
|
struct bson_buffer {
|
||||||
char* buffer;
|
char* buffer;
|
||||||
int size;
|
int size;
|
||||||
int position;
|
int position;
|
||||||
|
int max_size;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Allocate and return a new buffer.
|
/* Allocate and return a new buffer.
|
||||||
@ -43,10 +45,19 @@ bson_buffer_t bson_buffer_new(void) {
|
|||||||
free(buffer);
|
free(buffer);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
buffer->max_size = DEFAULT_MAX_SIZE;
|
||||||
|
|
||||||
return buffer;
|
return buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void bson_buffer_set_max_size(bson_buffer_t buffer, int max_size) {
|
||||||
|
buffer->max_size = max_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
int bson_buffer_get_max_size(bson_buffer_t buffer) {
|
||||||
|
return buffer->max_size;
|
||||||
|
}
|
||||||
|
|
||||||
/* Free the memory allocated for `buffer`.
|
/* Free the memory allocated for `buffer`.
|
||||||
* Return non-zero on failure. */
|
* Return non-zero on failure. */
|
||||||
int bson_buffer_free(bson_buffer_t buffer) {
|
int bson_buffer_free(bson_buffer_t buffer) {
|
||||||
|
@ -29,6 +29,10 @@ typedef int bson_buffer_position;
|
|||||||
* Return NULL on allocation failure. */
|
* Return NULL on allocation failure. */
|
||||||
bson_buffer_t bson_buffer_new(void);
|
bson_buffer_t bson_buffer_new(void);
|
||||||
|
|
||||||
|
/* Set the max size for this buffer.
|
||||||
|
* Note: this is not a hard limit. */
|
||||||
|
void bson_buffer_set_max_size(bson_buffer_t buffer, int max_size);
|
||||||
|
|
||||||
/* Free the memory allocated for `buffer`.
|
/* Free the memory allocated for `buffer`.
|
||||||
* Return non-zero on failure. */
|
* Return non-zero on failure. */
|
||||||
int bson_buffer_free(bson_buffer_t buffer);
|
int bson_buffer_free(bson_buffer_t buffer);
|
||||||
|
@ -599,17 +599,22 @@ static void write_doc(bson_buffer_t buffer, VALUE hash, VALUE check_keys, VALUE
|
|||||||
length = bson_buffer_get_position(buffer) - start_position;
|
length = bson_buffer_get_position(buffer) - start_position;
|
||||||
|
|
||||||
// make sure that length doesn't exceed 4MB
|
// make sure that length doesn't exceed 4MB
|
||||||
if (length > max_bson_size) {
|
if (length > bson_buffer_get_max_size(buffer)) {
|
||||||
bson_buffer_free(buffer);
|
bson_buffer_free(buffer);
|
||||||
rb_raise(InvalidDocument, "Document too large: BSON documents are limited to %d bytes.", max_bson_size);
|
rb_raise(InvalidDocument,
|
||||||
|
"Document too large: This BSON documents is limited to %d bytes.",
|
||||||
|
bson_buffer_get_max_size(buffer));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
SAFE_WRITE_AT_POS(buffer, length_location, (const char*)&length, 4);
|
SAFE_WRITE_AT_POS(buffer, length_location, (const char*)&length, 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
static VALUE method_serialize(VALUE self, VALUE doc, VALUE check_keys, VALUE move_id) {
|
static VALUE method_serialize(VALUE self, VALUE doc, VALUE check_keys,
|
||||||
|
VALUE move_id, VALUE max_size) {
|
||||||
|
|
||||||
VALUE result;
|
VALUE result;
|
||||||
bson_buffer_t buffer = bson_buffer_new();
|
bson_buffer_t buffer = bson_buffer_new();
|
||||||
|
bson_buffer_set_max_size(buffer, FIX2INT(max_size));
|
||||||
if (buffer == NULL) {
|
if (buffer == NULL) {
|
||||||
rb_raise(rb_eNoMemError, "failed to allocate memory in buffer.c");
|
rb_raise(rb_eNoMemError, "failed to allocate memory in buffer.c");
|
||||||
}
|
}
|
||||||
@ -973,7 +978,7 @@ void Init_cbson() {
|
|||||||
CBson = rb_define_module("CBson");
|
CBson = rb_define_module("CBson");
|
||||||
ext_version = rb_str_new2(VERSION);
|
ext_version = rb_str_new2(VERSION);
|
||||||
rb_define_const(CBson, "VERSION", ext_version);
|
rb_define_const(CBson, "VERSION", ext_version);
|
||||||
rb_define_module_function(CBson, "serialize", method_serialize, 3);
|
rb_define_module_function(CBson, "serialize", method_serialize, 4);
|
||||||
rb_define_module_function(CBson, "deserialize", method_deserialize, 1);
|
rb_define_module_function(CBson, "deserialize", method_deserialize, 1);
|
||||||
rb_define_module_function(CBson, "max_bson_size", method_max_bson_size, 0);
|
rb_define_module_function(CBson, "max_bson_size", method_max_bson_size, 0);
|
||||||
rb_define_module_function(CBson, "update_max_bson_size", method_update_max_bson_size, 1);
|
rb_define_module_function(CBson, "update_max_bson_size", method_update_max_bson_size, 1);
|
||||||
|
Binary file not shown.
@ -69,7 +69,8 @@ public class RubyBSONEncoder extends BSONEncoder {
|
|||||||
private static final BigInteger LONG_MIN = BigInteger.valueOf(-MAX - 1);
|
private static final BigInteger LONG_MIN = BigInteger.valueOf(-MAX - 1);
|
||||||
|
|
||||||
|
|
||||||
public RubyBSONEncoder(Ruby runtime, boolean check_keys, boolean move_id){
|
public RubyBSONEncoder(Ruby runtime, boolean check_keys, boolean move_id, int max_bson_size){
|
||||||
|
_max_bson_size = max_bson_size;
|
||||||
_check_keys = check_keys;
|
_check_keys = check_keys;
|
||||||
_move_id = move_id;
|
_move_id = move_id;
|
||||||
_runtime = runtime;
|
_runtime = runtime;
|
||||||
|
@ -20,8 +20,8 @@
|
|||||||
module BSON
|
module BSON
|
||||||
class BSON_C
|
class BSON_C
|
||||||
|
|
||||||
def self.serialize(obj, check_keys=false, move_id=false)
|
def self.serialize(obj, check_keys=false, move_id=false, max_bson_size=BSON::DEFAULT_MAX_BSON_SIZE)
|
||||||
ByteBuffer.new(CBson.serialize(obj, check_keys, move_id))
|
ByteBuffer.new(CBson.serialize(obj, check_keys, move_id, max_bson_size))
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.deserialize(buf=nil)
|
def self.deserialize(buf=nil)
|
||||||
|
@ -4,9 +4,9 @@ module BSON
|
|||||||
|
|
||||||
# TODO: Pool or cache instances of RubyBSONEncoder so that
|
# TODO: Pool or cache instances of RubyBSONEncoder so that
|
||||||
# we don't create a new one on each call to #serialize.
|
# we don't create a new one on each call to #serialize.
|
||||||
def self.serialize(obj, check_keys=false, move_id=false)
|
def self.serialize(obj, check_keys=false, move_id=false, max_bson_size=BSON::DEFAULT_MAX_BSON_SIZE)
|
||||||
raise InvalidDocument, "BSON_JAVA.serialize takes a Hash" unless obj.is_a?(Hash)
|
raise InvalidDocument, "BSON_JAVA.serialize takes a Hash" unless obj.is_a?(Hash)
|
||||||
enc = Java::OrgJbson::RubyBSONEncoder.new(JRuby.runtime, check_keys, move_id)
|
enc = Java::OrgJbson::RubyBSONEncoder.new(JRuby.runtime, check_keys, move_id, max_bson_size)
|
||||||
ByteBuffer.new(enc.encode(obj))
|
ByteBuffer.new(enc.encode(obj))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -46,8 +46,8 @@ module BSON
|
|||||||
NUMBER_LONG = 18
|
NUMBER_LONG = 18
|
||||||
MAXKEY = 127
|
MAXKEY = 127
|
||||||
|
|
||||||
def initialize
|
def initialize(max_bson_size=BSON::DEFAULT_MAX_BSON_SIZE)
|
||||||
@buf = ByteBuffer.new
|
@buf = ByteBuffer.new('', max_bson_size)
|
||||||
@encoder = BSON_RUBY
|
@encoder = BSON_RUBY
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -105,8 +105,8 @@ module BSON
|
|||||||
|
|
||||||
# Serializes an object.
|
# Serializes an object.
|
||||||
# Implemented to ensure an API compatible with BSON extension.
|
# Implemented to ensure an API compatible with BSON extension.
|
||||||
def self.serialize(obj, check_keys=false, move_id=false)
|
def self.serialize(obj, check_keys=false, move_id=false, max_bson_size=BSON::DEFAULT_MAX_BSON_SIZE)
|
||||||
new.serialize(obj, check_keys, move_id)
|
new(max_bson_size).serialize(obj, check_keys, move_id)
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.deserialize(buf=nil)
|
def self.deserialize(buf=nil)
|
||||||
@ -137,8 +137,9 @@ module BSON
|
|||||||
end
|
end
|
||||||
|
|
||||||
serialize_eoo_element(@buf)
|
serialize_eoo_element(@buf)
|
||||||
if @buf.size > @@max_bson_size
|
if @buf.size > @buf.max_size
|
||||||
raise InvalidDocument, "Document is too large (#{@buf.size}). BSON documents are limited to #{@@max_bson_size} bytes."
|
raise InvalidDocument, "Document is too large (#{@buf.size}). " +
|
||||||
|
"This BSON documents is limited to #{@buf.max_size} bytes."
|
||||||
end
|
end
|
||||||
@buf.put_int(@buf.size, 0)
|
@buf.put_int(@buf.size, 0)
|
||||||
@buf
|
@buf
|
||||||
|
@ -20,9 +20,9 @@
|
|||||||
module BSON
|
module BSON
|
||||||
class ByteBuffer
|
class ByteBuffer
|
||||||
|
|
||||||
attr_reader :order
|
attr_reader :order, :max_size
|
||||||
|
|
||||||
def initialize(initial_data="")
|
def initialize(initial_data="", max_size=BSON::DEFAULT_MAX_BSON_SIZE)
|
||||||
@str = case initial_data
|
@str = case initial_data
|
||||||
when String then
|
when String then
|
||||||
if initial_data.respond_to?(:force_encoding)
|
if initial_data.respond_to?(:force_encoding)
|
||||||
@ -40,6 +40,7 @@ module BSON
|
|||||||
@order = :little_endian
|
@order = :little_endian
|
||||||
@int_pack_order = 'V'
|
@int_pack_order = 'V'
|
||||||
@double_pack_order = 'E'
|
@double_pack_order = 'E'
|
||||||
|
@max_size = max_size
|
||||||
end
|
end
|
||||||
|
|
||||||
if RUBY_VERSION >= '1.9'
|
if RUBY_VERSION >= '1.9'
|
||||||
|
@ -357,7 +357,7 @@ module Mongo
|
|||||||
message = BSON::ByteBuffer.new("\0\0\0\0")
|
message = BSON::ByteBuffer.new("\0\0\0\0")
|
||||||
BSON::BSON_RUBY.serialize_cstr(message, "#{@db.name}.#{@name}")
|
BSON::BSON_RUBY.serialize_cstr(message, "#{@db.name}.#{@name}")
|
||||||
message.put_int(0)
|
message.put_int(0)
|
||||||
message.put_binary(BSON::BSON_CODER.serialize(selector, false, true).to_s)
|
message.put_binary(BSON::BSON_CODER.serialize(selector, false, true, @connection.max_bson_size).to_s)
|
||||||
|
|
||||||
@connection.instrument(:remove, :database => @db.name, :collection => @name, :selector => selector) do
|
@connection.instrument(:remove, :database => @db.name, :collection => @name, :selector => selector) do
|
||||||
if safe
|
if safe
|
||||||
@ -404,7 +404,7 @@ module Mongo
|
|||||||
update_options += 2 if opts[:multi]
|
update_options += 2 if opts[:multi]
|
||||||
message.put_int(update_options)
|
message.put_int(update_options)
|
||||||
message.put_binary(BSON::BSON_CODER.serialize(selector, false, true).to_s)
|
message.put_binary(BSON::BSON_CODER.serialize(selector, false, true).to_s)
|
||||||
message.put_binary(BSON::BSON_CODER.serialize(document, false, true).to_s)
|
message.put_binary(BSON::BSON_CODER.serialize(document, false, true, @connection.max_bson_size).to_s)
|
||||||
|
|
||||||
@connection.instrument(:update, :database => @db.name, :collection => @name, :selector => selector, :document => document) do
|
@connection.instrument(:update, :database => @db.name, :collection => @name, :selector => selector, :document => document) do
|
||||||
if safe
|
if safe
|
||||||
@ -897,7 +897,7 @@ module Mongo
|
|||||||
message = BSON::ByteBuffer.new("\0\0\0\0")
|
message = BSON::ByteBuffer.new("\0\0\0\0")
|
||||||
BSON::BSON_RUBY.serialize_cstr(message, "#{@db.name}.#{collection_name}")
|
BSON::BSON_RUBY.serialize_cstr(message, "#{@db.name}.#{collection_name}")
|
||||||
documents.each do |doc|
|
documents.each do |doc|
|
||||||
message.put_binary(BSON::BSON_CODER.serialize(doc, check_keys, true).to_s)
|
message.put_binary(BSON::BSON_CODER.serialize(doc, check_keys, true, @connection.max_bson_size).to_s)
|
||||||
end
|
end
|
||||||
raise InvalidOperation, "Exceded maximum insert size of 16,000,000 bytes" if message.size > 16_000_000
|
raise InvalidOperation, "Exceded maximum insert size of 16,000,000 bytes" if message.size > 16_000_000
|
||||||
|
|
||||||
|
@ -512,12 +512,11 @@ module Mongo
|
|||||||
@read_primary = false
|
@read_primary = false
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@max_bson_size = config['maxBsonObjectSize'] || Mongo::DEFAULT_MAX_BSON_SIZE
|
||||||
set_primary(@host_to_try)
|
set_primary(@host_to_try)
|
||||||
end
|
end
|
||||||
|
|
||||||
if connected?
|
if !connected?
|
||||||
BSON::BSON_CODER.update_max_bson_size(self)
|
|
||||||
else
|
|
||||||
raise ConnectionFailure, "Failed to connect to a master node at #{@host_to_try[0]}:#{@host_to_try[1]}"
|
raise ConnectionFailure, "Failed to connect to a master node at #{@host_to_try[0]}:#{@host_to_try[1]}"
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -571,8 +570,7 @@ module Mongo
|
|||||||
#
|
#
|
||||||
# @return [Integer]
|
# @return [Integer]
|
||||||
def max_bson_size
|
def max_bson_size
|
||||||
config = self['admin'].command({:ismaster => 1})
|
@max_bson_size
|
||||||
config['maxBsonObjectSize'] || Mongo::DEFAULT_MAX_BSON_SIZE
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# Checkout a socket for reading (i.e., a secondary node).
|
# Checkout a socket for reading (i.e., a secondary node).
|
||||||
@ -619,7 +617,7 @@ module Mongo
|
|||||||
@logger.fatal "MONGODB [FATAL] #{msg}"
|
@logger.fatal "MONGODB [FATAL] #{msg}"
|
||||||
else
|
else
|
||||||
@logger.info "MONGODB [INFO] #{msg}"
|
@logger.info "MONGODB [INFO] #{msg}"
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Execute the block and log the operation described by name
|
# Execute the block and log the operation described by name
|
||||||
@ -635,6 +633,9 @@ module Mongo
|
|||||||
|
|
||||||
# Generic initialization code.
|
# Generic initialization code.
|
||||||
def setup(opts)
|
def setup(opts)
|
||||||
|
# Default maximum BSON object size
|
||||||
|
@max_bson_size = Mongo::DEFAULT_MAX_BSON_SIZE
|
||||||
|
|
||||||
# Authentication objects
|
# Authentication objects
|
||||||
@auths = opts.fetch(:auths, [])
|
@auths = opts.fetch(:auths, [])
|
||||||
|
|
||||||
|
@ -8,7 +8,8 @@ module Mongo
|
|||||||
if data.is_a?(String)
|
if data.is_a?(String)
|
||||||
self.host, self.port = split_nodes(data)
|
self.host, self.port = split_nodes(data)
|
||||||
else
|
else
|
||||||
self.host, self.port = data
|
self.host = data[0]
|
||||||
|
self.port = data[1].nil? ? Connection::DEFAULT_PORT : data[1].to_i
|
||||||
end
|
end
|
||||||
self.address = "#{host}:#{port}"
|
self.address = "#{host}:#{port}"
|
||||||
end
|
end
|
||||||
@ -134,7 +135,7 @@ module Mongo
|
|||||||
def split_nodes(host_string)
|
def split_nodes(host_string)
|
||||||
data = host_string.split(":")
|
data = host_string.split(":")
|
||||||
host = data[0]
|
host = data[0]
|
||||||
port = data[1].to_i || Connection::DEFAULT_PORT
|
port = data[1].nil? ? Connection::DEFAULT_PORT : data[1].to_i
|
||||||
|
|
||||||
[host, port]
|
[host, port]
|
||||||
end
|
end
|
||||||
|
@ -87,8 +87,10 @@ module Mongo
|
|||||||
# The list of seed nodes
|
# The list of seed nodes
|
||||||
@seeds = args
|
@seeds = args
|
||||||
|
|
||||||
|
@nodes = @seeds.dup
|
||||||
|
|
||||||
# The members of the replica set, stored as instances of Mongo::Node.
|
# The members of the replica set, stored as instances of Mongo::Node.
|
||||||
@nodes = []
|
@members = []
|
||||||
|
|
||||||
# Connection pool for primary node
|
# Connection pool for primary node
|
||||||
@primary = nil
|
@primary = nil
|
||||||
@ -139,7 +141,6 @@ module Mongo
|
|||||||
manager.connect
|
manager.connect
|
||||||
|
|
||||||
update_config(manager)
|
update_config(manager)
|
||||||
#BSON::BSON_CODER.update_max_bson_size(self)
|
|
||||||
initiate_auto_refresh
|
initiate_auto_refresh
|
||||||
|
|
||||||
if @primary.nil? #TODO: in v2.0, we'll let this be optional and do a lazy connect.
|
if @primary.nil? #TODO: in v2.0, we'll let this be optional and do a lazy connect.
|
||||||
@ -164,6 +165,7 @@ module Mongo
|
|||||||
@manager = manager
|
@manager = manager
|
||||||
@hosts = manager.hosts
|
@hosts = manager.hosts
|
||||||
@nodes = manager.nodes
|
@nodes = manager.nodes
|
||||||
|
@max_bson_size = manager.max_bson_size
|
||||||
end
|
end
|
||||||
|
|
||||||
# If ismaster doesn't match our current view
|
# If ismaster doesn't match our current view
|
||||||
@ -210,6 +212,11 @@ module Mongo
|
|||||||
super
|
super
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def nodes
|
||||||
|
warn "DEPRECATED"
|
||||||
|
@seeds
|
||||||
|
end
|
||||||
|
|
||||||
# Determine whether we're reading from a primary node. If false,
|
# Determine whether we're reading from a primary node. If false,
|
||||||
# this connection connects to a secondary node and @read_secondaries is true.
|
# this connection connects to a secondary node and @read_secondaries is true.
|
||||||
#
|
#
|
||||||
|
@ -2,7 +2,7 @@ module Mongo
|
|||||||
class PoolManager
|
class PoolManager
|
||||||
|
|
||||||
attr_reader :connection, :seeds, :arbiters, :primary, :secondaries,
|
attr_reader :connection, :seeds, :arbiters, :primary, :secondaries,
|
||||||
:primary_pool, :read_pool, :secondary_pools, :hosts, :nodes
|
:primary_pool, :read_pool, :secondary_pools, :hosts, :nodes, :max_bson_size
|
||||||
|
|
||||||
def initialize(connection, seeds)
|
def initialize(connection, seeds)
|
||||||
@connection = connection
|
@connection = connection
|
||||||
@ -12,10 +12,10 @@ module Mongo
|
|||||||
|
|
||||||
def connect
|
def connect
|
||||||
initialize_data
|
initialize_data
|
||||||
nodes = connect_to_members
|
members = connect_to_members
|
||||||
initialize_pools(nodes)
|
initialize_pools(members)
|
||||||
update_seed_list(nodes)
|
update_seed_list(members)
|
||||||
@nodes = nodes
|
@members = members
|
||||||
end
|
end
|
||||||
|
|
||||||
# Ensure that the view of the replica set is current by
|
# Ensure that the view of the replica set is current by
|
||||||
@ -106,34 +106,34 @@ module Mongo
|
|||||||
@secondaries = []
|
@secondaries = []
|
||||||
@secondary_pools = []
|
@secondary_pools = []
|
||||||
@hosts = []
|
@hosts = []
|
||||||
@nodes = []
|
@members = []
|
||||||
end
|
end
|
||||||
|
|
||||||
# Connect to each member of the replica set
|
# Connect to each member of the replica set
|
||||||
# as reported by the given seed node, and return
|
# as reported by the given seed node, and return
|
||||||
# as a list of Mongo::Node objects.
|
# as a list of Mongo::Node objects.
|
||||||
def connect_to_members
|
def connect_to_members
|
||||||
nodes = []
|
members = []
|
||||||
|
|
||||||
seed = get_valid_seed_node
|
seed = get_valid_seed_node
|
||||||
|
|
||||||
seed.node_list.each do |host|
|
seed.node_list.each do |host|
|
||||||
node = Mongo::Node.new(self.connection, host)
|
node = Mongo::Node.new(self.connection, host)
|
||||||
if node.connect && node.set_config
|
if node.connect && node.set_config
|
||||||
nodes << node
|
members << node
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
if nodes.empty?
|
if members.empty?
|
||||||
raise ConnectionFailure, "Failed to connect to any given member."
|
raise ConnectionFailure, "Failed to connect to any given member."
|
||||||
end
|
end
|
||||||
|
|
||||||
nodes
|
members
|
||||||
end
|
end
|
||||||
|
|
||||||
# Initialize the connection pools for the primary and secondary nodes.
|
# Initialize the connection pools for the primary and secondary nodes.
|
||||||
def initialize_pools(nodes)
|
def initialize_pools(members)
|
||||||
nodes.each do |member|
|
members.each do |member|
|
||||||
@hosts << member.host_string
|
@hosts << member.host_string
|
||||||
|
|
||||||
if member.primary?
|
if member.primary?
|
||||||
@ -151,7 +151,9 @@ module Mongo
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@arbiters = nodes.first.arbiters
|
@max_bson_size = members.first.config['maxBsonObjectSize'] ||
|
||||||
|
Mongo::DEFAULT_MAX_BSON_SIZE
|
||||||
|
@arbiters = members.first.arbiters
|
||||||
choose_read_pool
|
choose_read_pool
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -203,8 +205,8 @@ module Mongo
|
|||||||
"#{@seeds.map {|s| "#{s[0]}:#{s[1]}" }.join(', ')}"
|
"#{@seeds.map {|s| "#{s[0]}:#{s[1]}" }.join(', ')}"
|
||||||
end
|
end
|
||||||
|
|
||||||
def update_seed_list(nodes)
|
def update_seed_list(members)
|
||||||
@seeds = nodes.map { |n| n.host_port }
|
@seeds = members.map { |n| n.host_port }
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
require './test/test_helper'
|
require './test/test_helper'
|
||||||
|
|
||||||
class TestCollection < Test::Unit::TestCase
|
class TestCollection < Test::Unit::TestCase
|
||||||
@@connection ||= standard_connection(:op_timeout => 2)
|
@@connection ||= standard_connection(:op_timeout => 10)
|
||||||
@@db = @@connection.db(MONGO_TEST_DB)
|
@@db = @@connection.db(MONGO_TEST_DB)
|
||||||
@@test = @@db.collection("test")
|
@@test = @@db.collection("test")
|
||||||
@@version = @@connection.server_version
|
@@version = @@connection.server_version
|
||||||
|
@ -13,7 +13,7 @@ class TestConnection < Test::Unit::TestCase
|
|||||||
end
|
end
|
||||||
|
|
||||||
def teardown
|
def teardown
|
||||||
@conn[MONGO_TEST_DB].get_last_error
|
@conn.close
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_connection_failure
|
def test_connection_failure
|
||||||
@ -162,8 +162,8 @@ class TestConnection < Test::Unit::TestCase
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_nodes
|
def test_nodes
|
||||||
db = Connection.multi([['foo', 27017], ['bar', 27018]], :connect => false)
|
conn = Connection.multi([['foo', 27017], ['bar', 27018]], :connect => false)
|
||||||
nodes = db.nodes
|
nodes = conn.nodes
|
||||||
assert_equal 2, nodes.length
|
assert_equal 2, nodes.length
|
||||||
assert_equal ['foo', 27017], nodes[0]
|
assert_equal ['foo', 27017], nodes[0]
|
||||||
assert_equal ['bar', 27018], nodes[1]
|
assert_equal ['bar', 27018], nodes[1]
|
||||||
@ -191,35 +191,32 @@ class TestConnection < Test::Unit::TestCase
|
|||||||
end
|
end
|
||||||
|
|
||||||
def test_max_bson_size_value
|
def test_max_bson_size_value
|
||||||
|
conn = standard_connection(:connect => false)
|
||||||
|
|
||||||
|
admin_db = Object.new
|
||||||
|
admin_db.expects(:command).returns({'ok' => 1, 'ismaster' => 1, 'maxBsonObjectSize' => 15_000_000})
|
||||||
|
conn.expects(:[]).with('admin').returns(admin_db)
|
||||||
|
conn.connect
|
||||||
|
assert_equal 15_000_000, conn.max_bson_size
|
||||||
|
|
||||||
conn = standard_connection
|
conn = standard_connection
|
||||||
if conn.server_version > "1.7.2"
|
if conn.server_version > "1.7.2"
|
||||||
assert_equal conn['admin'].command({:ismaster => 1})['maxBsonObjectSize'], conn.max_bson_size
|
assert_equal conn['admin'].command({:ismaster => 1})['maxBsonObjectSize'], conn.max_bson_size
|
||||||
end
|
end
|
||||||
|
|
||||||
conn.connect
|
conn.connect
|
||||||
assert_equal BSON::BSON_CODER.max_bson_size, conn.max_bson_size
|
doc = {'n' => 'a' * (conn.max_bson_size)}
|
||||||
doc = {'n' => 'a' * (BSON_CODER.max_bson_size - 11)}
|
|
||||||
assert_raise InvalidDocument do
|
assert_raise InvalidDocument do
|
||||||
assert BSON::BSON_CODER.serialize(doc)
|
assert BSON::BSON_CODER.serialize(doc, false, true, @conn.max_bson_size)
|
||||||
end
|
|
||||||
|
|
||||||
limit = 7 * 1024 * 1024
|
|
||||||
conn.stubs(:max_bson_size).returns(limit)
|
|
||||||
conn.connect
|
|
||||||
assert_equal limit, conn.max_bson_size
|
|
||||||
assert_equal limit, BSON::BSON_CODER.max_bson_size
|
|
||||||
doc = {'n' => 'a' * ((limit) - 11)}
|
|
||||||
assert_raise_error InvalidDocument, "limited to #{limit}" do
|
|
||||||
assert BSON::BSON_CODER.serialize(doc)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_max_bson_size_with_old_mongod
|
def test_max_bson_size_with_no_reported_max_size
|
||||||
conn = standard_connection(:connect => false)
|
conn = standard_connection(:connect => false)
|
||||||
|
|
||||||
admin_db = Object.new
|
admin_db = Object.new
|
||||||
admin_db.expects(:command).returns({'ok' => 1, 'ismaster' => 1}).twice
|
admin_db.expects(:command).returns({'ok' => 1, 'ismaster' => 1})
|
||||||
conn.expects(:[]).with('admin').returns(admin_db).twice
|
conn.expects(:[]).with('admin').returns(admin_db)
|
||||||
|
|
||||||
conn.connect
|
conn.connect
|
||||||
assert_equal Mongo::DEFAULT_MAX_BSON_SIZE, BSON::BSON_CODER.max_bson_size
|
assert_equal Mongo::DEFAULT_MAX_BSON_SIZE, BSON::BSON_CODER.max_bson_size
|
||||||
@ -253,6 +250,10 @@ class TestConnection < Test::Unit::TestCase
|
|||||||
@conn.add_auth(@auth['db_name'], @auth['username'], @auth['password'])
|
@conn.add_auth(@auth['db_name'], @auth['username'], @auth['password'])
|
||||||
end
|
end
|
||||||
|
|
||||||
|
teardown do
|
||||||
|
@conn.clear_auths
|
||||||
|
end
|
||||||
|
|
||||||
should "save the authentication" do
|
should "save the authentication" do
|
||||||
assert_equal @auth, @conn.auths[0]
|
assert_equal @auth, @conn.auths[0]
|
||||||
end
|
end
|
||||||
@ -315,7 +316,7 @@ class TestConnection < Test::Unit::TestCase
|
|||||||
fake_socket = Mocha::Mock.new
|
fake_socket = Mocha::Mock.new
|
||||||
fake_socket.stubs(:close).raises(IOError.new)
|
fake_socket.stubs(:close).raises(IOError.new)
|
||||||
fake_socket.stub_everything
|
fake_socket.stub_everything
|
||||||
TCPSocket.expects(:new).returns(fake_socket)
|
TCPSocket.stubs(:new).returns(fake_socket)
|
||||||
|
|
||||||
@con.primary_pool.checkout_new_socket
|
@con.primary_pool.checkout_new_socket
|
||||||
assert_equal [], @con.primary_pool.close
|
assert_equal [], @con.primary_pool.close
|
||||||
|
@ -156,7 +156,8 @@ class DBTest < Test::Unit::TestCase
|
|||||||
assert Mongo::Connection.from_uri("mongodb://spongebob:squarepants@#{host_port}/#{@@db.name}")
|
assert Mongo::Connection.from_uri("mongodb://spongebob:squarepants@#{host_port}/#{@@db.name}")
|
||||||
|
|
||||||
assert_raise Mongo::AuthenticationError do
|
assert_raise Mongo::AuthenticationError do
|
||||||
Mongo::Connection.from_uri("mongodb://wrong:info@#{host_port}/#{@@db.name}")
|
con = Mongo::Connection.from_uri("mongodb://wrong:info@#{host_port}/#{@@db.name}")
|
||||||
|
con['test']['foo'].find_one
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -9,8 +9,8 @@ class ConnectionTest < Test::Unit::TestCase
|
|||||||
TCPSocket.stubs(:new).returns(new_mock_socket)
|
TCPSocket.stubs(:new).returns(new_mock_socket)
|
||||||
|
|
||||||
admin_db = new_mock_db
|
admin_db = new_mock_db
|
||||||
admin_db.expects(:command).returns({'ok' => 1, 'ismaster' => 1}).twice
|
admin_db.expects(:command).returns({'ok' => 1, 'ismaster' => 1})
|
||||||
@conn.expects(:[]).with('admin').returns(admin_db).twice
|
@conn.expects(:[]).with('admin').returns(admin_db)
|
||||||
@conn.connect
|
@conn.connect
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -52,8 +52,8 @@ class ConnectionTest < Test::Unit::TestCase
|
|||||||
@conn = Connection.from_uri("mongodb://localhost", :connect => false)
|
@conn = Connection.from_uri("mongodb://localhost", :connect => false)
|
||||||
|
|
||||||
admin_db = new_mock_db
|
admin_db = new_mock_db
|
||||||
admin_db.expects(:command).returns({'ok' => 1, 'ismaster' => 1}).twice
|
admin_db.expects(:command).returns({'ok' => 1, 'ismaster' => 1})
|
||||||
@conn.expects(:[]).with('admin').returns(admin_db).twice
|
@conn.expects(:[]).with('admin').returns(admin_db)
|
||||||
@conn.connect
|
@conn.connect
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -20,7 +20,7 @@ class NodeTest < Test::Unit::TestCase
|
|||||||
assert_equal "power.level.com:#{Connection::DEFAULT_PORT}", node.address
|
assert_equal "power.level.com:#{Connection::DEFAULT_PORT}", node.address
|
||||||
end
|
end
|
||||||
|
|
||||||
should "load a node from a stirng" do
|
should "load a node from a string" do
|
||||||
node = Node.new(@connection, 'localhost:1234')
|
node = Node.new(@connection, 'localhost:1234')
|
||||||
assert_equal 'localhost', node.host
|
assert_equal 'localhost', node.host
|
||||||
assert_equal 1234, node.port
|
assert_equal 1234, node.port
|
||||||
|
@ -1,59 +0,0 @@
|
|||||||
require './test/test_helper'
|
|
||||||
include Mongo
|
|
||||||
|
|
||||||
class ReplSetConnectionTest < Test::Unit::TestCase
|
|
||||||
context "Initialization: " do
|
|
||||||
context "connecting to a replica set" do
|
|
||||||
setup do
|
|
||||||
TCPSocket.stubs(:new).returns(new_mock_socket('localhost', 27017))
|
|
||||||
@conn = ReplSetConnection.new(['localhost', 27017], :connect => false, :read_secondary => true)
|
|
||||||
|
|
||||||
admin_db = new_mock_db
|
|
||||||
@hosts = ['localhost:27018', 'localhost:27019', 'localhost:27020']
|
|
||||||
|
|
||||||
admin_db.stubs(:command).returns({'ok' => 1, 'ismaster' => 1, 'hosts' => @hosts}).
|
|
||||||
then.returns({'ok' => 1, 'ismaster' => 0, 'hosts' => @hosts, 'secondary' => 1}).
|
|
||||||
then.returns({'ok' => 1, 'ismaster' => 0, 'hosts' => @hosts, 'secondary' => 1}).
|
|
||||||
then.returns({'ok' => 1, 'ismaster' => 0, 'arbiterOnly' => 1})
|
|
||||||
|
|
||||||
@conn.stubs(:[]).with('admin').returns(admin_db)
|
|
||||||
@conn.connect
|
|
||||||
end
|
|
||||||
|
|
||||||
should "store the hosts returned from the ismaster command" do
|
|
||||||
assert_equal 'localhost', @conn.primary_pool.host
|
|
||||||
assert_equal 27017, @conn.primary_pool.port
|
|
||||||
|
|
||||||
assert_equal 'localhost', @conn.secondary_pools[0].host
|
|
||||||
assert_equal 27018, @conn.secondary_pools[0].port
|
|
||||||
|
|
||||||
assert_equal 'localhost', @conn.secondary_pools[1].host
|
|
||||||
assert_equal 27019, @conn.secondary_pools[1].port
|
|
||||||
|
|
||||||
assert_equal 2, @conn.secondary_pools.length
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
context "connecting to a replica set and providing seed nodes" do
|
|
||||||
setup do
|
|
||||||
TCPSocket.stubs(:new).returns(new_mock_socket)
|
|
||||||
@conn = ReplSetConnection.new(['localhost', 27017], ['localhost', 27019], :connect => false)
|
|
||||||
|
|
||||||
admin_db = new_mock_db
|
|
||||||
@hosts = ['localhost:27017', 'localhost:27018', 'localhost:27019']
|
|
||||||
admin_db.stubs(:command).returns({'ok' => 1, 'ismaster' => 1, 'hosts' => @hosts})
|
|
||||||
@conn.stubs(:[]).with('admin').returns(admin_db)
|
|
||||||
@conn.connect
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
context "initializing with a mongodb uri" do
|
|
||||||
|
|
||||||
should "parse a uri specifying multiple nodes" do
|
|
||||||
@conn = Connection.from_uri("mongodb://localhost:27017,mydb.com:27018", :connect => false)
|
|
||||||
assert_equal ['localhost', 27017], @conn.nodes[0]
|
|
||||||
assert_equal ['mydb.com', 27018], @conn.nodes[1]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
Loading…
Reference in New Issue
Block a user