1
0
Fork 0
mirror of https://github.com/fog/fog.git synced 2022-11-09 13:51:43 -05:00

toward more flexible/resilient ssh

This commit is contained in:
geemus 2010-05-02 19:43:03 -07:00
parent ae603ca624
commit 41f78339ab

View file

@ -1,42 +1,108 @@
module Fog
module SSH
class SSH
def initialize(address, username, options = {})
def self.new(address, username, options = {})
unless options[:keys] || options[:password]
raise ArgumentError.new(':keys or :password are required to initialize SSH')
end
@address = address
@username = username
@options = options.merge!(:paranoid => false)
if Fog.mocking?
Fog::SSH::Mock.new(address, username, options)
else
Fog::SSH::Real.new(address, username, options)
end
end
def run(commands)
commands = [*commands]
results = []
Net::SSH.start(@address, @username, @options) do |ssh|
commands.each do |command|
ssh.open_channel do |channel|
channel.request_pty
result = { :command => command }
channel.exec(command.sub(/^sudo/, %q{sudo -p 'fog sudo password:'})) do |channel, success|
channel.on_data do |channel, data|
if data.strip == 'fog sudo password:'
channel.send_data("#{@options[:password]}\n")
else
result[:data] ||= ''
result[:data] << data
end
end
end
results << result
end
ssh.loop
def self.reset_data(keys=Mock.data.keys)
Mock.reset_data(keys)
end
class Mock
def self.data
@data ||= Hash.new do |hash, key|
hash[key] = {}
end
end
results
def initialize(address, username, options)
@address = address
@username = username
@options = options
end
def run(commands)
raise MockNotImplemented.new("Contributions welcome!")
end
end
class Real
def initialize(address, username, options)
@address = address
@username = username
@options = options.merge!(:paranoid => false)
end
def run(commands)
commands = [*commands]
results = []
begin
Net::SSH.start(@address, @username, @options) do |ssh|
commands.each do |command|
ssh.open_channel do |channel|
sudoable_command = command.sub(/^sudo/, %{sudo -p 'fog sudo password:'})
escaped_command = sudoable_command.sub(/'/, %{'"'"'})
channel.request_pty
result = Result.new(command)
channel.exec(%{bash -lc '#{escaped_command}'}) do |channel, success|
unless success
raise "Could not execute command: #{command.inspect}"
end
channel.on_data do |channel, data|
result.stdout << data
end
channel.on_extended_data do |channel, type, data|
next unless type == 1
result.stderr << data
end
channel.on_request('exit-status') do |channel, data|
result.status = data.read_long
end
channel.on_request('exit-signal') do |channel, data|
result.status = 255
end
end
results << result
end
ssh.loop
end
end
rescue Net::SSH::HostKeyMismatch => exception
exception.remember_host!
sleep 0.2
retry
end
results
end
end
class Result
attr_accessor :command, :stderr, :stdout, :status
def initialize(command)
@command = command
@stderr = ''
@stdout = ''
end
end
end
end