Move SSH communication to a plugin
This commit is contained in:
parent
a1cef830e3
commit
595e7cee0e
|
@ -66,7 +66,6 @@ module Vagrant
|
||||||
autoload :BoxCollection, 'vagrant/box_collection'
|
autoload :BoxCollection, 'vagrant/box_collection'
|
||||||
autoload :CLI, 'vagrant/cli'
|
autoload :CLI, 'vagrant/cli'
|
||||||
autoload :Command, 'vagrant/command'
|
autoload :Command, 'vagrant/command'
|
||||||
autoload :Communication, 'vagrant/communication'
|
|
||||||
autoload :Config, 'vagrant/config'
|
autoload :Config, 'vagrant/config'
|
||||||
autoload :DataStore, 'vagrant/data_store'
|
autoload :DataStore, 'vagrant/data_store'
|
||||||
autoload :Downloaders, 'vagrant/downloaders'
|
autoload :Downloaders, 'vagrant/downloaders'
|
||||||
|
|
|
@ -1,7 +0,0 @@
|
||||||
module Vagrant
|
|
||||||
module Communication
|
|
||||||
autoload :Base, 'vagrant/communication/base'
|
|
||||||
|
|
||||||
autoload :SSH, 'vagrant/communication/ssh'
|
|
||||||
end
|
|
||||||
end
|
|
|
@ -1,63 +0,0 @@
|
||||||
module Vagrant
|
|
||||||
module Communication
|
|
||||||
# The base class for any classes that provide an API for communicating
|
|
||||||
# with the virtual machine.
|
|
||||||
#
|
|
||||||
# There are various stages that require Vagrant to copy files or
|
|
||||||
# run commands on the target system, and communication classes provide
|
|
||||||
# the abstraction necessary to perform these tasks, via SSH or some
|
|
||||||
# other mechanism.
|
|
||||||
#
|
|
||||||
# Any subclasses of this class **must** implement all of the methods
|
|
||||||
# below.
|
|
||||||
class Base
|
|
||||||
# Checks if the target machine is ready for communication.
|
|
||||||
#
|
|
||||||
# @return [Boolean]
|
|
||||||
def ready?
|
|
||||||
end
|
|
||||||
|
|
||||||
# Download a file from the virtual machine to the local machine.
|
|
||||||
#
|
|
||||||
# @param [String] from Path of the file on the virtual machine.
|
|
||||||
# @param [String] to Path to where to save the remote file.
|
|
||||||
def download(from, to)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Upload a file to the virtual machine.
|
|
||||||
#
|
|
||||||
# @param [String] from Path to a file to upload.
|
|
||||||
# @param [String] to Path to where to save this file.
|
|
||||||
def upload(from, to)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Execute a command on the remote machine.
|
|
||||||
#
|
|
||||||
# @param [String] command Command to execute.
|
|
||||||
# @yield [type, data] Realtime output of the command being executed.
|
|
||||||
# @yieldparam [String] type Type of the output, `:stdout`, `:stderr`, etc.
|
|
||||||
# @yieldparam [String] data Data for the given output.
|
|
||||||
# @return [Integer] Exit code of the command.
|
|
||||||
def execute(command, opts=nil)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Execute a comand with super user privileges.
|
|
||||||
#
|
|
||||||
# See #execute for parameter information.
|
|
||||||
def sudo(command, opts=nil)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Executes a command and returns a boolean statement if it was successful
|
|
||||||
# or not.
|
|
||||||
#
|
|
||||||
# This is implemented by default as expecting `execute` to return 0.
|
|
||||||
def test(command, opts=nil)
|
|
||||||
# Disable error checking no matter what
|
|
||||||
opts = (opts || {}).merge(:error_check => false)
|
|
||||||
|
|
||||||
# Successful if the exit status is 0
|
|
||||||
execute(command, opts) == 0
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
|
@ -8,16 +8,22 @@ require 'vagrant/util/ansi_escape_code_remover'
|
||||||
require 'vagrant/util/file_mode'
|
require 'vagrant/util/file_mode'
|
||||||
require 'vagrant/util/platform'
|
require 'vagrant/util/platform'
|
||||||
require 'vagrant/util/retryable'
|
require 'vagrant/util/retryable'
|
||||||
|
require 'vagrant/util/ssh'
|
||||||
|
|
||||||
module Vagrant
|
module VagrantPlugins
|
||||||
module Communication
|
module CommunicatorSSH
|
||||||
# Provides communication with the VM via SSH.
|
# This class provides communication with the VM via SSH.
|
||||||
class SSH < Base
|
class Communicator < Vagrant.plugin("1", :communicator)
|
||||||
include Util::ANSIEscapeCodeRemover
|
include Util::ANSIEscapeCodeRemover
|
||||||
include Util::Retryable
|
include Util::Retryable
|
||||||
|
|
||||||
def initialize(vm)
|
def self.match?(machine)
|
||||||
@vm = vm
|
# All machines are currently expected to have SSH.
|
||||||
|
true
|
||||||
|
end
|
||||||
|
|
||||||
|
def initialize(machine)
|
||||||
|
@machine = machine
|
||||||
@logger = Log4r::Logger.new("vagrant::communication::ssh")
|
@logger = Log4r::Logger.new("vagrant::communication::ssh")
|
||||||
@connection = nil
|
@connection = nil
|
||||||
end
|
end
|
||||||
|
@ -31,7 +37,7 @@ module Vagrant
|
||||||
# If we reached this point then we successfully connected
|
# If we reached this point then we successfully connected
|
||||||
@logger.info("SSH is ready!")
|
@logger.info("SSH is ready!")
|
||||||
true
|
true
|
||||||
rescue Errors::VagrantError => e
|
rescue Vagrant::Errors::VagrantError => e
|
||||||
# We catch a `VagrantError` which would signal that something went
|
# We catch a `VagrantError` which would signal that something went
|
||||||
# wrong expectedly in the `connect`, which means we didn't connect.
|
# wrong expectedly in the `connect`, which means we didn't connect.
|
||||||
@logger.info("SSH not up: #{e.inspect}")
|
@logger.info("SSH not up: #{e.inspect}")
|
||||||
|
@ -41,7 +47,7 @@ module Vagrant
|
||||||
def execute(command, opts=nil, &block)
|
def execute(command, opts=nil, &block)
|
||||||
opts = {
|
opts = {
|
||||||
:error_check => true,
|
:error_check => true,
|
||||||
:error_class => Errors::VagrantError,
|
:error_class => Vagrant::Errors::VagrantError,
|
||||||
:error_key => :ssh_bad_exit_status,
|
:error_key => :ssh_bad_exit_status,
|
||||||
:command => command,
|
:command => command,
|
||||||
:sudo => false
|
:sudo => false
|
||||||
|
@ -93,7 +99,7 @@ module Vagrant
|
||||||
|
|
||||||
# Otherwise, it is a permission denied, so let's raise a proper
|
# Otherwise, it is a permission denied, so let's raise a proper
|
||||||
# exception
|
# exception
|
||||||
raise Errors::SCPPermissionDenied, :path => from.to_s
|
raise Vagrant::Errors::SCPPermissionDenied, :path => from.to_s
|
||||||
end
|
end
|
||||||
|
|
||||||
protected
|
protected
|
||||||
|
@ -120,7 +126,8 @@ module Vagrant
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
ssh_info = @vm.ssh.info
|
# XXX: We need to raise some exception if SSH is not ready
|
||||||
|
ssh_info = @machine.ssh_info
|
||||||
|
|
||||||
# Build the options we'll use to initiate the connection via Net::SSH
|
# Build the options we'll use to initiate the connection via Net::SSH
|
||||||
opts = {
|
opts = {
|
||||||
|
@ -134,38 +141,38 @@ module Vagrant
|
||||||
}
|
}
|
||||||
|
|
||||||
# Check that the private key permissions are valid
|
# Check that the private key permissions are valid
|
||||||
@vm.ssh.check_key_permissions(ssh_info[:private_key_path])
|
Vagrant::Util::SSH.check_key_permissions(ssh_info[:private_key_path])
|
||||||
|
|
||||||
# Connect to SSH, giving it a few tries
|
# Connect to SSH, giving it a few tries
|
||||||
connection = nil
|
connection = nil
|
||||||
begin
|
begin
|
||||||
exceptions = [Errno::ECONNREFUSED, Net::SSH::Disconnect, Timeout::Error]
|
exceptions = [Errno::ECONNREFUSED, Net::SSH::Disconnect, Timeout::Error]
|
||||||
connection = retryable(:tries => @vm.config.ssh.max_tries, :on => exceptions) do
|
connection = retryable(:tries => @machine.config.ssh.max_tries, :on => exceptions) do
|
||||||
Timeout.timeout(@vm.config.ssh.timeout) do
|
Timeout.timeout(@machine.config.ssh.timeout) do
|
||||||
@logger.info("Attempting to connect to SSH: #{ssh_info[:host]}:#{ssh_info[:port]}")
|
@logger.info("Attempting to connect to SSH: #{ssh_info[:host]}:#{ssh_info[:port]}")
|
||||||
Net::SSH.start(ssh_info[:host], ssh_info[:username], opts)
|
Net::SSH.start(ssh_info[:host], ssh_info[:username], opts)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
rescue Timeout::Error
|
rescue Timeout::Error
|
||||||
# This happens if we continued to timeout when attempting to connect.
|
# This happens if we continued to timeout when attempting to connect.
|
||||||
raise Errors::SSHConnectionTimeout
|
raise Vagrant::Errors::SSHConnectionTimeout
|
||||||
rescue Net::SSH::AuthenticationFailed
|
rescue Net::SSH::AuthenticationFailed
|
||||||
# This happens if authentication failed. We wrap the error in our
|
# This happens if authentication failed. We wrap the error in our
|
||||||
# own exception.
|
# own exception.
|
||||||
raise Errors::SSHAuthenticationFailed
|
raise Vagrant::Errors::SSHAuthenticationFailed
|
||||||
rescue Net::SSH::Disconnect
|
rescue Net::SSH::Disconnect
|
||||||
# This happens if the remote server unexpectedly closes the
|
# This happens if the remote server unexpectedly closes the
|
||||||
# connection. This is usually raised when SSH is running on the
|
# connection. This is usually raised when SSH is running on the
|
||||||
# other side but can't properly setup a connection. This is
|
# other side but can't properly setup a connection. This is
|
||||||
# usually a server-side issue.
|
# usually a server-side issue.
|
||||||
raise Errors::SSHDisconnected
|
raise Vagrant::Errors::SSHDisconnected
|
||||||
rescue Errno::ECONNREFUSED
|
rescue Errno::ECONNREFUSED
|
||||||
# This is raised if we failed to connect the max amount of times
|
# This is raised if we failed to connect the max amount of times
|
||||||
raise Errors::SSHConnectionRefused
|
raise Vagrant::Errors::SSHConnectionRefused
|
||||||
rescue NotImplementedError
|
rescue NotImplementedError
|
||||||
# This is raised if a private key type that Net-SSH doesn't support
|
# This is raised if a private key type that Net-SSH doesn't support
|
||||||
# is used. Show a nicer error.
|
# is used. Show a nicer error.
|
||||||
raise Errors::SSHKeyTypeNotSupported
|
raise Vagrant::Errors::SSHKeyTypeNotSupported
|
||||||
end
|
end
|
||||||
|
|
||||||
@connection = connection
|
@connection = connection
|
||||||
|
@ -187,7 +194,7 @@ module Vagrant
|
||||||
|
|
||||||
# Determine the shell to execute. If we are using `sudo` then we
|
# Determine the shell to execute. If we are using `sudo` then we
|
||||||
# need to wrap the shell in a `sudo` call.
|
# need to wrap the shell in a `sudo` call.
|
||||||
shell = @vm.config.ssh.shell
|
shell = @machine.config.ssh.shell
|
||||||
shell = "sudo -H #{shell}" if sudo
|
shell = "sudo -H #{shell}" if sudo
|
||||||
|
|
||||||
# Open the channel so we can execute or command
|
# Open the channel so we can execute or command
|
||||||
|
@ -248,7 +255,7 @@ module Vagrant
|
||||||
end
|
end
|
||||||
rescue Net::SCP::Error => e
|
rescue Net::SCP::Error => e
|
||||||
# If we get the exit code of 127, then this means SCP is unavailable.
|
# If we get the exit code of 127, then this means SCP is unavailable.
|
||||||
raise Errors::SCPUnavailable if e.message =~ /\(127\)/
|
raise Vagrant::Errors::SCPUnavailable if e.message =~ /\(127\)/
|
||||||
|
|
||||||
# Otherwise, just raise the error up
|
# Otherwise, just raise the error up
|
||||||
raise
|
raise
|
|
@ -0,0 +1,19 @@
|
||||||
|
require "vagrant"
|
||||||
|
|
||||||
|
module VagrantPlugins
|
||||||
|
module CommunicatorSSH
|
||||||
|
class Plugin < Vagrant.plugin("1")
|
||||||
|
name "ssh communiator"
|
||||||
|
description <<-DESC
|
||||||
|
This plugin allows Vagrant to communicate with remote machines using
|
||||||
|
SSH as the underlying protocol, powered internally by Ruby's
|
||||||
|
net-ssh library.
|
||||||
|
DESC
|
||||||
|
|
||||||
|
communicator("ssh") do
|
||||||
|
require File.expand_path("../communicator", __FILE__)
|
||||||
|
Communicator
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in New Issue