Create single interal modify block to sync VM modifications

This commit is contained in:
Mitchell Hashimoto 2011-07-08 11:59:29 -07:00
parent 4611800503
commit 6b2feae0e8
8 changed files with 109 additions and 13 deletions

View File

@ -23,6 +23,7 @@ module Vagrant
use VM::HostName use VM::HostName
use VM::Network use VM::Network
use VM::Customize use VM::Customize
use VM::Modify
use VM::Boot use VM::Boot
end) end)
@ -73,6 +74,7 @@ module Vagrant
use Action[:halt] use Action[:halt]
use VM::ClearForwardedPorts use VM::ClearForwardedPorts
use VM::ClearSharedFolders use VM::ClearSharedFolders
use VM::Modify
use VM::Export use VM::Export
use VM::PackageVagrantfile use VM::PackageVagrantfile
use VM::Package use VM::Package

View File

@ -18,6 +18,7 @@ module Vagrant
autoload :HostName, 'vagrant/action/vm/host_name' autoload :HostName, 'vagrant/action/vm/host_name'
autoload :Import, 'vagrant/action/vm/import' autoload :Import, 'vagrant/action/vm/import'
autoload :MatchMACAddress, 'vagrant/action/vm/match_mac_address' autoload :MatchMACAddress, 'vagrant/action/vm/match_mac_address'
autoload :Modify, 'vagrant/action/vm/modify'
autoload :Network, 'vagrant/action/vm/network' autoload :Network, 'vagrant/action/vm/network'
autoload :NFS, 'vagrant/action/vm/nfs' autoload :NFS, 'vagrant/action/vm/nfs'
autoload :Package, 'vagrant/action/vm/package' autoload :Package, 'vagrant/action/vm/package'

View File

@ -7,11 +7,15 @@ module Vagrant
end end
def call(env) def call(env)
if !env.env.config.vm.proc_stack.empty? if !env["config"].vm.proc_stack.empty?
env.ui.info I18n.t("vagrant.actions.vm.customize.running") # Create the proc which runs all of our procs
env.env.config.vm.run_procs!(env["vm"].vm) proc = lambda do |vm|
env["vm"].vm.save env.ui.info I18n.t("vagrant.actions.vm.customize.running")
env["vm"].reload! env["config"].vm.run_procs!(vm)
end
# Add it to modify sequence
env["vm.modify"].call(proc)
end end
@app.call(env) @app.call(env)

View File

@ -9,11 +9,15 @@ module Vagrant
def call(env) def call(env)
raise Errors::VMBaseMacNotSpecified if !env.env.config.vm.base_mac raise Errors::VMBaseMacNotSpecified if !env.env.config.vm.base_mac
env["config"].vm.customize do |vm| # Create the proc which we want to use to modify the virtual machine
proc = lambda do |vm|
env.ui.info I18n.t("vagrant.actions.vm.match_mac.matching") env.ui.info I18n.t("vagrant.actions.vm.match_mac.matching")
vm.network_adapters.first.mac_address = env["config"].vm.base_mac vm.network_adapters.first.mac_address = env["config"].vm.base_mac
end end
# Add the proc to the modification chain
env["vm.modify"].call(proc)
@app.call(env) @app.call(env)
end end
end end

View File

@ -0,0 +1,37 @@
module Vagrant
class Action
module VM
# This class allows other actions on the virtual machine object
# to be consolidated under a single write lock. This vastly speeds
# up modification of virtual machines. This should be used whereever
# possible when dealing with virtual machine modifications.
class Modify
include Util::StackedProcRunner
def initialize(app, env)
@app = app
# Initialize the proc_stack, which should already be empty
# but just making sure here.
proc_stack.clear
# Create the lambda in the environment which is to be called
# to add new procs to the modification sequence.
env["vm.modify"] = lambda do |*procs|
procs.each { |p| push_proc(&p) }
end
end
def call(env)
# Run the procs we have saved up, save the machine, and reload
# to verify we get the new settings
run_procs!(env["vm"].vm)
env["vm"].vm.save
env["vm"].reload!
@app.call(env)
end
end
end
end
end

View File

@ -8,23 +8,29 @@ class CustomizeVMActionTest < Test::Unit::TestCase
@vm = mock("vm") @vm = mock("vm")
@env["vm"] = @vm @env["vm"] = @vm
@env["vm.modify"] = mock("proc")
@internal_vm = mock("internal") @internal_vm = mock("internal")
@vm.stubs(:vm).returns(@internal_vm) @vm.stubs(:vm).returns(@internal_vm)
end end
should "not run anything if no customize blocks exist" do should "not run anything if no customize blocks exist" do
@env.env.config.vm.proc_stack.clear @env["config"].vm.proc_stack.clear
@internal_vm.expects(:save).never @env["vm.modify"].expects(:call).never
@app.expects(:call).with(@env).once @app.expects(:call).with(@env).once
@instance.call(@env) @instance.call(@env)
end end
should "run the VM customization procs then save the VM" do should "run the VM customization procs then save the VM" do
@env.env.config.vm.customize { |vm| } ran = false
@env.env.config.vm.expects(:run_procs!).with(@internal_vm) @env["config"].vm.customize { |vm| }
@internal_vm.expects(:save).once @env["config"].vm.expects(:run_procs!).with(@internal_vm)
@env["vm"].expects(:reload!).once
@env["vm.modify"].expects(:call).with() do |proc|
proc.call(@internal_vm)
true
end
@app.expects(:call).with(@env).once @app.expects(:call).with(@env).once
@instance.call(@env) @instance.call(@env)
end end

View File

@ -22,7 +22,11 @@ class MatchMACAddressVMActionTest < Test::Unit::TestCase
@internal_vm.expects(:network_adapters).returns([nic]).once.in_sequence(update_seq) @internal_vm.expects(:network_adapters).returns([nic]).once.in_sequence(update_seq)
@app.expects(:call).with(@env).once.in_sequence(update_seq) @app.expects(:call).with(@env).once.in_sequence(update_seq)
@env["config"].vm.expects(:customize).yields(@internal_vm) @env["vm.modify"].expects(:call).with() do |proc|
proc.call(@internal_vm)
true
end
@instance.call(@env) @instance.call(@env)
end end

View File

@ -0,0 +1,38 @@
require "test_helper"
class ModifyVMActionTest < Test::Unit::TestCase
setup do
@klass = Vagrant::Action::VM::Modify
@app, @env = action_env
@vm = mock("vm")
@vm.stubs(:ssh).returns(mock("ssh"))
@env["vm"] = @vm
@internal_vm = mock("internal")
@vm.stubs(:vm).returns(@internal_vm)
@instance = @klass.new(@app, @env)
end
context "initialization" do
should "have the vm.modify function setup in the environment" do
assert @env.has_key?("vm.modify")
end
end
context "calling" do
should "run the procs with the VM as an argument and save the VM" do
seq = sequence("procseq")
proc = Proc.new { |vm| }
@env["vm.modify"].call(proc)
proc.expects(:call).with(@internal_vm).once.in_sequence(seq)
@internal_vm.expects(:save).once.in_sequence(seq)
@vm.expects(:reload!).once.in_sequence(seq)
@instance.call(@env)
end
end
end