Skip to content

Commit

Permalink
(hashicorp#6827) Introduce tty flag for ssh command execution
Browse files Browse the repository at this point in the history
Prior to this commit, if a user ran the `vagrant ssh -c CMD` command, it
would not allow the user to configure pseudo-terminal allocation. This
commit introduces a -t flag for the `vagrant ssh` command which defaults
to true if not specified.
  • Loading branch information
briancain committed Jun 3, 2017
1 parent 875aee3 commit 8370a9b
Show file tree
Hide file tree
Showing 3 changed files with 107 additions and 2 deletions.
4 changes: 3 additions & 1 deletion lib/vagrant/action/builtin/ssh_run.rb
Expand Up @@ -44,7 +44,9 @@ def call(env)

# Allow the user to specify a tty or non-tty manually, but if they
# don't then we default to a TTY
if !opts[:extra_args].include?("-t") && !opts[:extra_args].include?("-T")
if !opts[:extra_args].include?("-t") &&
!opts[:extra_args].include?("-T") &&
env[:tty]
opts[:extra_args] << "-t"
end

Expand Down
8 changes: 7 additions & 1 deletion plugins/commands/ssh/command.rb
Expand Up @@ -9,6 +9,7 @@ def self.synopsis

def execute
options = {}
options[:tty] = true

opts = OptionParser.new do |o|
o.banner = "Usage: vagrant ssh [options] [name|id] [-- extra ssh args]"
Expand All @@ -23,6 +24,10 @@ def execute
o.on("-p", "--plain", "Plain mode, leaves authentication up to user") do |p|
options[:plain_mode] = p
end

o.on("-t", "--[no-]tty", "Enables tty when executing an ssh command (defaults to true)") do |t|
options[:tty] = t
end
end

# Parse out the extra args to send to SSH, which is everything
Expand All @@ -48,7 +53,8 @@ def execute
@logger.debug("Executing single command on remote machine: #{options[:command]}")
env = vm.action(:ssh_run,
ssh_opts: ssh_opts,
ssh_run_command: options[:command],)
ssh_run_command: options[:command],
tty: options[:tty],)

# Exit with the exit status of the command or a 0 if we didn't
# get one.
Expand Down
97 changes: 97 additions & 0 deletions test/unit/vagrant/action/builtin/ssh_run_test.rb
@@ -0,0 +1,97 @@
require File.expand_path("../../../../base", __FILE__)

describe Vagrant::Action::Builtin::SSHRun do
let(:app) { lambda { |env| } }
let(:env) { { machine: machine } }

# SSH configuration information mock
let(:ssh) do
double("ssh",
timeout: 1,
host: nil,
port: 5986,
guest_port: 5986,
pty: false,
keep_alive: false,
insert_key: false,
shell: 'bash -l'
)
end

# Configuration mock
let(:config) { double("config", ssh: ssh) }

let(:machine) do
double("machine",
config: config,)
end

let(:machine_ssh_info) { {} }
let(:ssh_klass) { Vagrant::Util::SSH }

before(:each) do
# Stub the methods so that even if we test incorrectly, no side
# effects actually happen.
allow(ssh_klass).to receive(:exec)
allow(machine).to receive(:ssh_info).and_return(machine_ssh_info)
end

it "should raise an exception if SSH is not ready" do
not_ready_machine = double("machine")
allow(not_ready_machine).to receive(:ssh_info).and_return(nil)

env[:machine] = not_ready_machine
expect { described_class.new(app, env).call(env) }.
to raise_error(Vagrant::Errors::SSHNotReady)
end

it "should exec with the SSH info in the env if given" do
ssh_info = { foo: :bar }
opts = {:extra_args=>["bash -l -c 'echo test'"], :subprocess=>true}

expect(ssh_klass).to receive(:exec).
with(ssh_info, opts)

env[:ssh_info] = ssh_info
env[:ssh_run_command] = "echo test"
described_class.new(app, env).call(env)
end


it "should exec with the SSH info in the env if given and disable tty" do
ssh_info = { foo: :bar }
opts = {:extra_args=>["-t", "bash -l -c 'echo test'"], :subprocess=>true}
env[:tty] = true

expect(ssh_klass).to receive(:exec).
with(ssh_info, opts)

env[:ssh_info] = ssh_info
env[:ssh_run_command] = "echo test"
described_class.new(app, env).call(env)
end

it "should exec with the SSH info in the env if given and not disable tty" do
ssh_info = { foo: :bar }
opts = {:extra_args=>["bash -l -c 'echo test'"], :subprocess=>true}
env[:tty] = false

expect(ssh_klass).to receive(:exec).
with(ssh_info, opts)

env[:ssh_info] = ssh_info
env[:ssh_run_command] = "echo test"
described_class.new(app, env).call(env)
end

it "should exec with the options given in `ssh_opts`" do
ssh_opts = { foo: :bar }

expect(ssh_klass).to receive(:exec).
with(machine_ssh_info, ssh_opts)

env[:ssh_opts] = ssh_opts
env[:ssh_run_command] = "echo test"
described_class.new(app, env).call(env)
end
end

0 comments on commit 8370a9b

Please sign in to comment.