Create single interal modify block to sync VM modifications
This commit is contained in:
parent
4611800503
commit
6b2feae0e8
|
@ -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
|
||||||
|
|
|
@ -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'
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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
|
Loading…
Reference in New Issue