require 'serverspec/backend/exec' module Serverspec module Backend class Ssh < Exec def run_command(cmd, opt={}) cmd = build_command(cmd) cmd = add_pre_command(cmd) ret = ssh_exec!(cmd) if @example @example.metadata[:command] = cmd @example.metadata[:stdout] = ret[:stdout] end ret end def build_command(cmd) cmd = super(cmd) cmd = "sudo #{cmd}" if RSpec.configuration.ssh.options[:user] != 'root' cmd end def add_pre_command(cmd) cmd = super(cmd) user = RSpec.configuration.ssh.options[:user] pre_command = Serverspec.configuration.pre_command if pre_command && user != 'root' cmd = "sudo #{cmd}" end cmd end private def ssh_exec!(command) stdout_data = '' stderr_data = '' exit_status = nil exit_signal = nil ssh = RSpec.configuration.ssh ssh.open_channel do |channel| channel.request_pty do |ch, success| abort "Could not obtain pty " if !success end channel.exec("#{command}") do |ch, success| abort "FAILED: couldn't execute command (ssh.channel.exec)" if !success channel.on_data do |ch, data| if data =~ /^\[sudo\] password for/ abort "Please set sudo password by using SUDO_PASSWORD or ASK_SUDO_PASSWORD environment variable" if RSpec.configuration.sudo_password.nil? channel.send_data "#{RSpec.configuration.sudo_password}\n" else stdout_data += data end end channel.on_extended_data do |ch, type, data| stderr_data += data end channel.on_request("exit-status") do |ch, data| exit_status = data.read_long end channel.on_request("exit-signal") do |ch, data| exit_signal = data.read_long end end end ssh.loop { :stdout => stdout_data, :stderr => stderr_data, :exit_status => exit_status, :exit_signal => exit_signal } end end end end