Created
July 10, 2015 00:06
-
-
Save inoccu/86a212eb7f2e8a33d574 to your computer and use it in GitHub Desktop.
Vagrant 1.7.2 on VirtualBox 5.0 (Windows)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
require "forwardable" | |
require "log4r" | |
require "vagrant/util/retryable" | |
require File.expand_path("../base", __FILE__) | |
module VagrantPlugins | |
module ProviderVirtualBox | |
module Driver | |
class Meta < Base | |
# This is raised if the VM is not found when initializing a driver | |
# with a UUID. | |
class VMNotFound < StandardError; end | |
# We use forwardable to do all our driver forwarding | |
extend Forwardable | |
# The UUID of the virtual machine we represent | |
attr_reader :uuid | |
# The version of virtualbox that is running. | |
attr_reader :version | |
include Vagrant::Util::Retryable | |
def initialize(uuid=nil) | |
# Setup the base | |
super() | |
@logger = Log4r::Logger.new("vagrant::provider::virtualbox::meta") | |
@uuid = uuid | |
# Read and assign the version of VirtualBox we know which | |
# specific driver to instantiate. | |
begin | |
@version = read_version || "" | |
rescue Vagrant::Errors::CommandUnavailable, | |
Vagrant::Errors::CommandUnavailableWindows | |
# This means that VirtualBox was not found, so we raise this | |
# error here. | |
raise Vagrant::Errors::VirtualBoxNotDetected | |
end | |
# Instantiate the proper version driver for VirtualBox | |
@logger.debug("Finding driver for VirtualBox version: #{@version}") | |
driver_map = { | |
"4.0" => Version_4_0, | |
"4.1" => Version_4_1, | |
"4.2" => Version_4_2, | |
"4.3" => Version_4_3, | |
"5.0" => Version_5_0 | |
} | |
if @version.start_with?("4.2.14") | |
# VirtualBox 4.2.14 just doesn't work with Vagrant, so show error | |
raise Vagrant::Errors::VirtualBoxBrokenVersion040214 | |
end | |
driver_klass = nil | |
driver_map.each do |key, klass| | |
if @version.start_with?(key) | |
driver_klass = klass | |
break | |
end | |
end | |
if !driver_klass | |
supported_versions = driver_map.keys.sort.join(", ") | |
raise Vagrant::Errors::VirtualBoxInvalidVersion, | |
supported_versions: supported_versions | |
end | |
@logger.info("Using VirtualBox driver: #{driver_klass}") | |
@driver = driver_klass.new(@uuid) | |
if @uuid | |
# Verify the VM exists, and if it doesn't, then don't worry | |
# about it (mark the UUID as nil) | |
raise VMNotFound if [email protected]_exists?(@uuid) | |
end | |
end | |
def_delegators :@driver, :clear_forwarded_ports, | |
:clear_shared_folders, | |
:create_dhcp_server, | |
:create_host_only_network, | |
:delete, | |
:delete_unused_host_only_networks, | |
:discard_saved_state, | |
:enable_adapters, | |
:execute_command, | |
:export, | |
:forward_ports, | |
:halt, | |
:import, | |
:read_forwarded_ports, | |
:read_bridged_interfaces, | |
:read_dhcp_servers, | |
:read_guest_additions_version, | |
:read_guest_ip, | |
:read_guest_property, | |
:read_host_only_interfaces, | |
:read_mac_address, | |
:read_mac_addresses, | |
:read_machine_folder, | |
:read_network_interfaces, | |
:read_state, | |
:read_used_ports, | |
:read_vms, | |
:remove_dhcp_server, | |
:resume, | |
:set_mac_address, | |
:set_name, | |
:share_folders, | |
:ssh_port, | |
:start, | |
:suspend, | |
:verify!, | |
:verify_image, | |
:vm_exists? | |
protected | |
# This returns the version of VirtualBox that is running. | |
# | |
# @return [String] | |
def read_version | |
# The version string is usually in one of the following formats: | |
# | |
# * 4.1.8r1234 | |
# * 4.1.8r1234_OSE | |
# * 4.1.8_MacPortsr1234 | |
# | |
# Below accounts for all of these. | |
# Note: We split this into multiple lines because apparently "".split("_") | |
# is [], so we have to check for an empty array in between. | |
output = "" | |
retryable(on: Vagrant::Errors::VirtualBoxVersionEmpty, tries: 3, sleep: 1) do | |
output = execute("--version") | |
if output =~ /vboxdrv kernel module is not loaded/ || | |
output =~ /VirtualBox kernel modules are not loaded/i | |
raise Vagrant::Errors::VirtualBoxKernelModuleNotLoaded | |
elsif output =~ /Please install/ | |
# Check for installation incomplete warnings, for example: | |
# "WARNING: The character device /dev/vboxdrv does not | |
# exist. Please install the virtualbox-ose-dkms package and | |
# the appropriate headers, most likely linux-headers-generic." | |
raise Vagrant::Errors::VirtualBoxInstallIncomplete | |
elsif output.chomp == "" | |
# This seems to happen on Windows for uncertain reasons. | |
# Raise an error otherwise the error is that they have an | |
# incompatible version of VirtualBox which isn't true. | |
raise Vagrant::Errors::VirtualBoxVersionEmpty | |
end | |
end | |
parts = output.split("_") | |
return nil if parts.empty? | |
parts[0].split("r")[0] | |
end | |
end | |
end | |
end | |
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
require "vagrant" | |
module VagrantPlugins | |
module ProviderVirtualBox | |
class Plugin < Vagrant.plugin("2") | |
name "VirtualBox provider" | |
description <<-EOF | |
The VirtualBox provider allows Vagrant to manage and control | |
VirtualBox-based virtual machines. | |
EOF | |
provider(:virtualbox, priority: 6) do | |
require File.expand_path("../provider", __FILE__) | |
Provider | |
end | |
config(:virtualbox, :provider) do | |
require File.expand_path("../config", __FILE__) | |
Config | |
end | |
synced_folder(:virtualbox) do | |
require File.expand_path("../synced_folder", __FILE__) | |
SyncedFolder | |
end | |
provider_capability(:virtualbox, :forwarded_ports) do | |
require_relative "cap" | |
Cap | |
end | |
provider_capability(:virtualbox, :nic_mac_addresses) do | |
require_relative "cap" | |
Cap | |
end | |
end | |
autoload :Action, File.expand_path("../action", __FILE__) | |
# Drop some autoloads in here to optimize the performance of loading | |
# our drivers only when they are needed. | |
module Driver | |
autoload :Meta, File.expand_path("../driver/meta", __FILE__) | |
autoload :Version_4_0, File.expand_path("../driver/version_4_0", __FILE__) | |
autoload :Version_4_1, File.expand_path("../driver/version_4_1", __FILE__) | |
autoload :Version_4_2, File.expand_path("../driver/version_4_2", __FILE__) | |
autoload :Version_4_3, File.expand_path("../driver/version_4_3", __FILE__) | |
autoload :Version_5_0, File.expand_path("../driver/version_5_0", __FILE__) | |
end | |
module Model | |
autoload :ForwardedPort, File.expand_path("../model/forwarded_port", __FILE__) | |
end | |
module Util | |
autoload :CompileForwardedPorts, File.expand_path("../util/compile_forwarded_ports", __FILE__) | |
end | |
end | |
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
require 'log4r' | |
require "vagrant/util/platform" | |
require File.expand_path("../base", __FILE__) | |
module VagrantPlugins | |
module ProviderVirtualBox | |
module Driver | |
# Driver for VirtualBox 5.0.x | |
class Version_5_0 < Base | |
def initialize(uuid) | |
super() | |
@logger = Log4r::Logger.new("vagrant::provider::virtualbox_5_0") | |
@uuid = uuid | |
end | |
def clear_forwarded_ports | |
args = [] | |
read_forwarded_ports(@uuid).each do |nic, name, _, _| | |
args.concat(["--natpf#{nic}", "delete", name]) | |
end | |
execute("modifyvm", @uuid, *args) if !args.empty? | |
end | |
def clear_shared_folders | |
info = execute("showvminfo", @uuid, "--machinereadable", retryable: true) | |
info.split("\n").each do |line| | |
if line =~ /^SharedFolderNameMachineMapping\d+="(.+?)"$/ | |
execute("sharedfolder", "remove", @uuid, "--name", $1.to_s) | |
end | |
end | |
end | |
def create_dhcp_server(network, options) | |
execute("dhcpserver", "add", "--ifname", network, | |
"--ip", options[:dhcp_ip], | |
"--netmask", options[:netmask], | |
"--lowerip", options[:dhcp_lower], | |
"--upperip", options[:dhcp_upper], | |
"--enable") | |
end | |
def create_host_only_network(options) | |
# Create the interface | |
execute("hostonlyif", "create") =~ /^Interface '(.+?)' was successfully created$/ | |
name = $1.to_s | |
# Configure it | |
execute("hostonlyif", "ipconfig", name, | |
"--ip", options[:adapter_ip], | |
"--netmask", options[:netmask]) | |
# Return the details | |
return { | |
name: name, | |
ip: options[:adapter_ip], | |
netmask: options[:netmask], | |
dhcp: nil | |
} | |
end | |
def delete | |
execute("unregistervm", @uuid, "--delete") | |
end | |
def delete_unused_host_only_networks | |
networks = [] | |
execute("list", "hostonlyifs", retryable: true).split("\n").each do |line| | |
networks << $1.to_s if line =~ /^Name:\s+(.+?)$/ | |
end | |
execute("list", "vms", retryable: true).split("\n").each do |line| | |
if line =~ /^".+?"\s+\{(.+?)\}$/ | |
info = execute("showvminfo", $1.to_s, "--machinereadable", retryable: true) | |
info.split("\n").each do |inner_line| | |
if inner_line =~ /^hostonlyadapter\d+="(.+?)"$/ | |
networks.delete($1.to_s) | |
end | |
end | |
end | |
end | |
networks.each do |name| | |
# First try to remove any DHCP servers attached. We use `raw` because | |
# it is okay if this fails. It usually means that a DHCP server was | |
# never attached. | |
raw("dhcpserver", "remove", "--ifname", name) | |
# Delete the actual host only network interface. | |
execute("hostonlyif", "remove", name) | |
end | |
end | |
def discard_saved_state | |
execute("discardstate", @uuid) | |
end | |
def enable_adapters(adapters) | |
args = [] | |
adapters.each do |adapter| | |
args.concat(["--nic#{adapter[:adapter]}", adapter[:type].to_s]) | |
if adapter[:bridge] | |
args.concat(["--bridgeadapter#{adapter[:adapter]}", | |
adapter[:bridge], "--cableconnected#{adapter[:adapter]}", "on"]) | |
end | |
if adapter[:hostonly] | |
args.concat(["--hostonlyadapter#{adapter[:adapter]}", | |
adapter[:hostonly], "--cableconnected#{adapter[:adapter]}", "on"]) | |
end | |
if adapter[:intnet] | |
args.concat(["--intnet#{adapter[:adapter]}", | |
adapter[:intnet], "--cableconnected#{adapter[:adapter]}", "on"]) | |
end | |
if adapter[:mac_address] | |
args.concat(["--macaddress#{adapter[:adapter]}", | |
adapter[:mac_address]]) | |
end | |
if adapter[:nic_type] | |
args.concat(["--nictype#{adapter[:adapter]}", adapter[:nic_type].to_s]) | |
end | |
end | |
execute("modifyvm", @uuid, *args) | |
end | |
def execute_command(command) | |
execute(*command) | |
end | |
def export(path) | |
execute("export", @uuid, "--output", path.to_s) | |
end | |
def forward_ports(ports) | |
args = [] | |
ports.each do |options| | |
pf_builder = [options[:name], | |
options[:protocol] || "tcp", | |
options[:hostip] || "", | |
options[:hostport], | |
options[:guestip] || "", | |
options[:guestport]] | |
args.concat(["--natpf#{options[:adapter] || 1}", | |
pf_builder.join(",")]) | |
end | |
execute("modifyvm", @uuid, *args) if !args.empty? | |
end | |
def halt | |
execute("controlvm", @uuid, "poweroff") | |
end | |
def import(ovf) | |
ovf = Vagrant::Util::Platform.cygwin_windows_path(ovf) | |
output = "" | |
total = "" | |
last = 0 | |
# Dry-run the import to get the suggested name and path | |
@logger.debug("Doing dry-run import to determine parallel-safe name...") | |
output = execute("import", "-n", ovf) | |
result = /Suggested VM name "(.+?)"/.match(output) | |
if !result | |
raise Vagrant::Errors::VirtualBoxNoName, output: output | |
end | |
suggested_name = result[1].to_s | |
# Append millisecond plus a random to the path in case we're | |
# importing the same box elsewhere. | |
specified_name = "#{suggested_name}_#{(Time.now.to_f * 1000.0).to_i}_#{rand(100000)}" | |
@logger.debug("-- Parallel safe name: #{specified_name}") | |
# Build the specified name param list | |
name_params = [ | |
"--vsys", "0", | |
"--vmname", specified_name, | |
] | |
# Extract the disks list and build the disk target params | |
disk_params = [] | |
disks = output.scan(/(\d+): Hard disk image: source image=.+, target path=(.+),/) | |
disks.each do |unit_num, path| | |
disk_params << "--vsys" | |
disk_params << "0" | |
disk_params << "--unit" | |
disk_params << unit_num | |
disk_params << "--disk" | |
if Vagrant::Util::Platform.windows? | |
# we use the block form of sub here to ensure that if the specified_name happens to end with a number (which is fairly likely) then | |
# we won't end up having the character sequence of a \ followed by a number be interpreted as a back reference. For example, if | |
# specified_name were "abc123", then "\\abc123\\".reverse would be "\\321cba\\", and the \3 would be treated as a back reference by the sub | |
disk_params << path.reverse.sub("\\#{suggested_name}\\".reverse) { "\\#{specified_name}\\".reverse }.reverse # Replace only last occurrence | |
else | |
disk_params << path.reverse.sub("/#{suggested_name}/".reverse, "/#{specified_name}/".reverse).reverse # Replace only last occurrence | |
end | |
end | |
execute("import", ovf , *name_params, *disk_params) do |type, data| | |
if type == :stdout | |
# Keep track of the stdout so that we can get the VM name | |
output << data | |
elsif type == :stderr | |
# Append the data so we can see the full view | |
total << data.gsub("\r", "") | |
# Break up the lines. We can't get the progress until we see an "OK" | |
lines = total.split("\n") | |
if lines.include?("OK.") | |
# The progress of the import will be in the last line. Do a greedy | |
# regular expression to find what we're looking for. | |
match = /.+(\d{2})%/.match(lines.last) | |
if match | |
current = match[1].to_i | |
if current > last | |
last = current | |
yield current if block_given? | |
end | |
end | |
end | |
end | |
end | |
output = execute("list", "vms", retryable: true) | |
match = /^"#{Regexp.escape(specified_name)}" \{(.+?)\}$/.match(output) | |
return match[1].to_s if match | |
nil | |
end | |
def max_network_adapters | |
36 | |
end | |
def read_forwarded_ports(uuid=nil, active_only=false) | |
uuid ||= @uuid | |
@logger.debug("read_forward_ports: uuid=#{uuid} active_only=#{active_only}") | |
results = [] | |
current_nic = nil | |
info = execute("showvminfo", uuid, "--machinereadable", retryable: true) | |
info.split("\n").each do |line| | |
# This is how we find the nic that a FP is attached to, | |
# since this comes first. | |
current_nic = $1.to_i if line =~ /^nic(\d+)=".+?"$/ | |
# If we care about active VMs only, then we check the state | |
# to verify the VM is running. | |
if active_only && line =~ /^VMState="(.+?)"$/ && $1.to_s != "running" | |
return [] | |
end | |
# Parse out the forwarded port information | |
if line =~ /^Forwarding.+?="(.+?),.+?,.*?,(.+?),.*?,(.+?)"$/ | |
result = [current_nic, $1.to_s, $2.to_i, $3.to_i] | |
@logger.debug(" - #{result.inspect}") | |
results << result | |
end | |
end | |
results | |
end | |
def read_bridged_interfaces | |
execute("list", "bridgedifs").split("\n\n").collect do |block| | |
info = {} | |
block.split("\n").each do |line| | |
if line =~ /^Name:\s+(.+?)$/ | |
info[:name] = $1.to_s | |
elsif line =~ /^IPAddress:\s+(.+?)$/ | |
info[:ip] = $1.to_s | |
elsif line =~ /^NetworkMask:\s+(.+?)$/ | |
info[:netmask] = $1.to_s | |
elsif line =~ /^Status:\s+(.+?)$/ | |
info[:status] = $1.to_s | |
end | |
end | |
# Return the info to build up the results | |
info | |
end | |
end | |
def read_dhcp_servers | |
execute("list", "dhcpservers", retryable: true).split("\n\n").collect do |block| | |
info = {} | |
block.split("\n").each do |line| | |
if network = line[/^NetworkName:\s+HostInterfaceNetworking-(.+?)$/, 1] | |
info[:network] = network | |
info[:network_name] = "HostInterfaceNetworking-#{network}" | |
elsif ip = line[/^IP:\s+(.+?)$/, 1] | |
info[:ip] = ip | |
elsif lower = line[/^lowerIPAddress:\s+(.+?)$/, 1] | |
info[:lower] = lower | |
elsif upper = line[/^upperIPAddress:\s+(.+?)$/, 1] | |
info[:upper] = upper | |
end | |
end | |
info | |
end | |
end | |
def read_guest_additions_version | |
output = execute("guestproperty", "get", @uuid, "/VirtualBox/GuestAdd/Version", | |
retryable: true) | |
if output =~ /^Value: (.+?)$/ | |
# Split the version by _ since some distro versions modify it | |
# to look like this: 4.1.2_ubuntu, and the distro part isn't | |
# too important. | |
value = $1.to_s | |
return value.split("_").first | |
end | |
# If we can't get the guest additions version by guest property, try | |
# to get it from the VM info itself. | |
info = execute("showvminfo", @uuid, "--machinereadable", retryable: true) | |
info.split("\n").each do |line| | |
return $1.to_s if line =~ /^GuestAdditionsVersion="(.+?)"$/ | |
end | |
return nil | |
end | |
def read_guest_ip(adapter_number) | |
ip = read_guest_property("/VirtualBox/GuestInfo/Net/#{adapter_number}/V4/IP") | |
if !valid_ip_address?(ip) | |
raise Vagrant::Errors::VirtualBoxGuestPropertyNotFound, | |
guest_property: "/VirtualBox/GuestInfo/Net/#{adapter_number}/V4/IP" | |
end | |
return ip | |
end | |
def read_guest_property(property) | |
output = execute("guestproperty", "get", @uuid, property) | |
if output =~ /^Value: (.+?)$/ | |
$1.to_s | |
else | |
raise Vagrant::Errors::VirtualBoxGuestPropertyNotFound, guest_property: property | |
end | |
end | |
def read_host_only_interfaces | |
execute("list", "hostonlyifs", retryable: true).split("\n\n").collect do |block| | |
info = {} | |
block.split("\n").each do |line| | |
if line =~ /^Name:\s+(.+?)$/ | |
info[:name] = $1.to_s | |
elsif line =~ /^IPAddress:\s+(.+?)$/ | |
info[:ip] = $1.to_s | |
elsif line =~ /^NetworkMask:\s+(.+?)$/ | |
info[:netmask] = $1.to_s | |
elsif line =~ /^Status:\s+(.+?)$/ | |
info[:status] = $1.to_s | |
end | |
end | |
info | |
end | |
end | |
def read_mac_address | |
info = execute("showvminfo", @uuid, "--machinereadable", retryable: true) | |
info.split("\n").each do |line| | |
return $1.to_s if line =~ /^macaddress1="(.+?)"$/ | |
end | |
nil | |
end | |
def read_mac_addresses | |
macs = {} | |
info = execute("showvminfo", @uuid, "--machinereadable", retryable: true) | |
info.split("\n").each do |line| | |
if matcher = /^macaddress(\d+)="(.+?)"$/.match(line) | |
adapter = matcher[1].to_i | |
mac = matcher[2].to_s | |
macs[adapter] = mac | |
end | |
end | |
macs | |
end | |
def read_machine_folder | |
execute("list", "systemproperties", retryable: true).split("\n").each do |line| | |
if line =~ /^Default machine folder:\s+(.+?)$/i | |
return $1.to_s | |
end | |
end | |
nil | |
end | |
def read_network_interfaces | |
nics = {} | |
info = execute("showvminfo", @uuid, "--machinereadable", retryable: true) | |
info.split("\n").each do |line| | |
if line =~ /^nic(\d+)="(.+?)"$/ | |
adapter = $1.to_i | |
type = $2.to_sym | |
nics[adapter] ||= {} | |
nics[adapter][:type] = type | |
elsif line =~ /^hostonlyadapter(\d+)="(.+?)"$/ | |
adapter = $1.to_i | |
network = $2.to_s | |
nics[adapter] ||= {} | |
nics[adapter][:hostonly] = network | |
elsif line =~ /^bridgeadapter(\d+)="(.+?)"$/ | |
adapter = $1.to_i | |
network = $2.to_s | |
nics[adapter] ||= {} | |
nics[adapter][:bridge] = network | |
end | |
end | |
nics | |
end | |
def read_state | |
output = execute("showvminfo", @uuid, "--machinereadable", retryable: true) | |
if output =~ /^name="<inaccessible>"$/ | |
return :inaccessible | |
elsif output =~ /^VMState="(.+?)"$/ | |
return $1.to_sym | |
end | |
nil | |
end | |
def read_used_ports | |
ports = [] | |
execute("list", "vms", retryable: true).split("\n").each do |line| | |
if line =~ /^".+?" \{(.+?)\}$/ | |
uuid = $1.to_s | |
# Ignore our own used ports | |
next if uuid == @uuid | |
read_forwarded_ports(uuid, true).each do |_, _, hostport, _| | |
ports << hostport | |
end | |
end | |
end | |
ports | |
end | |
def read_vms | |
results = {} | |
execute("list", "vms", retryable: true).split("\n").each do |line| | |
if line =~ /^"(.+?)" \{(.+?)\}$/ | |
results[$1.to_s] = $2.to_s | |
end | |
end | |
results | |
end | |
def remove_dhcp_server(network_name) | |
execute("dhcpserver", "remove", "--netname", network_name) | |
end | |
def set_mac_address(mac) | |
execute("modifyvm", @uuid, "--macaddress1", mac) | |
end | |
def set_name(name) | |
execute("modifyvm", @uuid, "--name", name, retryable: true) | |
rescue Vagrant::Errors::VBoxManageError => e | |
raise if !e.extra_data[:stderr].include?("VERR_ALREADY_EXISTS") | |
# We got VERR_ALREADY_EXISTS. This means that we're renaming to | |
# a VM name that already exists. Raise a custom error. | |
raise Vagrant::Errors::VirtualBoxNameExists, | |
stderr: e.extra_data[:stderr] | |
end | |
def share_folders(folders) | |
folders.each do |folder| | |
args = ["--name", | |
folder[:name], | |
"--hostpath", | |
folder[:hostpath]] | |
args << "--transient" if folder.key?(:transient) && folder[:transient] | |
# Enable symlinks on the shared folder | |
execute("setextradata", @uuid, "VBoxInternal2/SharedFoldersEnableSymlinksCreate/#{folder[:name]}", "1") | |
# Add the shared folder | |
execute("sharedfolder", "add", @uuid, *args) | |
end | |
end | |
def ssh_port(expected_port) | |
@logger.debug("Searching for SSH port: #{expected_port.inspect}") | |
# Look for the forwarded port only by comparing the guest port | |
read_forwarded_ports.each do |_, _, hostport, guestport| | |
return hostport if guestport == expected_port | |
end | |
nil | |
end | |
def resume | |
@logger.debug("Resuming paused VM...") | |
execute("controlvm", @uuid, "resume") | |
end | |
def start(mode) | |
command = ["startvm", @uuid, "--type", mode.to_s] | |
r = raw(*command) | |
if r.exit_code == 0 || r.stdout =~ /VM ".+?" has been successfully started/ | |
# Some systems return an exit code 1 for some reason. For that | |
# we depend on the output. | |
return true | |
end | |
# If we reached this point then it didn't work out. | |
raise Vagrant::Errors::VBoxManageError, | |
command: command.inspect, | |
stderr: r.stderr | |
end | |
def suspend | |
execute("controlvm", @uuid, "savestate") | |
end | |
def unshare_folders(names) | |
names.each do |name| | |
begin | |
execute( | |
"sharedfolder", "remove", @uuid, | |
"--name", name, | |
"--transient") | |
execute( | |
"setextradata", @uuid, | |
"VBoxInternal2/SharedFoldersEnableSymlinksCreate/#{name}") | |
rescue Vagrant::Errors::VBoxManageError => e | |
if e.extra_data[:stderr].include?("VBOX_E_FILE_ERROR") | |
# The folder doesn't exist. ignore. | |
else | |
raise | |
end | |
end | |
end | |
end | |
def verify! | |
# This command sometimes fails if kernel drivers aren't properly loaded | |
# so we just run the command and verify that it succeeded. | |
execute("list", "hostonlyifs", retryable: true) | |
end | |
def verify_image(path) | |
r = raw("import", path.to_s, "--dry-run") | |
return r.exit_code == 0 | |
end | |
def vm_exists?(uuid) | |
5.times do |i| | |
result = raw("showvminfo", uuid) | |
return true if result.exit_code == 0 | |
# GH-2479: Sometimes this happens. In this case, retry. If | |
# we don't see this text, the VM really probably doesn't exist. | |
return false if !result.stderr.include?("CO_E_SERVER_EXEC_FAILURE") | |
# Sleep a bit though to give VirtualBox time to fix itself | |
sleep 2 | |
end | |
# If we reach this point, it means that we consistently got the | |
# failure, do a standard vboxmanage now. This will raise an | |
# exception if it fails again. | |
execute("showvminfo", uuid) | |
return true | |
end | |
protected | |
def valid_ip_address?(ip) | |
# Filter out invalid IP addresses | |
# GH-4658 VirtualBox can report an IP address of 0.0.0.0 for FreeBSD guests. | |
if ip == "0.0.0.0" | |
return false | |
else | |
return true | |
end | |
end | |
end | |
end | |
end | |
end |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment