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

138 lines
3.6 KiB
Ruby
Raw Normal View History

2012-04-20 13:41:39 -04:00
require 'delegate'
2010-04-14 23:32:56 -04:00
module Fog
2010-05-02 22:43:03 -04:00
module SSH
2010-04-14 23:32:56 -04:00
2010-05-02 22:43:03 -04:00
def self.new(address, username, options = {})
if Fog.mocking?
Fog::SSH::Mock.new(address, username, options)
else
Fog::SSH::Real.new(address, username, options)
end
end
class Mock
def self.data
@data ||= Hash.new do |hash, key|
2012-05-09 13:05:12 -04:00
hash[key] = []
2010-05-02 22:43:03 -04:00
end
end
2012-05-09 13:05:12 -04:00
def self.reset
@data= nil
end
2010-05-02 22:43:03 -04:00
def initialize(address, username, options)
@address = address
@username = username
@options = options
end
def run(commands, &blk)
2012-05-09 13:05:12 -04:00
self.class.data[@address] << {:commands => commands, :username => @username, :options => @options}
2010-05-02 22:43:03 -04:00
end
end
class Real
def initialize(address, username, options)
require 'net/ssh'
key_manager = Net::SSH::Authentication::KeyManager.new(nil, options)
unless options[:key_data] || options[:keys] || options[:password] || key_manager.agent
raise ArgumentError.new(':key_data, :keys, :password or a loaded ssh-agent is required to initialize SSH')
end
options[:timeout] = 30
if options[:key_data] || options[:keys]
options[:keys_only] = true
#Explicitly set these so net-ssh doesn't add the default keys
#as seen at https://github.com/net-ssh/net-ssh/blob/master/lib/net/ssh/authentication/session.rb#L131-146
options[:keys] = [] unless options[:keys]
options[:key_data] = [] unless options[:key_data]
end
2010-05-02 22:43:03 -04:00
@address = address
@username = username
@options = { :paranoid => false }.merge(options)
2010-05-02 22:43:03 -04:00
end
def run(commands, &blk)
2010-05-02 22:43:03 -04:00
commands = [*commands]
results = []
begin
Net::SSH.start(@address, @username, @options) do |ssh|
commands.each do |command|
result = Result.new(command)
ssh.open_channel do |ssh_channel|
ssh_channel.request_pty
ssh_channel.exec(command) do |channel, success|
2010-05-02 22:43:03 -04:00
unless success
raise "Could not execute command: #{command.inspect}"
end
channel.on_data do |ch, data|
2010-05-02 22:43:03 -04:00
result.stdout << data
2012-05-23 18:54:35 -04:00
yield [data, ''] if blk
2010-05-02 22:43:03 -04:00
end
channel.on_extended_data do |ch, type, data|
2010-05-02 22:43:03 -04:00
next unless type == 1
result.stderr << data
2012-05-23 18:54:35 -04:00
yield ['', data] if blk
2010-05-02 22:43:03 -04:00
end
channel.on_request('exit-status') do |ch, data|
2010-05-02 22:43:03 -04:00
result.status = data.read_long
end
channel.on_request('exit-signal') do |ch, data|
2010-05-02 22:43:03 -04:00
result.status = 255
end
2010-04-19 00:42:08 -04:00
end
end
2010-05-02 22:43:03 -04:00
ssh.loop
results << result
2010-04-19 00:42:08 -04:00
end
2010-04-14 23:32:56 -04:00
end
2010-05-02 22:43:03 -04:00
rescue Net::SSH::HostKeyMismatch => exception
exception.remember_host!
sleep 0.2
retry
2010-04-14 23:32:56 -04:00
end
2010-05-02 22:43:03 -04:00
results
end
end
class Result
attr_accessor :command, :stderr, :stdout, :status
def display_stdout
data = stdout.split("\r\n")
if data.is_a?(String)
Formatador.display_line(data)
elsif data.is_a?(Array)
Formatador.display_lines(data)
end
end
def display_stderr
Formatador.display_line(stderr.split("\r\n"))
end
def initialize(command)
2010-05-02 22:43:03 -04:00
@command = command
@stderr = ''
@stdout = ''
2010-04-14 23:32:56 -04:00
end
2010-05-02 22:43:03 -04:00
2010-04-14 23:32:56 -04:00
end
end
end