2012-07-10 01:31:53 +00:00
|
|
|
require "digest/sha1"
|
2012-07-10 02:36:18 +00:00
|
|
|
require "tmpdir"
|
2012-07-10 01:31:53 +00:00
|
|
|
|
|
|
|
require "archive/tar/minitar"
|
|
|
|
require "log4r"
|
2010-12-01 04:13:45 +00:00
|
|
|
|
2010-09-11 17:17:26 +00:00
|
|
|
module Vagrant
|
2012-07-10 01:31:53 +00:00
|
|
|
# Represents a collection a boxes found on disk. This provides methods
|
|
|
|
# for accessing/finding individual boxes, adding new boxes, or deleting
|
|
|
|
# boxes.
|
2010-12-01 04:13:45 +00:00
|
|
|
class BoxCollection
|
2012-07-10 01:31:53 +00:00
|
|
|
# The directory where the boxes in this collection are stored.
|
|
|
|
#
|
|
|
|
# @return [Pathname]
|
2011-12-04 19:39:44 +00:00
|
|
|
attr_reader :directory
|
2010-09-11 17:17:26 +00:00
|
|
|
|
2012-07-10 01:31:53 +00:00
|
|
|
# Initializes the collection.
|
|
|
|
#
|
|
|
|
# @param [Pathname] directory The directory that contains the collection
|
|
|
|
# of boxes.
|
|
|
|
def initialize(directory)
|
|
|
|
@directory = directory
|
|
|
|
@logger = Log4r::Logger.new("vagrant::box_collection")
|
|
|
|
end
|
|
|
|
|
|
|
|
# This adds a new box to the system.
|
|
|
|
#
|
|
|
|
# There are some exceptional cases:
|
|
|
|
# * BoxAlreadyExists - The box you're attempting to add already exists.
|
|
|
|
# * BoxProviderDoesntMatch - If the given box provider doesn't match the
|
|
|
|
# actual box provider in the untarred box.
|
|
|
|
# * BoxUnpackageFailure - An invalid tar file.
|
|
|
|
# * BoxUpgradeRequired - You're attempting to add a box when there is a
|
|
|
|
# V1 box with the same name that must first be upgraded.
|
|
|
|
#
|
|
|
|
# Preconditions:
|
|
|
|
# * File given in `path` must exist.
|
|
|
|
#
|
|
|
|
# @param [Pathname] path Path to the box file on disk.
|
|
|
|
# @param [String] name Logical name for the box.
|
|
|
|
# @param [Symbol] provider The provider that the box should be for. This
|
|
|
|
# will be verified with the `metadata.json` file in the box and is
|
2012-07-10 02:36:18 +00:00
|
|
|
# meant as a basic check. If this isn't given, then whatever provider
|
|
|
|
# the box represents will be added.
|
|
|
|
def add(path, name, provider=nil)
|
|
|
|
# A helper to check if a box exists. We store this in a variable
|
|
|
|
# since we call it multiple times.
|
|
|
|
check_box_exists = lambda do |box_provider|
|
|
|
|
if find(name, box_provider)
|
|
|
|
@logger.error("Box already exists, can't add: #{name} #{box_provider}")
|
|
|
|
raise Errors::BoxAlreadyExists, :name => name, :provider => box_provider
|
|
|
|
end
|
2012-07-10 01:31:53 +00:00
|
|
|
end
|
|
|
|
|
2012-07-10 02:36:18 +00:00
|
|
|
log_provider = provider ? provider : "any provider"
|
|
|
|
@logger.debug("Adding box: #{name} (#{log_provider}) from #{path}")
|
|
|
|
|
|
|
|
# Verify the box doesn't exist early if we're given a provider. This
|
|
|
|
# can potentially speed things up considerably since we don't need
|
|
|
|
# to unpack any files.
|
|
|
|
check_box_exists.call(provider) if provider
|
2012-07-10 01:31:53 +00:00
|
|
|
|
2012-07-10 02:36:18 +00:00
|
|
|
# Create a temporary directory since we're not sure at this point if
|
|
|
|
# the box we're unpackaging already exists (if no provider was given)
|
|
|
|
Dir.mktmpdir("vagrant-") do |temp_dir|
|
|
|
|
# Extract the box into a temporary directory.
|
|
|
|
@logger.debug("Unpacking box into temporary directory: #{temp_dir}")
|
2012-07-10 01:31:53 +00:00
|
|
|
|
|
|
|
begin
|
2012-07-10 02:36:18 +00:00
|
|
|
Archive::Tar::Minitar.unpack(path.to_s, temp_dir)
|
2012-07-10 01:31:53 +00:00
|
|
|
rescue SystemCallError
|
|
|
|
raise Errors::BoxUnpackageFailure
|
|
|
|
end
|
|
|
|
|
2012-07-10 02:36:18 +00:00
|
|
|
# Verify that the box we just added matches the provider
|
|
|
|
# we expected.
|
|
|
|
box = Box.new(name, provider, Pathname.new(temp_dir))
|
|
|
|
|
|
|
|
# Verify that the provider matches. If not, then we error and never
|
|
|
|
# move to the final location.
|
|
|
|
box_provider = box.metadata["provider"]
|
|
|
|
if provider
|
|
|
|
# Verify that the given provider matches what the box has.
|
|
|
|
if box_provider.to_sym != provider
|
|
|
|
@logger.error("Added box provider doesnt match expected: #{box_provider}")
|
|
|
|
raise Errors::BoxProviderDoesntMatch, :expected => provider, :actual => box_provider
|
|
|
|
end
|
|
|
|
else
|
|
|
|
# We weren't given a provider, so store this one.
|
|
|
|
provider = box_provider.to_sym
|
|
|
|
|
|
|
|
# Verify the box doesn't already exist
|
|
|
|
check_box_exists.call(provider)
|
|
|
|
end
|
2012-07-10 01:31:53 +00:00
|
|
|
|
2012-07-10 02:36:18 +00:00
|
|
|
# Create the directory that'll store our box
|
|
|
|
final_dir = @directory.join(name, provider.to_s)
|
|
|
|
@logger.debug("Final box directory: #{final_dir}")
|
|
|
|
final_dir.mkpath
|
2012-07-10 01:31:53 +00:00
|
|
|
|
2012-07-10 02:36:18 +00:00
|
|
|
# Move to the final destination
|
|
|
|
File.rename(temp_dir, final_dir.to_s)
|
2012-07-10 01:31:53 +00:00
|
|
|
|
2012-07-10 02:36:18 +00:00
|
|
|
# Recreate the directory. This avoids a bug in Ruby where `mktmpdir`
|
|
|
|
# cleanup doesn't check if the directory is already gone. Ruby bug
|
|
|
|
# #6715: http://bugs.ruby-lang.org/issues/6715
|
|
|
|
Dir.mkdir(temp_dir, 0700)
|
2012-07-10 01:31:53 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
# Return the box
|
2012-07-10 02:36:18 +00:00
|
|
|
find(name, provider)
|
2012-07-10 01:31:53 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
# This returns an array of all the boxes on the system, given by
|
|
|
|
# their name and their provider.
|
|
|
|
#
|
|
|
|
# @return [Array] Array of `[name, provider]` pairs of the boxes
|
|
|
|
# installed on this system. An optional third element in the array
|
|
|
|
# may specify `:v1` if the box is a version 1 box.
|
|
|
|
def all
|
|
|
|
results = []
|
|
|
|
|
|
|
|
@logger.debug("Finding all boxes in: #{@directory}")
|
|
|
|
@directory.children(true).each do |child|
|
|
|
|
box_name = child.basename.to_s
|
2010-12-01 04:13:45 +00:00
|
|
|
|
2012-07-10 01:31:53 +00:00
|
|
|
# If this is a V1 box, we still return that name, but specify
|
|
|
|
# that the box is a V1 box.
|
|
|
|
if v1_box?(child)
|
|
|
|
@logger.debug("V1 box found: #{box_name}")
|
|
|
|
results << [box_name, :virtualbox, :v1]
|
|
|
|
next
|
|
|
|
end
|
|
|
|
|
|
|
|
# Otherwise, traverse the subdirectories and see what providers
|
|
|
|
# we have.
|
|
|
|
child.children(true).each do |provider|
|
|
|
|
# Verify this is a potentially valid box. If it looks
|
|
|
|
# correct enough then include it.
|
|
|
|
if provider.directory? && provider.join("metadata.json").file?
|
|
|
|
provider_name = provider.basename.to_s.to_sym
|
|
|
|
@logger.debug("Box: #{box_name} (#{provider_name})")
|
|
|
|
results << [box_name, provider_name]
|
|
|
|
else
|
|
|
|
@logger.debug("Invalid box, ignoring: #{provider}")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
results
|
2010-09-11 17:17:26 +00:00
|
|
|
end
|
|
|
|
|
2012-07-10 01:31:53 +00:00
|
|
|
# Find a box in the collection with the given name and provider.
|
|
|
|
#
|
|
|
|
# @param [String] name Name of the box (logical name).
|
|
|
|
# @Param [String] provider Provider that the box implements.
|
|
|
|
# @return [Box] The box found, or `nil` if not found.
|
|
|
|
def find(name, provider)
|
|
|
|
# First look directly for the box we're asking for.
|
|
|
|
box_directory = @directory.join(name, provider.to_s, "metadata.json")
|
|
|
|
@logger.info("Searching for box: #{name} (#{provider}) in #{box_directory}")
|
|
|
|
if box_directory.file?
|
|
|
|
@logger.debug("Box found: #{name} (#{provider})")
|
|
|
|
return Box.new(name, provider, box_directory.dirname)
|
2010-09-11 17:17:26 +00:00
|
|
|
end
|
|
|
|
|
2012-07-10 01:31:53 +00:00
|
|
|
# Check if a V1 version of this box exists, and if so, raise an
|
|
|
|
# exception notifying the caller that the box exists but needs
|
|
|
|
# to be upgraded. We don't do the upgrade here because it can be
|
|
|
|
# a fairly intensive activity and don't want to immediately degrade
|
|
|
|
# user performance on a find.
|
|
|
|
#
|
|
|
|
# To determine if it is a V1 box we just do a simple heuristic
|
|
|
|
# based approach.
|
|
|
|
@logger.info("Searching for V1 box: #{name}")
|
|
|
|
if v1_box?(name)
|
|
|
|
@logger.warn("V1 box found: #{name}")
|
|
|
|
raise Errors::BoxUpgradeRequired, :name => name
|
|
|
|
end
|
|
|
|
|
|
|
|
# Didn't find it, return nil
|
|
|
|
@logger.info("Box not found: #{name} (#{provider})")
|
2010-09-11 17:17:26 +00:00
|
|
|
nil
|
|
|
|
end
|
|
|
|
|
2012-07-10 01:31:53 +00:00
|
|
|
# Upgrades a V1 box with the given name to a V2 box. If a box with the
|
|
|
|
# given name doesn't exist, then a `BoxNotFound` exception will be raised.
|
|
|
|
# If the given box is found but is not a V1 box then `true` is returned
|
|
|
|
# because this just works fine.
|
|
|
|
#
|
|
|
|
# @return [Boolean] `true` otherwise an exception is raised.
|
|
|
|
def upgrade(name)
|
|
|
|
@logger.debug("Upgrade request for box: #{name}")
|
|
|
|
box_dir = @directory.join(name)
|
2011-12-10 21:09:03 +00:00
|
|
|
|
2012-07-10 01:31:53 +00:00
|
|
|
# If the box doesn't exist at all, raise an exception
|
|
|
|
raise Errors::BoxNotFound, :name => name if !box_dir.directory?
|
|
|
|
|
|
|
|
if v1_box?(name)
|
|
|
|
@logger.debug("V1 box #{name} found. Upgrading!")
|
|
|
|
|
|
|
|
# First, we create a temporary directory within the box to store
|
|
|
|
# the intermediary moved files. We randomize this in case there is
|
|
|
|
# already a directory named "virtualbox" in here for some reason.
|
|
|
|
temp_dir = box_dir.join("vagrant-#{Digest::SHA1.hexdigest(name)}")
|
|
|
|
@logger.debug("Temporary directory for upgrading: #{temp_dir}")
|
2011-12-09 22:22:03 +00:00
|
|
|
|
2012-07-10 01:31:53 +00:00
|
|
|
# Make the temporary directory
|
|
|
|
temp_dir.mkpath
|
2010-09-11 17:17:26 +00:00
|
|
|
|
2012-07-10 01:31:53 +00:00
|
|
|
# Move all the things into the temporary directory
|
|
|
|
box_dir.children(true).each do |child|
|
|
|
|
# Don't move the temp_dir
|
|
|
|
next if child == temp_dir
|
|
|
|
|
|
|
|
# Move every other directory into the temporary directory
|
|
|
|
@logger.debug("Copying to upgrade directory: #{child}")
|
|
|
|
FileUtils.mv(child, temp_dir.join(child.basename))
|
|
|
|
end
|
|
|
|
|
|
|
|
# If there is no metadata.json file, make one, since this is how
|
|
|
|
# we determine if the box is a V2 box.
|
|
|
|
metadata_file = temp_dir.join("metadata.json")
|
|
|
|
if !metadata_file.file?
|
|
|
|
metadata_file.open("w") do |f|
|
|
|
|
f.write(JSON.generate({}))
|
|
|
|
end
|
2010-09-11 17:17:26 +00:00
|
|
|
end
|
2012-07-10 01:31:53 +00:00
|
|
|
|
|
|
|
# Rename the temporary directory to the provider.
|
|
|
|
temp_dir.rename(box_dir.join("virtualbox"))
|
|
|
|
@logger.info("Box '#{name}' upgraded from V1 to V2.")
|
2010-09-11 17:17:26 +00:00
|
|
|
end
|
2012-07-10 01:31:53 +00:00
|
|
|
|
|
|
|
# We did it! Or the v1 box didn't exist so it doesn't matter.
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
# This checks if the given name represents a V1 box on the system.
|
|
|
|
#
|
|
|
|
# @return [Boolean]
|
|
|
|
def v1_box?(name)
|
|
|
|
# We detect a V1 box given by whether there is a "box.ovf" which
|
|
|
|
# is a heuristic but is pretty accurate.
|
|
|
|
@directory.join(name, "box.ovf").file?
|
2010-09-11 17:17:26 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|