2012-11-07 05:28:44 +00:00
|
|
|
require "pathname"
|
|
|
|
|
|
|
|
require "vagrant"
|
2013-01-18 20:56:19 +00:00
|
|
|
require "vagrant/config/v2/util"
|
2012-11-07 05:28:44 +00:00
|
|
|
|
2012-11-08 04:38:41 +00:00
|
|
|
require File.expand_path("../vm_provider", __FILE__)
|
2012-11-07 05:28:44 +00:00
|
|
|
require File.expand_path("../vm_provisioner", __FILE__)
|
|
|
|
require File.expand_path("../vm_subvm", __FILE__)
|
|
|
|
|
|
|
|
module VagrantPlugins
|
|
|
|
module Kernel_V2
|
|
|
|
class VMConfig < Vagrant.plugin("2", :config)
|
|
|
|
DEFAULT_VM_NAME = :default
|
|
|
|
|
|
|
|
attr_accessor :base_mac
|
2012-11-15 03:59:18 +00:00
|
|
|
attr_accessor :box
|
|
|
|
attr_accessor :box_url
|
2013-01-22 19:37:49 +00:00
|
|
|
attr_accessor :graceful_halt_retry_count
|
|
|
|
attr_accessor :graceful_halt_retry_interval
|
2012-11-15 03:59:18 +00:00
|
|
|
attr_accessor :guest
|
2012-11-07 05:28:44 +00:00
|
|
|
attr_accessor :host_name
|
2013-01-11 22:44:27 +00:00
|
|
|
attr_accessor :usable_port_range
|
2012-11-07 05:28:44 +00:00
|
|
|
attr_reader :forwarded_ports
|
2013-01-23 17:21:49 +00:00
|
|
|
attr_reader :synced_folders
|
2012-11-07 05:28:44 +00:00
|
|
|
attr_reader :networks
|
2012-11-08 04:38:41 +00:00
|
|
|
attr_reader :providers
|
2012-11-07 05:28:44 +00:00
|
|
|
attr_reader :provisioners
|
|
|
|
|
|
|
|
def initialize
|
2013-01-22 19:37:49 +00:00
|
|
|
@forwarded_ports = []
|
|
|
|
@graceful_halt_retry_count = UNSET_VALUE
|
|
|
|
@graceful_halt_retry_interval = UNSET_VALUE
|
2013-01-23 17:21:49 +00:00
|
|
|
@synced_folders = {}
|
2013-01-22 19:37:49 +00:00
|
|
|
@networks = []
|
|
|
|
@provisioners = []
|
2012-12-23 19:02:51 +00:00
|
|
|
|
|
|
|
# The providers hash defaults any key to a provider object
|
|
|
|
@providers = Hash.new do |hash, key|
|
2013-02-05 20:08:17 +00:00
|
|
|
hash[key] = VagrantConfigProvider.new(key)
|
2012-12-23 19:02:51 +00:00
|
|
|
end
|
2012-11-07 05:28:44 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
# Custom merge method since some keys here are merged differently.
|
|
|
|
def merge(other)
|
|
|
|
result = super
|
|
|
|
result.instance_variable_set(:@forwarded_ports, @forwarded_ports + other.forwarded_ports)
|
2013-01-23 17:21:49 +00:00
|
|
|
result.instance_variable_set(:@synced_folders, @synced_folders.merge(other.synced_folders))
|
2012-11-07 05:28:44 +00:00
|
|
|
result.instance_variable_set(:@networks, @networks + other.networks)
|
|
|
|
result.instance_variable_set(:@provisioners, @provisioners + other.provisioners)
|
|
|
|
result
|
|
|
|
end
|
|
|
|
|
2013-01-23 17:21:49 +00:00
|
|
|
# Defines a synced folder pair. This pair of folders will be synced
|
|
|
|
# to/from the machine. Note that if the machine you're using doesn't
|
|
|
|
# support multi-directional syncing (perhaps an rsync backed synced
|
|
|
|
# folder) then the host is always synced to the guest but guest data
|
|
|
|
# may not be synced back to the host.
|
|
|
|
#
|
|
|
|
# @param [String] hostpath Path to the host folder to share. If this
|
|
|
|
# is a relative path, it is relative to the location of the
|
|
|
|
# Vagrantfile.
|
|
|
|
# @param [String] guestpath Path on the guest to mount the shared
|
|
|
|
# folder.
|
|
|
|
# @param [Hash] options Additional options.
|
|
|
|
def synced_folder(hostpath, guestpath, options=nil)
|
|
|
|
options ||= {}
|
2013-01-28 23:57:13 +00:00
|
|
|
options[:id] ||= guestpath
|
2013-01-23 17:42:19 +00:00
|
|
|
options[:guestpath] = guestpath
|
|
|
|
options[:hostpath] = hostpath
|
2013-01-23 17:21:49 +00:00
|
|
|
|
2013-01-23 17:42:19 +00:00
|
|
|
@synced_folders[options[:id]] = options
|
2012-11-07 05:28:44 +00:00
|
|
|
end
|
|
|
|
|
2013-01-05 01:48:54 +00:00
|
|
|
# Define a way to access the machine via a network. This exposes a
|
|
|
|
# high-level abstraction for networking that may not directly map
|
|
|
|
# 1-to-1 for every provider. For example, AWS has no equivalent to
|
|
|
|
# "port forwarding." But most providers will attempt to implement this
|
|
|
|
# in a way that behaves similarly.
|
|
|
|
#
|
|
|
|
# `type` can be one of:
|
|
|
|
#
|
|
|
|
# * `:forwarded_port` - A port that is accessible via localhost
|
|
|
|
# that forwards into the machine.
|
|
|
|
# * `:private_network` - The machine gets an IP that is not directly
|
|
|
|
# publicly accessible, but ideally accessible from this machine.
|
|
|
|
# * `:public_network` - The machine gets an IP on a shared network.
|
|
|
|
#
|
|
|
|
# @param [Symbol] type Type of network
|
2012-11-07 05:28:44 +00:00
|
|
|
def network(type, *args)
|
|
|
|
@networks << [type, args]
|
|
|
|
end
|
|
|
|
|
2012-11-08 04:38:41 +00:00
|
|
|
# Configures a provider for this VM.
|
|
|
|
#
|
|
|
|
# @param [Symbol] name The name of the provider.
|
2012-11-19 06:55:30 +00:00
|
|
|
def provider(name, &block)
|
2013-02-05 20:08:17 +00:00
|
|
|
@providers[name].add_config_block(block) if block_given?
|
2012-11-08 04:38:41 +00:00
|
|
|
end
|
|
|
|
|
2012-11-07 05:28:44 +00:00
|
|
|
def provision(name, options=nil, &block)
|
|
|
|
@provisioners << VagrantConfigProvisioner.new(name, options, &block)
|
|
|
|
end
|
|
|
|
|
|
|
|
def defined_vms
|
|
|
|
@defined_vms ||= {}
|
|
|
|
end
|
|
|
|
|
|
|
|
# This returns the keys of the sub-vms in the order they were
|
|
|
|
# defined.
|
|
|
|
def defined_vm_keys
|
|
|
|
@defined_vm_keys ||= []
|
|
|
|
end
|
|
|
|
|
|
|
|
def define(name, options=nil, &block)
|
|
|
|
name = name.to_sym
|
|
|
|
options ||= {}
|
2013-01-31 03:24:09 +00:00
|
|
|
options[:config_version] ||= "2"
|
2012-11-07 05:28:44 +00:00
|
|
|
|
|
|
|
# Add the name to the array of VM keys. This array is used to
|
|
|
|
# preserve the order in which VMs are defined.
|
|
|
|
defined_vm_keys << name
|
|
|
|
|
|
|
|
# Add the SubVM to the hash of defined VMs
|
|
|
|
if !defined_vms[name]
|
|
|
|
defined_vms[name] ||= VagrantConfigSubVM.new
|
|
|
|
end
|
|
|
|
|
|
|
|
defined_vms[name].options.merge!(options)
|
2013-01-31 03:24:09 +00:00
|
|
|
defined_vms[name].config_procs << [options[:config_version], block] if block
|
2012-11-07 05:28:44 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def finalize!
|
|
|
|
# If we haven't defined a single VM, then we need to define a
|
|
|
|
# default VM which just inherits the rest of the configuration.
|
|
|
|
define(DEFAULT_VM_NAME) if defined_vm_keys.empty?
|
2013-02-05 20:08:17 +00:00
|
|
|
|
|
|
|
# Compile all the provider configurations
|
|
|
|
@providers.each do |name, config|
|
|
|
|
config.finalize!
|
|
|
|
end
|
2012-11-07 05:28:44 +00:00
|
|
|
end
|
|
|
|
|
2013-01-18 20:42:49 +00:00
|
|
|
def validate(machine)
|
2013-01-18 20:33:37 +00:00
|
|
|
errors = []
|
|
|
|
errors << I18n.t("vagrant.config.vm.box_missing") if !box
|
|
|
|
errors << I18n.t("vagrant.config.vm.box_not_found", :name => box) if \
|
2013-01-18 20:42:49 +00:00
|
|
|
box && !box_url && !machine.box
|
2012-11-07 05:28:44 +00:00
|
|
|
|
2013-01-23 17:42:19 +00:00
|
|
|
@synced_folders.each do |id, options|
|
|
|
|
hostpath = Pathname.new(options[:hostpath]).expand_path(machine.env.root_path)
|
2012-11-07 05:28:44 +00:00
|
|
|
|
|
|
|
if !hostpath.directory? && !options[:create]
|
2013-01-18 20:33:37 +00:00
|
|
|
errors << I18n.t("vagrant.config.vm.shared_folder_hostpath_missing",
|
2013-01-23 17:42:19 +00:00
|
|
|
:path => options[:hostpath])
|
2012-11-07 05:28:44 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
if options[:nfs] && (options[:owner] || options[:group])
|
|
|
|
# Owner/group don't work with NFS
|
2013-01-18 20:33:37 +00:00
|
|
|
errors << I18n.t("vagrant.config.vm.shared_folder_nfs_owner_group",
|
2013-01-23 17:42:19 +00:00
|
|
|
:path => options[:hostpath])
|
2012-11-07 05:28:44 +00:00
|
|
|
end
|
|
|
|
end
|
2013-01-18 20:33:37 +00:00
|
|
|
|
2013-01-18 20:56:19 +00:00
|
|
|
# We're done with VM level errors so prepare the section
|
|
|
|
errors = { "vm" => errors }
|
|
|
|
|
2013-01-18 21:33:02 +00:00
|
|
|
# Validate only the _active_ provider
|
|
|
|
if machine.provider_config
|
|
|
|
provider_errors = machine.provider_config.validate(machine)
|
2013-01-18 21:36:12 +00:00
|
|
|
if provider_errors
|
|
|
|
errors = Vagrant::Config::V2::Util.merge_errors(errors, provider_errors)
|
|
|
|
end
|
2013-01-18 20:56:19 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
# Validate provisioners
|
|
|
|
@provisioners.each do |vm_provisioner|
|
|
|
|
if vm_provisioner.config
|
|
|
|
provisioner_errors = vm_provisioner.config.validate(machine)
|
2013-01-18 21:36:12 +00:00
|
|
|
if provisioner_errors
|
|
|
|
errors = Vagrant::Config::V2::Util.merge_errors(errors, provisioner_errors)
|
|
|
|
end
|
2013-01-18 20:56:19 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
errors
|
2012-11-07 05:28:44 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|