commands/box/remove: if box is in use, warn user, ask for confirmation

This commit is contained in:
Mitchell Hashimoto 2014-04-23 06:13:16 -07:00
parent a1e6d0a85e
commit 85f4a4d5ee
3 changed files with 121 additions and 0 deletions

View File

@ -71,6 +71,49 @@ module Vagrant
box = env[:box_collection].find( box = env[:box_collection].find(
box_name, box_provider, box_version) box_name, box_provider, box_version)
# Verify that this box is not in use by an active machine,
# otherwise warn the user.
users = []
env[:machine_index].each do |entry|
box_data = entry.extra_data["box"]
next if !box_data
# If all the data matches AND the entry is a seemingly
# valid entry, then track it.
if box_data["name"] == box.name &&
box_data["provider"] == box.provider.to_s &&
box_data["version"] == box.version.to_s &&
entry.valid?(env[:home_path])
users << entry
end
end
if !users.empty?
# Build up the output to show the user.
users = users.map do |entry|
"#{entry.name} (ID: #{entry.id})"
end.join("\n")
force_key = :force_confirm_box_remove
message = I18n.t(
"vagrant.commands.box.remove_in_use_query",
name: box.name,
provider: box.provider,
version: box.version,
users: users) + " "
# Ask the user if we should do this
stack = Builder.new.tap do |b|
b.use Confirm, message, force_key
end
result = env[:action_runner].run(stack, env)
if !result[:result]
# They said "no", so just return
return @app.call(env)
end
end
env[:ui].info(I18n.t("vagrant.commands.box.removing", env[:ui].info(I18n.t("vagrant.commands.box.removing",
:name => box.name, :name => box.name,
:provider => box.provider, :provider => box.provider,

View File

@ -1251,6 +1251,15 @@ en:
vm_not_running: "VM is not currently running. Please, first bring it up with `vagrant up` then run this command." vm_not_running: "VM is not currently running. Please, first bring it up with `vagrant up` then run this command."
box: box:
no_installed_boxes: "There are no installed boxes! Use `vagrant box add` to add some." no_installed_boxes: "There are no installed boxes! Use `vagrant box add` to add some."
remove_in_use_query: |-
Box '%{name}' (v%{version}) with provider '%{provider}' appears
to still be in use by at least one Vagrant environment. Removing
the box could corrupt the environment. We recommend destroying
these environments first:
%{users}
Are you sure you want to remove this box? [y/N]
removing: |- removing: |-
Removing box '%{name}' (v%{version}) with provider '%{provider}'... Removing box '%{name}' (v%{version}) with provider '%{provider}'...
destroy: destroy:

View File

@ -6,12 +6,16 @@ describe Vagrant::Action::Builtin::BoxRemove do
let(:app) { lambda { |env| } } let(:app) { lambda { |env| } }
let(:env) { { let(:env) { {
box_collection: box_collection, box_collection: box_collection,
home_path: home_path,
machine_index: machine_index,
ui: Vagrant::UI::Silent.new, ui: Vagrant::UI::Silent.new,
} } } }
subject { described_class.new(app, env) } subject { described_class.new(app, env) }
let(:box_collection) { double("box_collection") } let(:box_collection) { double("box_collection") }
let(:home_path) { "foo" }
let(:machine_index) { [] }
let(:iso_env) { isolated_environment } let(:iso_env) { isolated_environment }
let(:box) do let(:box) do
@ -74,6 +78,71 @@ describe Vagrant::Action::Builtin::BoxRemove do
expect(env[:box_removed]).to equal(box) expect(env[:box_removed]).to equal(box)
end end
context "checking if a box is in use" do
def new_entry(name, provider, version, valid=true)
Vagrant::MachineIndex::Entry.new.tap do |entry|
entry.extra_data["box"] = {
"name" => "foo",
"provider" => "virtualbox",
"version" => "1.0",
}
entry.stub(valid?: valid)
end
end
let(:action_runner) { double("action_runner") }
before do
env[:action_runner] = action_runner
box_collection.stub(
all: [
["foo", "1.0", :virtualbox],
["foo", "1.1", :virtualbox],
])
env[:box_name] = "foo"
env[:box_version] = "1.0"
end
it "does delete if the box is not in use" do
expect(box_collection).to receive(:find).with(
"foo", :virtualbox, "1.0").and_return(box)
expect(box).to receive(:destroy!).once
subject.call(env)
end
it "does delete if the box is in use and user confirms" do
machine_index << new_entry("foo", "virtualbox", "1.0")
result = { result: true }
expect(action_runner).to receive(:run).
with(anything, env).and_return(result)
expect(box_collection).to receive(:find).with(
"foo", :virtualbox, "1.0").and_return(box)
expect(box).to receive(:destroy!).once
subject.call(env)
end
it "doesn't delete if the box is in use" do
machine_index << new_entry("foo", "virtualbox", "1.0")
result = { result: false }
expect(action_runner).to receive(:run).
with(anything, env).and_return(result)
expect(box_collection).to receive(:find).with(
"foo", :virtualbox, "1.0").and_return(box)
expect(box).to receive(:destroy!).never
subject.call(env)
end
end
it "errors if the box doesn't exist" do it "errors if the box doesn't exist" do
box_collection.stub(all: []) box_collection.stub(all: [])