540 lines
19 KiB
Ruby
540 lines
19 KiB
Ruby
require 'fileutils'
|
|
require 'pathname'
|
|
require 'set'
|
|
|
|
require 'log4r'
|
|
|
|
require 'vagrant/util/file_mode'
|
|
require 'vagrant/util/platform'
|
|
|
|
module Vagrant
|
|
# Represents a single Vagrant environment. A "Vagrant environment" is
|
|
# defined as basically a folder with a "Vagrantfile." This class allows
|
|
# access to the VMs, CLI, etc. all in the scope of this environment.
|
|
class Environment
|
|
DEFAULT_HOME = "~/.vagrant.d"
|
|
DEFAULT_LOCAL_DATA = ".vagrant"
|
|
DEFAULT_RC = "~/.vagrantrc"
|
|
|
|
# This is the global config, comprised of loading configuration from
|
|
# the default, home, and root Vagrantfiles. This configuration is only
|
|
# really useful for reading the list of virtual machines, since each
|
|
# individual VM can override _most_ settings.
|
|
attr_reader :config_global
|
|
|
|
# The `cwd` that this environment represents
|
|
attr_reader :cwd
|
|
|
|
# The valid name for a Vagrantfile for this environment.
|
|
attr_reader :vagrantfile_name
|
|
|
|
# The {UI} object to communicate with the outside world.
|
|
attr_reader :ui
|
|
|
|
# The directory to the "home" folder that Vagrant will use to store
|
|
# global state.
|
|
attr_reader :home_path
|
|
|
|
# The directory to the directory where local, environment-specific
|
|
# data is stored.
|
|
attr_reader :local_data_path
|
|
|
|
# The directory where temporary files for Vagrant go.
|
|
attr_reader :tmp_path
|
|
|
|
# The directory where boxes are stored.
|
|
attr_reader :boxes_path
|
|
|
|
# The path where the plugins are stored (gems)
|
|
attr_reader :gems_path
|
|
|
|
# The path to the default private key
|
|
attr_reader :default_private_key_path
|
|
|
|
# This is a set of the vagrantrc files already loaded so that they
|
|
# are only loaded once.
|
|
@@loaded_rc = Set.new
|
|
|
|
# Initializes a new environment with the given options. The options
|
|
# is a hash where the main available key is `cwd`, which defines where
|
|
# the environment represents. There are other options available but
|
|
# they shouldn't be used in general. If `cwd` is nil, then it defaults
|
|
# to the `Dir.pwd` (which is the cwd of the executing process).
|
|
def initialize(opts=nil)
|
|
opts = {
|
|
:cwd => nil,
|
|
:home_path => nil,
|
|
:local_data_path => nil,
|
|
:lock_path => nil,
|
|
:ui_class => nil,
|
|
:vagrantfile_name => nil
|
|
}.merge(opts || {})
|
|
|
|
# Set the default working directory to look for the vagrantfile
|
|
opts[:cwd] ||= ENV["VAGRANT_CWD"] if ENV.has_key?("VAGRANT_CWD")
|
|
opts[:cwd] ||= Dir.pwd
|
|
opts[:cwd] = Pathname.new(opts[:cwd])
|
|
raise Errors::EnvironmentNonExistentCWD if !opts[:cwd].directory?
|
|
|
|
# Set the default ui class
|
|
opts[:ui_class] ||= UI::Silent
|
|
|
|
# Set the Vagrantfile name up. We append "Vagrantfile" and "vagrantfile" so that
|
|
# those continue to work as well, but anything custom will take precedence.
|
|
opts[:vagrantfile_name] ||= []
|
|
opts[:vagrantfile_name] = [opts[:vagrantfile_name]] if !opts[:vagrantfile_name].is_a?(Array)
|
|
opts[:vagrantfile_name] += ["Vagrantfile", "vagrantfile"]
|
|
|
|
# Set instance variables for all the configuration parameters.
|
|
@cwd = opts[:cwd]
|
|
@home_path = opts[:home_path]
|
|
@lock_path = opts[:lock_path]
|
|
@vagrantfile_name = opts[:vagrantfile_name]
|
|
@ui = opts[:ui_class].new("vagrant")
|
|
|
|
@lock_acquired = false
|
|
|
|
@logger = Log4r::Logger.new("vagrant::environment")
|
|
@logger.info("Environment initialized (#{self})")
|
|
@logger.info(" - cwd: #{cwd}")
|
|
|
|
# Setup the home directory
|
|
setup_home_path
|
|
@tmp_path = @home_path.join("tmp")
|
|
@boxes_path = @home_path.join("boxes")
|
|
@gems_path = @home_path.join("gems")
|
|
|
|
# Setup the local data directory. If a configuration path is given,
|
|
# then it is expanded relative to the working directory. Otherwise,
|
|
# we use the default which is expanded relative to the root path.
|
|
@local_data_path = nil
|
|
if opts[:local_data_path]
|
|
@local_data_path = Pathname.new(File.expand_path(opts[:local_data_path], @cwd))
|
|
elsif !root_path.nil?
|
|
@local_data_path = root_path.join(DEFAULT_LOCAL_DATA)
|
|
end
|
|
|
|
setup_local_data_path
|
|
|
|
# Setup the default private key
|
|
@default_private_key_path = @home_path.join("insecure_private_key")
|
|
copy_insecure_private_key
|
|
|
|
# Load the plugins
|
|
load_plugins
|
|
|
|
# Initialize the configuration. This will load our global configuration.
|
|
initialize_config
|
|
end
|
|
|
|
# Return a human-friendly string for pretty printed or inspected
|
|
# instances.
|
|
#
|
|
# @return [String]
|
|
def inspect
|
|
"#<#{self.class}: #{@cwd}>"
|
|
end
|
|
|
|
#---------------------------------------------------------------
|
|
# Helpers
|
|
#---------------------------------------------------------------
|
|
|
|
# This returns the provider name for the default provider for this
|
|
# environment. The provider returned is currently hardcoded to "virtualbox"
|
|
# but one day should be a detected valid, best-case provider for this
|
|
# environment.
|
|
#
|
|
# @return [Symbol] Name of the default provider.
|
|
def default_provider
|
|
:virtualbox
|
|
end
|
|
|
|
# Returns the collection of boxes for the environment.
|
|
#
|
|
# @return [BoxCollection]
|
|
def boxes
|
|
@_boxes ||= BoxCollection.new(boxes_path)
|
|
end
|
|
|
|
# This returns a machine with the proper provider for this environment.
|
|
# The machine named by `name` must be in this environment.
|
|
#
|
|
# @param [Symbol] name Name of the machine (as configured in the
|
|
# Vagrantfile).
|
|
# @param [Symbol] provider The provider that this machine should be
|
|
# backed by.
|
|
# @return [Machine]
|
|
def machine(name, provider)
|
|
@logger.info("Getting machine: #{name} (#{provider})")
|
|
|
|
# Compose the cache key of the name and provider, and return from
|
|
# the cache if we have that.
|
|
cache_key = [name, provider]
|
|
@machines ||= {}
|
|
if @machines.has_key?(cache_key)
|
|
@logger.info("Returning cached machine: #{name} (#{provider})")
|
|
return @machines[cache_key]
|
|
end
|
|
|
|
@logger.info("Uncached load of machine.")
|
|
sub_vm = config_global.vm.defined_vms[name]
|
|
if !sub_vm
|
|
raise Errors::MachineNotFound, :name => name, :provider => provider
|
|
end
|
|
|
|
provider_cls = Vagrant.plugin("2").manager.providers[provider]
|
|
if !provider_cls
|
|
raise Errors::ProviderNotFound, :machine => name, :provider => provider
|
|
end
|
|
|
|
# Build the machine configuration. This requires two passes: The first pass
|
|
# loads in the machine sub-configuration. Since this can potentially
|
|
# define a new box to base the machine from, we then make a second pass
|
|
# with the box Vagrantfile (if it has one).
|
|
vm_config_key = "vm_#{name}".to_sym
|
|
@config_loader.set(vm_config_key, sub_vm.config_procs)
|
|
config = @config_loader.load([:default, :home, :root, vm_config_key])
|
|
|
|
box = nil
|
|
begin
|
|
box = boxes.find(config.vm.box, provider)
|
|
rescue Errors::BoxUpgradeRequired
|
|
# Upgrade the box if we must
|
|
@logger.info("Upgrading box during config load: #{config.vm.box}")
|
|
boxes.upgrade(config.vm.box)
|
|
retry
|
|
end
|
|
|
|
# If a box was found, then we attempt to load the Vagrantfile for
|
|
# that box. We don't require a box since we allow providers to download
|
|
# boxes and so on.
|
|
if box
|
|
box_vagrantfile = find_vagrantfile(box.directory)
|
|
if box_vagrantfile
|
|
# The box has a custom Vagrantfile, so we load that into the config
|
|
# as well.
|
|
@logger.info("Box exists with Vagrantfile. Reloading machine config.")
|
|
box_config_key = "box_#{box.name}_#{box.provider}".to_sym
|
|
@config_loader.set(box_config_key, box_vagrantfile)
|
|
config = @config_loader.load([:default, box_config_key, :home, :root, vm_config_key])
|
|
end
|
|
end
|
|
|
|
# Get the provider configuration from the final loaded configuration
|
|
provider_config = config.vm.providers[provider].config
|
|
|
|
# Create the machine and cache it for future calls. This will also
|
|
# return the machine from this method.
|
|
@machines[cache_key] = Machine.new(name, provider_cls, provider_config,
|
|
config, box, self)
|
|
end
|
|
|
|
# This returns a list of the configured machines for this environment.
|
|
# Each of the names returned by this method is valid to be used with
|
|
# the {#machine} method.
|
|
#
|
|
# @return [Array<Symbol>] Configured machine names.
|
|
def machine_names
|
|
config_global.vm.defined_vm_keys.dup
|
|
end
|
|
|
|
# Returns the primary VM associated with this environment. This
|
|
# method is only applicable for multi-VM environments. This can
|
|
# potentially be nil if no primary VM is specified.
|
|
#
|
|
# @param [Symbol] provider The provider to back the primary machine.
|
|
# @return [VM]
|
|
def primary_machine(provider)
|
|
if machine_names.length == 1
|
|
return machine(machine_names[0], provider)
|
|
end
|
|
|
|
config_global.vm.defined_vms.each do |name, subvm|
|
|
return machine(name, provider) if subvm.options[:primary]
|
|
end
|
|
|
|
nil
|
|
end
|
|
|
|
# Makes a call to the CLI with the given arguments as if they
|
|
# came from the real command line (sometimes they do!). An example:
|
|
#
|
|
# env.cli("package", "--vagrantfile", "Vagrantfile")
|
|
#
|
|
def cli(*args)
|
|
CLI.new(args.flatten, self).execute
|
|
end
|
|
|
|
# Returns the host object associated with this environment.
|
|
#
|
|
# @return [Class]
|
|
def host
|
|
return @host if defined?(@host)
|
|
|
|
# Attempt to figure out the host class. Note that the order
|
|
# matters here, so please don't touch. Specifically: The symbol
|
|
# check is done after the detect check because the symbol check
|
|
# will return nil, and we don't want to trigger a detect load.
|
|
host_klass = config_global.vagrant.host
|
|
if host_klass.nil? || host_klass == :detect
|
|
hosts = Vagrant.plugin("2").manager.hosts
|
|
|
|
# Get the flattened list of available hosts
|
|
host_klass = Hosts.detect(hosts)
|
|
end
|
|
|
|
# If no host class is detected, we use the base class.
|
|
host_klass ||= Vagrant.plugin("2", :host)
|
|
|
|
@host ||= host_klass.new(@ui)
|
|
end
|
|
|
|
# Action runner for executing actions in the context of this environment.
|
|
#
|
|
# @return [Action::Runner]
|
|
def action_runner
|
|
@action_runner ||= Action::Runner.new do
|
|
{
|
|
:action_runner => action_runner,
|
|
:box_collection => boxes,
|
|
:global_config => config_global,
|
|
:host => host,
|
|
:root_path => root_path,
|
|
:tmp_path => tmp_path,
|
|
:ui => @ui
|
|
}
|
|
end
|
|
end
|
|
|
|
# Loads on initial access and reads data from the global data store.
|
|
# The global data store is global to Vagrant everywhere (in every environment),
|
|
# so it can be used to store system-wide information. Note that "system-wide"
|
|
# typically means "for this user" since the location of the global data
|
|
# store is in the home directory.
|
|
#
|
|
# @return [DataStore]
|
|
def global_data
|
|
@global_data ||= DataStore.new(File.expand_path("global_data.json", home_path))
|
|
end
|
|
|
|
# Loads (on initial access) and reads data from the local data
|
|
# store. This file is always at the root path as the file "~/.vagrant"
|
|
# and contains a JSON dump of a hash. See {DataStore} for more
|
|
# information.
|
|
#
|
|
# @return [DataStore]
|
|
def local_data
|
|
@local_data ||= DataStore.new(@local_data_path.join("environment_data"))
|
|
end
|
|
|
|
# The root path is the path where the top-most (loaded last)
|
|
# Vagrantfile resides. It can be considered the project root for
|
|
# this environment.
|
|
#
|
|
# @return [String]
|
|
def root_path
|
|
return @root_path if defined?(@root_path)
|
|
|
|
root_finder = lambda do |path|
|
|
# Note: To remain compatible with Ruby 1.8, we have to use
|
|
# a `find` here instead of an `each`.
|
|
found = vagrantfile_name.find do |rootfile|
|
|
File.exist?(File.join(path.to_s, rootfile))
|
|
end
|
|
|
|
return path if found
|
|
return nil if path.root? || !File.exist?(path)
|
|
root_finder.call(path.parent)
|
|
end
|
|
|
|
@root_path = root_finder.call(cwd)
|
|
end
|
|
|
|
# This returns the path which Vagrant uses to determine the location
|
|
# of the file lock. This is specific to each operating system.
|
|
def lock_path
|
|
@lock_path || tmp_path.join("vagrant.lock")
|
|
end
|
|
|
|
# This locks Vagrant for the duration of the block passed to this
|
|
# method. During this time, any other environment which attempts
|
|
# to lock which points to the same lock file will fail.
|
|
def lock
|
|
# This allows multiple locks in the same process to be nested
|
|
return yield if @lock_acquired
|
|
|
|
File.open(lock_path, "w+") do |f|
|
|
# The file locking fails only if it returns "false." If it
|
|
# succeeds it returns a 0, so we must explicitly check for
|
|
# the proper error case.
|
|
raise Errors::EnvironmentLockedError if f.flock(File::LOCK_EX | File::LOCK_NB) === false
|
|
|
|
begin
|
|
# Mark that we have a lock
|
|
@lock_acquired = true
|
|
|
|
yield
|
|
ensure
|
|
# We need to make sure that no matter what this is always
|
|
# reset to false so we don't think we have a lock when we
|
|
# actually don't.
|
|
@lock_acquired = false
|
|
end
|
|
end
|
|
end
|
|
|
|
#---------------------------------------------------------------
|
|
# Load Methods
|
|
#---------------------------------------------------------------
|
|
|
|
# This initializes the config loader for this environment. The config
|
|
# loader is cached so that prior Vagrantfiles aren't loaded multiple
|
|
# times.
|
|
def initialize_config
|
|
@logger.info("Initialzing config...")
|
|
|
|
home_vagrantfile = nil
|
|
root_vagrantfile = nil
|
|
home_vagrantfile = find_vagrantfile(home_path) if home_path
|
|
root_vagrantfile = find_vagrantfile(root_path) if root_path
|
|
|
|
# Create the configuration loader and set the sources that are global.
|
|
# We use this to load the configuration, and the list of machines we are
|
|
# managing. Then, the actual individual configuration is loaded for
|
|
# each {#machine} call.
|
|
@config_loader = Config::Loader.new(Config::VERSIONS, Config::VERSIONS_ORDER)
|
|
@config_loader.set(:default, File.expand_path("config/default.rb", Vagrant.source_root))
|
|
@config_loader.set(:home, home_vagrantfile) if home_vagrantfile
|
|
@config_loader.set(:root, root_vagrantfile) if root_vagrantfile
|
|
|
|
# Make the initial call to get the "global" config. This is mostly
|
|
# only useful to get the list of machines that we are managing.
|
|
@config_global = @config_loader.load([:default, :home, :root])
|
|
|
|
# Old order: default, box, home, root, vm
|
|
end
|
|
|
|
# This sets the `@home_path` variable properly.
|
|
#
|
|
# @return [Pathname]
|
|
def setup_home_path
|
|
@home_path = Pathname.new(File.expand_path(@home_path ||
|
|
ENV["VAGRANT_HOME"] ||
|
|
DEFAULT_HOME))
|
|
@logger.info("Home path: #{@home_path}")
|
|
|
|
# Setup the list of child directories that need to be created if they
|
|
# don't already exist.
|
|
dirs = [@home_path]
|
|
subdirs = ["tmp", "boxes", "gems"]
|
|
dirs += subdirs.collect { |subdir| @home_path.join(subdir) }
|
|
|
|
# Go through each required directory, creating it if it doesn't exist
|
|
dirs.each do |dir|
|
|
next if File.directory?(dir)
|
|
|
|
begin
|
|
@logger.info("Creating: #{dir}")
|
|
FileUtils.mkdir_p(dir)
|
|
rescue Errno::EACCES
|
|
raise Errors::HomeDirectoryNotAccessible, :home_path => @home_path.to_s
|
|
end
|
|
end
|
|
end
|
|
|
|
# This creates the local data directory and show an error if it
|
|
# couldn't properly be created.
|
|
def setup_local_data_path
|
|
if @local_data_path.nil?
|
|
@logger.warn("No local data path is set. Local data cannot be stored.")
|
|
return
|
|
end
|
|
|
|
@logger.info("Local data path: #{@local_data_path}")
|
|
|
|
# If the local data path is a file, then we are probably seeing an
|
|
# old (V1) "dotfile." In this case, we upgrade it. The upgrade process
|
|
# will remove the old data file if it is successful.
|
|
if @local_data_path.file?
|
|
upgrade_v1_dotfile(@local_data_path)
|
|
end
|
|
|
|
begin
|
|
@logger.debug("Creating: #{@local_data_path}")
|
|
FileUtils.mkdir_p(@local_data_path)
|
|
rescue Errno::EACCES
|
|
raise Errors::LocalDataDirectoryNotAccessible,
|
|
:local_data_path => @local_data_path.to_s
|
|
end
|
|
end
|
|
|
|
protected
|
|
|
|
# This method copies the private key into the home directory if it
|
|
# doesn't already exist.
|
|
#
|
|
# This must be done because `ssh` requires that the key is chmod
|
|
# 0600, but if Vagrant is installed as a separate user, then the
|
|
# effective uid won't be able to read the key. So the key is copied
|
|
# to the home directory and chmod 0600.
|
|
def copy_insecure_private_key
|
|
if !@default_private_key_path.exist?
|
|
@logger.info("Copying private key to home directory")
|
|
FileUtils.cp(File.expand_path("keys/vagrant", Vagrant.source_root),
|
|
@default_private_key_path)
|
|
end
|
|
|
|
if !Util::Platform.windows?
|
|
# On Windows, permissions don't matter as much, so don't worry
|
|
# about doing chmod.
|
|
if Util::FileMode.from_octal(@default_private_key_path.stat.mode) != "600"
|
|
@logger.info("Changing permissions on private key to 0600")
|
|
@default_private_key_path.chmod(0600)
|
|
end
|
|
end
|
|
end
|
|
|
|
# Finds the Vagrantfile in the given directory.
|
|
#
|
|
# @param [Pathname] path Path to search in.
|
|
# @return [Pathname]
|
|
def find_vagrantfile(search_path)
|
|
@vagrantfile_name.each do |vagrantfile|
|
|
current_path = search_path.join(vagrantfile)
|
|
return current_path if current_path.exist?
|
|
end
|
|
|
|
nil
|
|
end
|
|
|
|
# Loads the Vagrant plugins by properly setting up RubyGems so that
|
|
# our private gem repository is on the path.
|
|
def load_plugins
|
|
# Add our private gem path to the gem path and reset the paths
|
|
# that Rubygems knows about.
|
|
ENV["GEM_PATH"] = "#{@gems_path}#{::File::PATH_SEPARATOR}#{ENV["GEM_PATH"]}"
|
|
::Gem.clear_paths
|
|
|
|
# Load the plugins
|
|
rc_path = File.expand_path(ENV["VAGRANT_RC"] || DEFAULT_RC)
|
|
if File.file?(rc_path) && @@loaded_rc.add?(rc_path)
|
|
@logger.debug("Loading RC file: #{rc_path}")
|
|
load rc_path
|
|
else
|
|
@logger.debug("RC file not found. Not loading: #{rc_path}")
|
|
end
|
|
end
|
|
|
|
# This upgrades a Vagrant 1.0.x "dotfile" to the new V2 format.
|
|
#
|
|
# This is a destructive process. Once the upgrade is complete, the
|
|
# old dotfile is removed, and the environment becomes incompatible for
|
|
# Vagrant 1.0 environments.
|
|
#
|
|
# @param [Pathname] path The path to the dotfile
|
|
def upgrade_v1_dotfile(path)
|
|
raise "V1 environment detected. An auto-upgrade process will be made soon."
|
|
end
|
|
end
|
|
end
|