mongo-ruby-driver/test/tools/repl_set_manager.rb

155 lines
3.5 KiB
Ruby
Raw Normal View History

2010-12-13 20:22:51 +00:00
#!/usr/bin/ruby
2010-12-13 21:25:23 +00:00
STDOUT.sync = true
unless defined? Mongo
require File.join(File.dirname(__FILE__), '..', '..', 'lib', 'mongo')
end
2010-12-13 20:22:51 +00:00
class ReplSetManager
2010-12-13 21:25:23 +00:00
attr_accessor :host, :start_port, :ports
2010-12-13 20:22:51 +00:00
def initialize(opts={})
@start_port = opts[:start_port] || 30000
2010-12-13 21:25:23 +00:00
@ports = []
2010-12-13 20:22:51 +00:00
@name = opts[:name] || 'replica-set-foo'
@count = opts[:count] || 3
@host = opts[:host] || 'localhost'
@retries = opts[:retries] || 60
@config = {"_id" => @name, "members" => []}
@path = File.join(File.expand_path(File.dirname(__FILE__)), "data")
@mongods = {}
end
def start_set
puts "Starting a replica set with #{@count} nodes"
system("killall mongod")
2010-12-13 21:25:23 +00:00
2010-12-13 20:22:51 +00:00
@count.times do |n|
@mongods[n] ||= {}
2010-12-13 21:25:23 +00:00
port = @start_port + n
@ports << port
@mongods[n]['port'] = port
@mongods[n]['db_path'] = get_path("rs-#{port}")
@mongods[n]['log_path'] = get_path("log-#{port}")
2010-12-13 20:22:51 +00:00
system("rm -rf #{@mongods[n]['db_path']}")
system("mkdir -p #{@mongods[n]['db_path']}")
@mongods[n]['start'] = "mongod --replSet #{@name} --logpath '#{@mongods[n]['log_path']}' " +
" --dbpath #{@mongods[n]['db_path']} --port #{@mongods[n]['port']} --fork"
start(n)
member = {'_id' => n, 'host' => "#{@host}:#{@mongods[n]['port']}"}
if n == @count-1
@mongods[n]['arbiter'] = true
member['arbiterOnly'] = true
end
@config['members'] << member
end
init
ensure_up
end
def kill(node)
system("kill -2 #{@mongods[node]['pid']}")
@mongods[node]['up'] = false
sleep(1)
end
2010-12-13 21:25:23 +00:00
def kill_primary
node = get_node_with_state(1)
kill(node)
return node
end
def kill_secondary
node = get_node_with_state(2)
kill(node)
return node
end
2010-12-13 20:22:51 +00:00
def start(node)
system(@mongods[node]['start'])
@mongods[node]['up'] = true
sleep(1)
@mongods[node]['pid'] = File.open(File.join(@mongods[node]['db_path'], 'mongod.lock')).read.strip
end
alias :restart :start
def ensure_up
2010-12-13 21:25:23 +00:00
print "Ensuring members are up..."
2010-12-13 20:22:51 +00:00
@con = get_connection
attempt(Mongo::OperationFailure) do
status = @con['admin'].command({'replSetGetStatus' => 1})
2010-12-13 21:25:23 +00:00
print "."
2010-12-13 20:22:51 +00:00
if status['members'].all? { |m| [1, 2, 7].include?(m['state']) }
puts "All members up!"
2010-12-13 21:25:23 +00:00
return status
2010-12-13 20:22:51 +00:00
else
raise Mongo::OperationFailure
end
end
end
private
def init
get_connection
attempt(Mongo::OperationFailure) do
@con['admin'].command({'replSetInitiate' => @config})
end
end
2010-12-13 21:25:23 +00:00
def get_node_with_state(state)
status = ensure_up
node = status['members'].detect {|m| m['state'] == state}
if node
host_port = node['name'].split(':')
port = host_port[1] ? host_port[1].to_i : 27017
key = @mongods.keys.detect {|key| @mongods[key]['port'] == port}
return key
else
return false
end
end
2010-12-13 20:22:51 +00:00
def get_connection
attempt(Mongo::ConnectionFailure) do
node = @mongods.keys.detect {|key| !@mongods[key]['arbiter'] && @mongods[key]['up'] }
@con = Mongo::Connection.new(@host, @mongods[node]['port'], :slave_ok => true)
end
return @con
end
def get_path(name)
2010-12-13 21:25:23 +00:00
File.join(@path, name)
2010-12-13 20:22:51 +00:00
end
def attempt(exception)
raise "No block given!" unless block_given?
count = 0
while count < @retries do
begin
yield
return
rescue exception
sleep(1)
count += 1
end
end
raise exception
end
end