RUBY-232 handle authentication with connection pooling
This commit is contained in:
parent
285752a7ad
commit
7c4740c47c
@ -35,7 +35,7 @@ module Mongo
|
|||||||
STANDARD_HEADER_SIZE = 16
|
STANDARD_HEADER_SIZE = 16
|
||||||
RESPONSE_HEADER_SIZE = 20
|
RESPONSE_HEADER_SIZE = 20
|
||||||
|
|
||||||
attr_reader :logger, :size, :auths, :primary, :safe, :primary_pool, :host_to_try
|
attr_reader :logger, :size, :auths, :primary, :safe, :primary_pool, :host_to_try, :pool_size
|
||||||
|
|
||||||
# Counter for generating unique request ids.
|
# Counter for generating unique request ids.
|
||||||
@@current_request_id = 0
|
@@current_request_id = 0
|
||||||
@ -188,10 +188,11 @@ module Mongo
|
|||||||
#
|
#
|
||||||
# @raise [AuthenticationError] raises an exception if any one
|
# @raise [AuthenticationError] raises an exception if any one
|
||||||
# authentication fails.
|
# authentication fails.
|
||||||
def apply_saved_authentication
|
def apply_saved_authentication(opts={})
|
||||||
return false if @auths.empty?
|
return false if @auths.empty?
|
||||||
@auths.each do |auth|
|
@auths.each do |auth|
|
||||||
self[auth['db_name']].authenticate(auth['username'], auth['password'], false)
|
self[auth['db_name']].issue_authentication(auth['username'], auth['password'], false,
|
||||||
|
:socket => opts[:socket])
|
||||||
end
|
end
|
||||||
true
|
true
|
||||||
end
|
end
|
||||||
@ -241,6 +242,14 @@ module Mongo
|
|||||||
true
|
true
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def authenticate_pools
|
||||||
|
@primary_pool.authenticate_existing
|
||||||
|
end
|
||||||
|
|
||||||
|
def logout_pools(db)
|
||||||
|
@primary_pool.logout_existing(db)
|
||||||
|
end
|
||||||
|
|
||||||
# Return a hash with all database names
|
# Return a hash with all database names
|
||||||
# and their respective sizes on disk.
|
# and their respective sizes on disk.
|
||||||
#
|
#
|
||||||
@ -411,7 +420,13 @@ module Mongo
|
|||||||
request_id = add_message_headers(message, operation)
|
request_id = add_message_headers(message, operation)
|
||||||
packed_message = message.to_s
|
packed_message = message.to_s
|
||||||
begin
|
begin
|
||||||
sock = socket || (command ? checkout_writer : checkout_reader)
|
if socket
|
||||||
|
sock = socket
|
||||||
|
checkin = false
|
||||||
|
else
|
||||||
|
sock = (command ? checkout_writer : checkout_reader)
|
||||||
|
checkin = true
|
||||||
|
end
|
||||||
|
|
||||||
result = ''
|
result = ''
|
||||||
@safe_mutexes[sock].synchronize do
|
@safe_mutexes[sock].synchronize do
|
||||||
@ -419,7 +434,9 @@ module Mongo
|
|||||||
result = receive(sock, request_id)
|
result = receive(sock, request_id)
|
||||||
end
|
end
|
||||||
ensure
|
ensure
|
||||||
command ? checkin_writer(sock) : checkin_reader(sock)
|
if checkin
|
||||||
|
command ? checkin_writer(sock) : checkin_reader(sock)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
result
|
result
|
||||||
end
|
end
|
||||||
@ -588,7 +605,7 @@ module Mongo
|
|||||||
socket = TCPSocket.new(host, port)
|
socket = TCPSocket.new(host, port)
|
||||||
socket.setsockopt(Socket::IPPROTO_TCP, Socket::TCP_NODELAY, 1)
|
socket.setsockopt(Socket::IPPROTO_TCP, Socket::TCP_NODELAY, 1)
|
||||||
|
|
||||||
config = self['admin'].command({:ismaster => 1}, :sock => socket)
|
config = self['admin'].command({:ismaster => 1}, :socket => socket)
|
||||||
rescue OperationFailure, SocketError, SystemCallError, IOError => ex
|
rescue OperationFailure, SocketError, SystemCallError, IOError => ex
|
||||||
close
|
close
|
||||||
ensure
|
ensure
|
||||||
@ -598,16 +615,13 @@ module Mongo
|
|||||||
config
|
config
|
||||||
end
|
end
|
||||||
|
|
||||||
# Set the specified node as primary, and
|
# Set the specified node as primary.
|
||||||
# apply any saved authentication credentials.
|
|
||||||
def set_primary(node)
|
def set_primary(node)
|
||||||
host, port = *node
|
host, port = *node
|
||||||
@primary = [host, port]
|
@primary = [host, port]
|
||||||
@primary_pool = Pool.new(self, host, port, :size => @pool_size, :timeout => @timeout)
|
@primary_pool = Pool.new(self, host, port, :size => @pool_size, :timeout => @timeout)
|
||||||
apply_saved_authentication
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
## Low-level connection methods.
|
## Low-level connection methods.
|
||||||
|
|
||||||
def receive(sock, expected_response)
|
def receive(sock, expected_response)
|
||||||
|
@ -92,7 +92,8 @@ module Mongo
|
|||||||
# @param [String] password
|
# @param [String] password
|
||||||
# @param [Boolean] save_auth
|
# @param [Boolean] save_auth
|
||||||
# Save this authentication to the connection object using Connection#add_auth. This
|
# Save this authentication to the connection object using Connection#add_auth. This
|
||||||
# will ensure that the authentication will be applied on database reconnect.
|
# will ensure that the authentication will be applied on database reconnect. Note
|
||||||
|
# that this value must be true when using connection pooling.
|
||||||
#
|
#
|
||||||
# @return [Boolean]
|
# @return [Boolean]
|
||||||
#
|
#
|
||||||
@ -100,8 +101,19 @@ module Mongo
|
|||||||
#
|
#
|
||||||
# @core authenticate authenticate-instance_method
|
# @core authenticate authenticate-instance_method
|
||||||
def authenticate(username, password, save_auth=true)
|
def authenticate(username, password, save_auth=true)
|
||||||
doc = command({:getnonce => 1}, :check_response => false)
|
if @connection.pool_size > 1
|
||||||
raise "error retrieving nonce: #{doc}" unless ok?(doc)
|
if !save_auth
|
||||||
|
raise MongoArgumentError, "If using connection pooling, :save_auth must be set to true."
|
||||||
|
end
|
||||||
|
@connection.authenticate_pools
|
||||||
|
end
|
||||||
|
|
||||||
|
issue_authentication(username, password, save_auth)
|
||||||
|
end
|
||||||
|
|
||||||
|
def issue_authentication(username, password, save_auth=true, opts={})
|
||||||
|
doc = command({:getnonce => 1}, :check_response => false, :socket => opts[:socket])
|
||||||
|
raise MongoDBError, "Error retrieving nonce: #{doc}" unless ok?(doc)
|
||||||
nonce = doc['nonce']
|
nonce = doc['nonce']
|
||||||
|
|
||||||
auth = BSON::OrderedHash.new
|
auth = BSON::OrderedHash.new
|
||||||
@ -109,7 +121,7 @@ module Mongo
|
|||||||
auth['user'] = username
|
auth['user'] = username
|
||||||
auth['nonce'] = nonce
|
auth['nonce'] = nonce
|
||||||
auth['key'] = Mongo::Support.auth_key(username, password, nonce)
|
auth['key'] = Mongo::Support.auth_key(username, password, nonce)
|
||||||
if ok?(self.command(auth, :check_response => false))
|
if ok?(self.command(auth, :check_response => false, :socket => opts[:socket]))
|
||||||
if save_auth
|
if save_auth
|
||||||
@connection.add_auth(@name, username, password)
|
@connection.add_auth(@name, username, password)
|
||||||
end
|
end
|
||||||
@ -179,14 +191,22 @@ module Mongo
|
|||||||
end
|
end
|
||||||
|
|
||||||
# Deauthorizes use for this database for this connection. Also removes
|
# Deauthorizes use for this database for this connection. Also removes
|
||||||
# any saved authorization in the connection class associated with this
|
# any saved authentication in the connection class associated with this
|
||||||
# database.
|
# database.
|
||||||
#
|
#
|
||||||
# @raise [MongoDBError] if logging out fails.
|
# @raise [MongoDBError] if logging out fails.
|
||||||
#
|
#
|
||||||
# @return [Boolean]
|
# @return [Boolean]
|
||||||
def logout
|
def logout(opts={})
|
||||||
doc = command(:logout => 1)
|
if @connection.pool_size > 1
|
||||||
|
@connection.logout_pools(@name)
|
||||||
|
end
|
||||||
|
|
||||||
|
issue_logout(opts)
|
||||||
|
end
|
||||||
|
|
||||||
|
def issue_logout(opts={})
|
||||||
|
doc = command({:logout => 1}, :socket => opts[:socket])
|
||||||
if ok?(doc)
|
if ok?(doc)
|
||||||
@connection.remove_auth(@name)
|
@connection.remove_auth(@name)
|
||||||
true
|
true
|
||||||
@ -455,14 +475,14 @@ module Mongo
|
|||||||
#
|
#
|
||||||
# @option opts [Boolean] :check_response (true) If +true+, raises an exception if the
|
# @option opts [Boolean] :check_response (true) If +true+, raises an exception if the
|
||||||
# command fails.
|
# command fails.
|
||||||
# @option opts [Socket] :sock a socket to use for sending the command. This is mainly for internal use.
|
# @option opts [Socket] :socket a socket to use for sending the command. This is mainly for internal use.
|
||||||
#
|
#
|
||||||
# @return [Hash]
|
# @return [Hash]
|
||||||
#
|
#
|
||||||
# @core commands command_instance-method
|
# @core commands command_instance-method
|
||||||
def command(selector, opts={})
|
def command(selector, opts={})
|
||||||
check_response = opts.fetch(:check_response, true)
|
check_response = opts.fetch(:check_response, true)
|
||||||
sock = opts[:sock]
|
socket = opts[:socket]
|
||||||
raise MongoArgumentError, "command must be given a selector" unless selector.is_a?(Hash) && !selector.empty?
|
raise MongoArgumentError, "command must be given a selector" unless selector.is_a?(Hash) && !selector.empty?
|
||||||
if selector.keys.length > 1 && RUBY_VERSION < '1.9' && selector.class != BSON::OrderedHash
|
if selector.keys.length > 1 && RUBY_VERSION < '1.9' && selector.class != BSON::OrderedHash
|
||||||
raise MongoArgumentError, "DB#command requires an OrderedHash when hash contains multiple keys"
|
raise MongoArgumentError, "DB#command requires an OrderedHash when hash contains multiple keys"
|
||||||
@ -470,7 +490,7 @@ module Mongo
|
|||||||
|
|
||||||
begin
|
begin
|
||||||
result = Cursor.new(system_command_collection,
|
result = Cursor.new(system_command_collection,
|
||||||
:limit => -1, :selector => selector, :socket => sock).next_document
|
:limit => -1, :selector => selector, :socket => socket).next_document
|
||||||
rescue OperationFailure => ex
|
rescue OperationFailure => ex
|
||||||
raise OperationFailure, "Database command '#{selector.keys.first}' failed: #{ex.message}"
|
raise OperationFailure, "Database command '#{selector.keys.first}' failed: #{ex.message}"
|
||||||
end
|
end
|
||||||
|
@ -167,6 +167,20 @@ module Mongo
|
|||||||
@read_secondary || @slave_ok
|
@read_secondary || @slave_ok
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def authenticate_pools
|
||||||
|
super
|
||||||
|
@secondary_pools.each do |pool|
|
||||||
|
pool.authenticate_existing
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def logout_pools(db)
|
||||||
|
super
|
||||||
|
@secondary_pools.each do |pool|
|
||||||
|
pool.logout_existing(db)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def check_is_master(node)
|
def check_is_master(node)
|
||||||
|
@ -37,6 +37,9 @@ module Mongo
|
|||||||
# Condition variable for signal and wait
|
# Condition variable for signal and wait
|
||||||
@queue = ConditionVariable.new
|
@queue = ConditionVariable.new
|
||||||
|
|
||||||
|
# Operations to perform on a socket
|
||||||
|
@socket_ops = Hash.new { |h, k| h[k] = [] }
|
||||||
|
|
||||||
@sockets = []
|
@sockets = []
|
||||||
@checked_out = []
|
@checked_out = []
|
||||||
end
|
end
|
||||||
@ -75,11 +78,42 @@ module Mongo
|
|||||||
rescue => ex
|
rescue => ex
|
||||||
raise ConnectionFailure, "Failed to connect socket: #{ex}"
|
raise ConnectionFailure, "Failed to connect socket: #{ex}"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# If any saved authentications exist, we want to apply those
|
||||||
|
# when creating new sockets.
|
||||||
|
@connection.apply_saved_authentication(:socket => socket)
|
||||||
|
|
||||||
@sockets << socket
|
@sockets << socket
|
||||||
@checked_out << socket
|
@checked_out << socket
|
||||||
socket
|
socket
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# If a use calls DB#authentication, and several sockets exist,
|
||||||
|
# then we need a way to apply the authentication on each socket.
|
||||||
|
# So we store the apply_authentication method, and this will be
|
||||||
|
# applied right before the next use of each socket.
|
||||||
|
def authenticate_existing
|
||||||
|
@connection_mutex.synchronize do
|
||||||
|
@sockets.each do |socket|
|
||||||
|
@socket_ops[socket] << Proc.new do
|
||||||
|
@connection.apply_saved_authentication(:socket => socket)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Store the logout op for each existing socket to be applied before
|
||||||
|
# the next use of each socket.
|
||||||
|
def logout_existing(db)
|
||||||
|
@connection_mutex.synchronize do
|
||||||
|
@sockets.each do |socket|
|
||||||
|
@socket_ops[socket] << Proc.new do
|
||||||
|
@connection.db(db).issue_logout(:socket => socket)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
# Checks out the first available socket from the pool.
|
# Checks out the first available socket from the pool.
|
||||||
#
|
#
|
||||||
# This method is called exclusively from #checkout;
|
# This method is called exclusively from #checkout;
|
||||||
@ -110,14 +144,24 @@ module Mongo
|
|||||||
checkout_new_socket
|
checkout_new_socket
|
||||||
end
|
end
|
||||||
|
|
||||||
return socket if socket
|
if socket
|
||||||
|
|
||||||
# Otherwise, wait
|
# This call all procs, in order, scoped to existing sockets.
|
||||||
if @logger
|
# At the moment, we use this to lazily authenticate and
|
||||||
@logger.warn "MONGODB Waiting for available connection; " +
|
# logout existing socket connections.
|
||||||
"#{@checked_out.size} of #{@size} connections checked out."
|
@socket_ops[socket].reject! do |op|
|
||||||
|
op.call
|
||||||
|
end
|
||||||
|
|
||||||
|
return socket
|
||||||
|
else
|
||||||
|
# Otherwise, wait
|
||||||
|
if @logger
|
||||||
|
@logger.warn "MONGODB Waiting for available connection; " +
|
||||||
|
"#{@checked_out.size} of #{@size} connections checked out."
|
||||||
|
end
|
||||||
|
@queue.wait(@connection_mutex)
|
||||||
end
|
end
|
||||||
@queue.wait(@connection_mutex)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
101
test/auxillary/threaded_authentication_test.rb
Normal file
101
test/auxillary/threaded_authentication_test.rb
Normal file
@ -0,0 +1,101 @@
|
|||||||
|
$:.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
|
||||||
|
require 'mongo'
|
||||||
|
require 'thread'
|
||||||
|
require 'test/unit'
|
||||||
|
require './test/test_helper'
|
||||||
|
|
||||||
|
# NOTE: This test requires bouncing the server.
|
||||||
|
# It also requires that a user exists on the admin database.
|
||||||
|
class AuthenticationTest < Test::Unit::TestCase
|
||||||
|
include Mongo
|
||||||
|
|
||||||
|
def setup
|
||||||
|
@conn = standard_connection(:pool_size => 10)
|
||||||
|
@db1 = @conn.db('mongo-ruby-test-auth1')
|
||||||
|
@db2 = @conn.db('mongo-ruby-test-auth2')
|
||||||
|
@admin = @conn.db('admin')
|
||||||
|
end
|
||||||
|
|
||||||
|
def teardown
|
||||||
|
@db1.authenticate('user1', 'secret')
|
||||||
|
@db2.authenticate('user2', 'secret')
|
||||||
|
@conn.drop_database('mongo-ruby-test-auth1')
|
||||||
|
@conn.drop_database('mongo-ruby-test-auth2')
|
||||||
|
end
|
||||||
|
|
||||||
|
def threaded_exec
|
||||||
|
threads = []
|
||||||
|
|
||||||
|
100.times do
|
||||||
|
threads << Thread.new do
|
||||||
|
yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
100.times do |n|
|
||||||
|
threads[n].join
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_authenticate
|
||||||
|
@admin.authenticate('bob', 'secret')
|
||||||
|
@db1.add_user('user1', 'secret')
|
||||||
|
@db2.add_user('user2', 'secret')
|
||||||
|
@admin.logout
|
||||||
|
|
||||||
|
threaded_exec do
|
||||||
|
assert_raise Mongo::OperationFailure do
|
||||||
|
@db1['stuff'].insert({:a => 2}, :safe => true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
threaded_exec do
|
||||||
|
assert_raise Mongo::OperationFailure do
|
||||||
|
@db2['stuff'].insert({:a => 2}, :safe => true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
@db1.authenticate('user1', 'secret')
|
||||||
|
@db2.authenticate('user2', 'secret')
|
||||||
|
|
||||||
|
threaded_exec do
|
||||||
|
assert @db1['stuff'].insert({:a => 2}, :safe => true)
|
||||||
|
end
|
||||||
|
|
||||||
|
threaded_exec do
|
||||||
|
assert @db2['stuff'].insert({:a => 2}, :safe => true)
|
||||||
|
end
|
||||||
|
|
||||||
|
puts "Please bounce the server."
|
||||||
|
gets
|
||||||
|
|
||||||
|
# Here we reconnect.
|
||||||
|
begin
|
||||||
|
@db1['stuff'].find.to_a
|
||||||
|
rescue Mongo::ConnectionFailure
|
||||||
|
end
|
||||||
|
|
||||||
|
threaded_exec do
|
||||||
|
assert @db1['stuff'].insert({:a => 2}, :safe => true)
|
||||||
|
end
|
||||||
|
|
||||||
|
threaded_exec do
|
||||||
|
assert @db2['stuff'].insert({:a => 2}, :safe => true)
|
||||||
|
end
|
||||||
|
|
||||||
|
@db1.logout
|
||||||
|
threaded_exec do
|
||||||
|
assert_raise Mongo::OperationFailure do
|
||||||
|
@db1['stuff'].insert({:a => 2}, :safe => true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
@db2.logout
|
||||||
|
threaded_exec do
|
||||||
|
assert_raise Mongo::OperationFailure do
|
||||||
|
assert @db2['stuff'].insert({:a => 2}, :safe => true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
@ -67,7 +67,6 @@ class ConnectionTest < Test::Unit::TestCase
|
|||||||
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}).twice
|
||||||
@conn.expects(:[]).with('admin').returns(admin_db).twice
|
@conn.expects(:[]).with('admin').returns(admin_db).twice
|
||||||
@conn.expects(:apply_saved_authentication)
|
|
||||||
@conn.connect
|
@conn.connect
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -57,6 +57,7 @@ class DBTest < Test::Unit::TestCase
|
|||||||
|
|
||||||
should "raise an error if logging out fails" do
|
should "raise an error if logging out fails" do
|
||||||
@db.expects(:command).returns({})
|
@db.expects(:command).returns({})
|
||||||
|
@conn.expects(:pool_size).returns(1)
|
||||||
assert_raise Mongo::MongoDBError do
|
assert_raise Mongo::MongoDBError do
|
||||||
@db.logout
|
@db.logout
|
||||||
end
|
end
|
||||||
|
Loading…
Reference in New Issue
Block a user