core: BoxAdd supports multiple URLs properly

This commit is contained in:
Mitchell Hashimoto 2014-01-24 12:50:07 -08:00
parent 38248c240c
commit f0607c6df0
4 changed files with 137 additions and 23 deletions

View File

@ -22,30 +22,49 @@ module Vagrant
def call(env) def call(env)
@download_interrupted = false @download_interrupted = false
url = env[:box_url] url = Array(env[:box_url])
# If we received a shorthand URL ("mitchellh/precise64"), # If we received a shorthand URL ("mitchellh/precise64"),
# then expand it properly. # then expand it properly.
expanded = false expanded = false
uri = URI.parse(url) url.each_index do |i|
if !uri.scheme && !File.file?(url) uri = URI.parse(url[i])
if !uri.scheme && !File.file?(url[i])
expanded = true expanded = true
url = "#{Vagrant.server_url}/#{url}" url[i] = "#{Vagrant.server_url}/#{url[i]}"
end
end end
is_metadata = false # Test if any of our URLs point to metadata
is_metadata_results = url.map do |u|
begin begin
is_metadata = metadata_url?(url, env) metadata_url?(u, env)
rescue Errors::DownloaderError => e rescue Errors::DownloaderError => e
raise if !expanded e
end
end
if expanded && url.length == 1
is_error = is_metadata_results.find do |b|
b.is_a?(Errors::DownloaderError)
end
if is_error
raise Errors::BoxAddShortNotFound, raise Errors::BoxAddShortNotFound,
error: e.extra_data[:message], error: is_error.extra_data[:message],
name: env[:box_url], name: env[:box_url],
url: url url: url
end end
end
is_metadata = is_metadata_results.any? { |b| b === true }
if is_metadata && url.length > 1
raise Errors::BoxAddMetadataMultiURL,
urls: url.join(", ")
end
if is_metadata if is_metadata
add_from_metadata(url, env) add_from_metadata(url.first, env, expanded)
else else
add_direct(url, env) add_direct(url, env)
end end
@ -55,7 +74,10 @@ module Vagrant
# Adds a box file directly (no metadata component, versioning, # Adds a box file directly (no metadata component, versioning,
# etc.) # etc.)
def add_direct(url, env) #
# @param [Array<String>] urls
# @param [Hash] env
def add_direct(urls, env)
name = env[:box_name] name = env[:box_name]
if !name || name == "" if !name || name == ""
raise Errors::BoxAddNameRequired raise Errors::BoxAddNameRequired
@ -65,7 +87,7 @@ module Vagrant
provider = Array(provider) if provider provider = Array(provider) if provider
box_add( box_add(
url, urls,
name, name,
"0", "0",
provider, provider,
@ -73,7 +95,7 @@ module Vagrant
end end
# Adds a box given that the URL is a metadata document. # Adds a box given that the URL is a metadata document.
def add_from_metadata(url, env) def add_from_metadata(url, env, expanded)
original_url = env[:box_url] original_url = env[:box_url]
provider = env[:box_provider] provider = env[:box_provider]
provider = Array(provider) if provider provider = Array(provider) if provider
@ -93,6 +115,12 @@ module Vagrant
File.open(metadata_path) do |f| File.open(metadata_path) do |f|
metadata = BoxMetadata.new(f) metadata = BoxMetadata.new(f)
end end
rescue Errors::DownloaderError => e
raise if !expanded
raise Errors::BoxAddShortNotFound,
error: e.extra_data[:message],
name: original_url,
url: url
ensure ensure
metadata_path.delete if metadata_path && metadata_path.file? metadata_path.delete if metadata_path && metadata_path.file?
end end
@ -160,7 +188,7 @@ module Vagrant
end end
box_add( box_add(
metadata_provider.url, [metadata_provider.url],
metadata.name, metadata.name,
metadata_version.version, metadata_version.version,
metadata_provider.name, env) metadata_provider.name, env)
@ -232,13 +260,13 @@ module Vagrant
# Shared helper to add a box once you know various details # Shared helper to add a box once you know various details
# about it. Shared between adding via metadata or by direct. # about it. Shared between adding via metadata or by direct.
# #
# @param [String] url # @param [Array<String>] urls
# @param [String] name # @param [String] name
# @param [String] version # @param [String] version
# @param [String] provider # @param [String] provider
# @param [Hash] env # @param [Hash] env
# @return [Box] # @return [Box]
def box_add(url, name, version, provider, env, **opts) def box_add(urls, name, version, provider, env, **opts)
env[:ui].output(I18n.t( env[:ui].output(I18n.t(
"vagrant.box_add_with_version", "vagrant.box_add_with_version",
name: name, name: name,
@ -259,8 +287,22 @@ module Vagrant
# Now we have a URL, we have to download this URL. # Now we have a URL, we have to download this URL.
box = nil box = nil
begin
box_url = nil
urls.each do |url|
env[:ui].detail(I18n.t(
"vagrant.box_downloading", url: url))
begin begin
box_url = download(url, env) box_url = download(url, env)
break
rescue Errors::DownloaderError => e
env[:ui].error(I18n.t(
"vagrant.box_download_error", message: e.message))
box_url = nil
end
end
# Add the box! # Add the box!
box = env[:box_collection].add( box = env[:box_collection].add(
@ -273,7 +315,7 @@ module Vagrant
# so we can resume the download later. # so we can resume the download later.
if !@download_interrupted if !@download_interrupted
@logger.debug("Deleting temporary box: #{box_url}") @logger.debug("Deleting temporary box: #{box_url}")
box_url.delete box_url.delete if box_url
end end
end end

View File

@ -120,6 +120,10 @@ module Vagrant
error_key(:batch_multi_error) error_key(:batch_multi_error)
end end
class BoxAddMetadataMultiURL < VagrantError
error_key(:box_add_metadata_multi_url)
end
class BoxAddNameMismatch < VagrantError class BoxAddNameMismatch < VagrantError
error_key(:box_add_name_mismatch) error_key(:box_add_name_mismatch)
end end

View File

@ -20,6 +20,10 @@ en:
Adding box '%{name}' (v%{version}) for '%{provider}' provider... Adding box '%{name}' (v%{version}) for '%{provider}' provider...
box_added: |- box_added: |-
Successfully added box '%{name}' for '%{provider}'! Successfully added box '%{name}' for '%{provider}'!
box_downloading: |-
Downloading: %{url}
box_download_error: |-
Error downloading: %{message}
box_expanding_url: |- box_expanding_url: |-
URL: %{url} URL: %{url}
box_loading_metadata: |- box_loading_metadata: |-
@ -289,6 +293,13 @@ en:
Name: %{name} Name: %{name}
Provider: %{provider} Provider: %{provider}
Version: %{version} Version: %{version}
box_add_metadata_multi_url: |-
Multiple URLs for a box can't be specified when adding
versioned boxes. Please specify a single URL to the box
metadata (JSON) information. The full list of URLs you
specified is shown below:
%{urls}
box_add_name_mismatch: |- box_add_name_mismatch: |-
The box you're adding has a name different from the name you The box you're adding has a name different from the name you
requested. For boxes with metadata, you cannot override the name. requested. For boxes with metadata, you cannot override the name.

View File

@ -77,6 +77,27 @@ describe Vagrant::Action::Builtin::BoxAdd do
subject.call(env) subject.call(env)
end end
it "adds from multiple URLs" do
box_path = iso_env.box2_file(:virtualbox)
env[:box_name] = "foo"
env[:box_url] = [
"/foo/bar/baz",
box_path.to_s,
]
box_collection.should_receive(:add).with do |path, name, version|
expect(checksum(path)).to eq(checksum(box_path))
expect(name).to eq("foo")
expect(version).to eq("0")
true
end.and_return(box)
app.should_receive(:call).with(env)
subject.call(env)
end
it "adds from HTTP URL" do it "adds from HTTP URL" do
box_path = iso_env.box2_file(:virtualbox) box_path = iso_env.box2_file(:virtualbox)
with_web_server(box_path) do |port| with_web_server(box_path) do |port|
@ -252,6 +273,42 @@ describe Vagrant::Action::Builtin::BoxAdd do
end end
end end
it "raises an error if multiple metadata URLs are given" do
box_path = iso_env.box2_file(:virtualbox)
tf = Tempfile.new("vagrant").tap do |f|
f.write(<<-RAW)
{
"name": "foo/bar",
"versions": [
{
"version": "0.5"
},
{
"version": "0.7",
"providers": [
{
"name": "virtualbox",
"url": "#{box_path}"
}
]
}
]
}
RAW
f.close
end
env[:box_url] = [
"/foo/bar/baz",
tf.path,
]
box_collection.should_receive(:add).never
app.should_receive(:call).never
expect { subject.call(env) }.
to raise_error(Vagrant::Errors::BoxAddMetadataMultiURL)
end
it "adds the latest version of a box with only one provider" do it "adds the latest version of a box with only one provider" do
box_path = iso_env.box2_file(:virtualbox) box_path = iso_env.box2_file(:virtualbox)
tf = Tempfile.new("vagrant").tap do |f| tf = Tempfile.new("vagrant").tap do |f|