Change caching behavior of config procs for the config loader

This commit is contained in:
Mitchell Hashimoto 2012-01-11 16:58:40 -08:00
parent 43a9f0928a
commit 7c9c6e34ce
2 changed files with 28 additions and 21 deletions

View File

@ -33,16 +33,29 @@ module Vagrant
# At this point, no configuration is actually loaded. Note that calling # At this point, no configuration is actually loaded. Note that calling
# `set` multiple times with the same name will override any previously # `set` multiple times with the same name will override any previously
# set values. In this way, the last set data for a given name wins. # set values. In this way, the last set data for a given name wins.
def set(name, data) def set(name, sources)
@logger.debug("Set #{name.inspect} = #{data.inspect}") @logger.debug("Set #{name.inspect} = #{sources.inspect}")
# Make all sources an array. # Sources should be an array
data = [data] if !data.kind_of?(Array) sources = [sources] if !sources.kind_of?(Array)
# Clear the cache for this key only if the data is different # Gather the procs for every source, since that is what we care about.
@proc_cache.delete(name) if @sources[name] && @sources[name] != data procs = []
sources.each do |source|
if !@proc_cache.has_key?(source)
# Load the procs for this source and cache them. This caching
# avoids the issue where a file may have side effects when loading
# and loading it multiple times causes unexpected behavior.
@logger.debug("Populating proc cache for #{source.inspect}")
@proc_cache[source] = procs_for_source(source)
end
@sources[name] = data # Add on to the array of procs we're going to use
procs.concat(@proc_cache[source])
end
# Set this source by name.
@sources[name] = procs
end end
# This loads the configured sources in the configured order and returns # This loads the configured sources in the configured order and returns
@ -62,22 +75,13 @@ module Vagrant
@load_order.each do |key| @load_order.each do |key|
next if !@sources.has_key?(key) next if !@sources.has_key?(key)
@sources[key].each do |source| @sources[key].each do |proc|
@logger.debug("Loading from: #{key}") @logger.debug("Loading from: #{key}")
# Load the procs, caching them by the source key. This makes
# sure that files are only loaded once, for example. The reason
# for this is because people may put side-effect code in their
# Vagrantfiles. This assures that the side effect only occurs
# once (which is what they expect)
@proc_cache[key] ||= procs_for_source(source)
# Call each proc with the top-level configuration. # Call each proc with the top-level configuration.
@proc_cache[key].each do |proc|
proc.call(top) proc.call(top)
end end
end end
end
@logger.debug("Configuration loaded successfully") @logger.debug("Configuration loaded successfully")
top top

View File

@ -25,8 +25,11 @@ describe Vagrant::Config::Loader do
it "should only load configuration files once" do it "should only load configuration files once" do
$_config_data = 0 $_config_data = 0
# We test both setting a file multiple times as well as multiple
# loads, since both should not cache the data.
file = temporary_file("$_config_data += 1")
instance.load_order = [:file] instance.load_order = [:file]
instance.set(:file, temporary_file("$_config_data += 1")) 5.times { instance.set(:file, file) }
5.times { instance.load } 5.times { instance.load }
$_config_data.should == 1 $_config_data.should == 1
@ -62,7 +65,7 @@ describe Vagrant::Config::Loader do
it "should raise proper error if there is a syntax error in a Vagrantfile" do it "should raise proper error if there is a syntax error in a Vagrantfile" do
instance.load_order = [:file] instance.load_order = [:file]
instance.set(:file, temporary_file("Vagrant:^Config")) expect { instance.set(:file, temporary_file("Vagrant:^Config")) }.
expect { instance.load }.to raise_exception(Vagrant::Errors::VagrantfileSyntaxError) to raise_exception(Vagrant::Errors::VagrantfileSyntaxError)
end end
end end