require 'delegate' module Fog module SSH 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| hash[key] = {} end end def initialize(address, username, options) @address = address @username = username @options = options end def run(commands) Fog::Mock.not_implemented 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 @address = address @username = username @debug = options.delete :debug @options = { :paranoid => false }.merge(options) end def run(commands) commands = [*commands] results = [] begin Net::SSH.start(@address, @username, @options) do |ssh| commands.each do |command| result = Result.new(command, @debug) ssh.open_channel do |ssh_channel| ssh_channel.request_pty ssh_channel.exec(command) do |channel, success| unless success raise "Could not execute command: #{command.inspect}" end channel.on_data do |ch, data| result.stdout << data end channel.on_extended_data do |ch, type, data| next unless type == 1 result.stderr << data end channel.on_request('exit-status') do |ch, data| result.status = data.read_long end channel.on_request('exit-signal') do |ch, data| result.status = 255 end end end ssh.loop results << result end end rescue Net::SSH::HostKeyMismatch => exception exception.remember_host! sleep 0.2 retry end results end end class DebugString < SimpleDelegator def initialize(string='') super end def <<(add_me) puts add_me super 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, debug=false) @command = command @stderr = debug ? DebugString.new : '' @stdout = debug ? DebugString.new : '' end end end end