summaryrefslogtreecommitdiffstats
path: root/lib/puppet/rails
diff options
context:
space:
mode:
authorBrice Figureau <brice-puppet@daysofwonder.com>2009-01-25 20:54:01 +0100
committerJames Turnbull <james@lovedthanlost.net>2009-02-06 21:54:33 +1100
commitf01882da6284d61312016d7de602af65da6d5731 (patch)
tree6d117b27dc8f1d96489bca4dee9c38b2b43f7f76 /lib/puppet/rails
parentb7ab54c7f094c34b9f80224a63521a2873d7c1c1 (diff)
downloadpuppet-f01882da6284d61312016d7de602af65da6d5731.tar.gz
puppet-f01882da6284d61312016d7de602af65da6d5731.tar.xz
puppet-f01882da6284d61312016d7de602af65da6d5731.zip
Change the way the tags and params are handled in rails
The rationale behind this patch is that it takes a lots of time to let rails unserialize the ParamValue and ResourceTag object on each compilation, just to throw them away the second after. The idea is to fetch directly (and batched host per host) the parameters and tags from the database and then returns them as hash. This allows the no-modification case to takes at least 2 times less than before. Signed-off-by: Brice Figureau <brice-puppet@daysofwonder.com>
Diffstat (limited to 'lib/puppet/rails')
-rw-r--r--lib/puppet/rails/host.rb69
-rw-r--r--lib/puppet/rails/resource.rb42
2 files changed, 88 insertions, 23 deletions
diff --git a/lib/puppet/rails/host.rb b/lib/puppet/rails/host.rb
index 187dc657a..a429cbfb1 100644
--- a/lib/puppet/rails/host.rb
+++ b/lib/puppet/rails/host.rb
@@ -3,7 +3,7 @@ require 'puppet/rails/fact_name'
require 'puppet/rails/source_file'
require 'puppet/util/rails/collection_merger'
-# Puppet::TIME_DEBUG = true
+Puppet::TIME_DEBUG = true
class Puppet::Rails::Host < ActiveRecord::Base
include Puppet::Util
@@ -12,9 +12,7 @@ class Puppet::Rails::Host < ActiveRecord::Base
has_many :fact_values, :dependent => :destroy
has_many :fact_names, :through => :fact_values
belongs_to :source_file
- has_many :resources,
- :include => :param_values,
- :dependent => :destroy
+ has_many :resources, :dependent => :destroy
# If the host already exists, get rid of its objects
def self.clean(host)
@@ -119,25 +117,49 @@ class Puppet::Rails::Host < ActiveRecord::Base
def setresources(list)
existing = nil
seconds = Benchmark.realtime {
+ existing = find_resources()
+ }
+ Puppet.notice("Searched for resources in %0.2f seconds" % seconds) if defined?(Puppet::TIME_DEBUG)
- # Preload the parameters with the resource query, but not the tags, since doing so makes the query take about 10x longer.
- # I've left the other queries in so that it's straightforward to switch between them for testing, if we so desire.
- #existing = resources.find(:all, :include => [{:param_values => :param_name, :resource_tags => :puppet_tag}, :source_file]).inject({}) do | hash, resource |
- #existing = resources.find(:all, :include => [{:resource_tags => :puppet_tag}, :source_file]).inject({}) do | hash, resource |
- existing = resources.find(:all, :include => [{:param_values => :param_name}, :source_file]).inject({}) do | hash, resource |
- hash[resource.ref] = resource
- hash
- end
+ seconds = Benchmark.realtime {
+ find_resources_parameters_tags(existing)
+ } if id
+ Puppet.notice("Searched for resource params and tags in %0.2f seconds" % seconds) if defined?(Puppet::TIME_DEBUG)
+
+ seconds = Benchmark.realtime {
+ compare_to_catalog(existing, list)
}
+ Puppet.notice("Resource comparison took %0.2f seconds" % seconds) if defined?(Puppet::TIME_DEBUG)
+ end
- Puppet.notice("Searched for resources in %0.2f seconds" % seconds) if defined?(Puppet::TIME_DEBUG)
+ def find_resources
+ resources.find(:all, :include => :source_file).inject({}) do | hash, resource |
+ hash[resource.ref] = resource
+ hash
+ end
+ end
+
+ def find_resources_parameters_tags(resources)
+ # initialize all resource parameters
+ resources.each do |key,resource|
+ resource.params_hash = []
+ end
+ resources_by_id = resources.inject({}) do |hash, res|
+ hash[res[1]['id']] = res[1]
+ hash
+ end
+
+ find_resources_parameters(resources_by_id)
+ find_resources_tags(resources_by_id)
+ end
+
+ def compare_to_catalog(resources, list)
compiled = list.inject({}) do |hash, resource|
hash[resource.ref] = resource
hash
end
-
- ar_hash_merge(existing, compiled,
+ ar_hash_merge(resources, compiled,
:create => Proc.new { |ref, resource|
resource.to_rails(self)
}, :delete => Proc.new { |resource|
@@ -147,6 +169,23 @@ class Puppet::Rails::Host < ActiveRecord::Base
})
end
+ def find_resources_parameters(resources)
+ params = Puppet::Rails::ParamValue.find_all_params_from_host(self)
+
+ # assign each loaded parameters/tags to the resource it belongs to
+ params.each do |param|
+ resources[param['resource_id']].add_param_to_hash(param)
+ end
+ end
+
+ def find_resources_tags(resources)
+ tags = Puppet::Rails::ResourceTag.find_all_tags_from_host(self)
+
+ tags.each do |tag|
+ resources[tag['resource_id']].add_tag_to_hash(tag)
+ end
+ end
+
def update_connect_time
self.last_connect = Time.now
save
diff --git a/lib/puppet/rails/resource.rb b/lib/puppet/rails/resource.rb
index 255b0e788..bd2739d53 100644
--- a/lib/puppet/rails/resource.rb
+++ b/lib/puppet/rails/resource.rb
@@ -5,6 +5,7 @@ require 'puppet/util/rails/collection_merger'
class Puppet::Rails::Resource < ActiveRecord::Base
include Puppet::Util::CollectionMerger
+ include Puppet::Util::ReferenceSerializer
has_many :param_values, :dependent => :destroy, :class_name => "Puppet::Rails::ParamValue"
has_many :param_names, :through => :param_values, :class_name => "Puppet::Rails::ParamName"
@@ -32,21 +33,46 @@ class Puppet::Rails::Resource < ActiveRecord::Base
self.source_file = Puppet::Rails::SourceFile.find_or_create_by_filename(file)
end
+ def title
+ unserialize_value(self[:title])
+ end
+
+ def add_param_to_hash(param)
+ @params_hash ||= []
+ @params_hash << param
+ end
+
+ def add_tag_to_hash(tag)
+ @tags_hash ||= []
+ @tags_hash << tag
+ end
+
+ def params_hash=(hash)
+ @params_hash = hash
+ end
+
+ def tags_hash=(hash)
+ @tags_hash = hash
+ end
+
# returns a hash of param_names.name => [param_values]
def get_params_hash(values = nil)
- values ||= param_values.find(:all, :include => :param_name)
- values.inject({}) do | hash, value |
- hash[value.param_name.name] ||= []
- hash[value.param_name.name] << value
+ values ||= @params_hash || Puppet::Rails::ParamValues.find_all_params_from_resource(id)
+ if values.size == 0
+ return {}
+ end
+ values.inject({}) do |hash, value|
+ hash[value['name']] ||= []
+ hash[value['name']] << value
hash
end
end
-
+
def get_tag_hash(tags = nil)
- tags ||= resource_tags.find(:all, :include => :puppet_tag)
+ tags ||= @tags_hash || Puppet::Rails::ResourceTag.find_all_tags_from_resource(id)
return tags.inject({}) do |hash, tag|
# We have to store the tag object, not just the tag name.
- hash[tag.puppet_tag.name] = tag
+ hash[tag['name']] = tag
hash
end
end
@@ -82,7 +108,7 @@ class Puppet::Rails::Resource < ActiveRecord::Base
end
def ref
- "%s[%s]" % [self[:restype].split("::").collect { |s| s.capitalize }.join("::"), self[:title]]
+ "%s[%s]" % [self[:restype].split("::").collect { |s| s.capitalize }.join("::"), self.title.to_s]
end
# Convert our object to a resource. Do not retain whether the object