lib/puppet/indirector/indirection.rb in puppet-0.25.5 vs lib/puppet/indirector/indirection.rb in puppet-2.6.0

- old
+ new

@@ -5,315 +5,304 @@ # The class that connects functional classes with their different collection # back-ends. Each indirection has a set of associated terminus classes, # each of which is a subclass of Puppet::Indirector::Terminus. class Puppet::Indirector::Indirection - include Puppet::Util::Cacher - include Puppet::Util::Docs + include Puppet::Util::Cacher + include Puppet::Util::Docs - @@indirections = [] + @@indirections = [] - # Find an indirection by name. This is provided so that Terminus classes - # can specifically hook up with the indirections they are associated with. - def self.instance(name) - @@indirections.find { |i| i.name == name } - end + # Find an indirection by name. This is provided so that Terminus classes + # can specifically hook up with the indirections they are associated with. + def self.instance(name) + @@indirections.find { |i| i.name == name } + end - # Return a list of all known indirections. Used to generate the - # reference. - def self.instances - @@indirections.collect { |i| i.name } - end + # Return a list of all known indirections. Used to generate the + # reference. + def self.instances + @@indirections.collect { |i| i.name } + end - # Find an indirected model by name. This is provided so that Terminus classes - # can specifically hook up with the indirections they are associated with. - def self.model(name) - return nil unless match = @@indirections.find { |i| i.name == name } - match.model - end + # Find an indirected model by name. This is provided so that Terminus classes + # can specifically hook up with the indirections they are associated with. + def self.model(name) + return nil unless match = @@indirections.find { |i| i.name == name } + match.model + end - attr_accessor :name, :model + attr_accessor :name, :model - # Create and return our cache terminus. - def cache - raise(Puppet::DevError, "Tried to cache when no cache class was set") unless cache_class - terminus(cache_class) - end + # Create and return our cache terminus. + def cache + raise(Puppet::DevError, "Tried to cache when no cache class was set") unless cache_class + terminus(cache_class) + end - # Should we use a cache? - def cache? - cache_class ? true : false - end + # Should we use a cache? + def cache? + cache_class ? true : false + end - attr_reader :cache_class - # Define a terminus class to be used for caching. - def cache_class=(class_name) - validate_terminus_class(class_name) if class_name - @cache_class = class_name - end + attr_reader :cache_class + # Define a terminus class to be used for caching. + def cache_class=(class_name) + validate_terminus_class(class_name) if class_name + @cache_class = class_name + end - # This is only used for testing. - def delete - @@indirections.delete(self) if @@indirections.include?(self) - end + # This is only used for testing. + def delete + @@indirections.delete(self) if @@indirections.include?(self) + end - # Set the time-to-live for instances created through this indirection. - def ttl=(value) - raise ArgumentError, "Indirection TTL must be an integer" unless value.is_a?(Fixnum) - @ttl = value - end + # Set the time-to-live for instances created through this indirection. + def ttl=(value) + raise ArgumentError, "Indirection TTL must be an integer" unless value.is_a?(Fixnum) + @ttl = value + end - # Default to the runinterval for the ttl. - def ttl - unless defined?(@ttl) - @ttl = Puppet[:runinterval].to_i - end - @ttl - end + # Default to the runinterval for the ttl. + def ttl + @ttl ||= Puppet[:runinterval].to_i + end - # Calculate the expiration date for a returned instance. - def expiration - Time.now + ttl - end + # Calculate the expiration date for a returned instance. + def expiration + Time.now + ttl + end - # Generate the full doc string. - def doc - text = "" + # Generate the full doc string. + def doc + text = "" - if defined? @doc and @doc - text += scrub(@doc) + "\n\n" - end + text += scrub(@doc) + "\n\n" if @doc - if s = terminus_setting() - text += "* **Terminus Setting**: %s" % terminus_setting - end - - text + if s = terminus_setting + text += "* **Terminus Setting**: #{terminus_setting}" end - def initialize(model, name, options = {}) - @model = model - @name = name + text + end - @cache_class = nil - @terminus_class = nil + def initialize(model, name, options = {}) + @model = model + @name = name - raise(ArgumentError, "Indirection %s is already defined" % @name) if @@indirections.find { |i| i.name == @name } - @@indirections << self + @cache_class = nil + @terminus_class = nil - if mod = options[:extend] - extend(mod) - options.delete(:extend) - end + raise(ArgumentError, "Indirection #{@name} is already defined") if @@indirections.find { |i| i.name == @name } + @@indirections << self - # This is currently only used for cache_class and terminus_class. - options.each do |name, value| - begin - send(name.to_s + "=", value) - rescue NoMethodError - raise ArgumentError, "%s is not a valid Indirection parameter" % name - end - end + if mod = options[:extend] + extend(mod) + options.delete(:extend) end - # Set up our request object. - def request(method, key, arguments = nil) - Puppet::Indirector::Request.new(self.name, method, key, arguments) + # This is currently only used for cache_class and terminus_class. + options.each do |name, value| + begin + send(name.to_s + "=", value) + rescue NoMethodError + raise ArgumentError, "#{name} is not a valid Indirection parameter" + end end + end - # Return the singleton terminus for this indirection. - def terminus(terminus_name = nil) - # Get the name of the terminus. - unless terminus_name ||= terminus_class - raise Puppet::DevError, "No terminus specified for %s; cannot redirect" % self.name - end + # Set up our request object. + def request(*args) + Puppet::Indirector::Request.new(self.name, *args) + end - return termini[terminus_name] ||= make_terminus(terminus_name) - end + # Return the singleton terminus for this indirection. + def terminus(terminus_name = nil) + # Get the name of the terminus. + raise Puppet::DevError, "No terminus specified for #{self.name}; cannot redirect" unless terminus_name ||= terminus_class - # This can be used to select the terminus class. - attr_accessor :terminus_setting + termini[terminus_name] ||= make_terminus(terminus_name) + end - # Determine the terminus class. - def terminus_class - unless @terminus_class - if setting = self.terminus_setting - self.terminus_class = Puppet.settings[setting].to_sym - else - raise Puppet::DevError, "No terminus class nor terminus setting was provided for indirection %s" % self.name - end - end - @terminus_class - end + # This can be used to select the terminus class. + attr_accessor :terminus_setting - def reset_terminus_class - @terminus_class = nil + # Determine the terminus class. + def terminus_class + unless @terminus_class + if setting = self.terminus_setting + self.terminus_class = Puppet.settings[setting].to_sym + else + raise Puppet::DevError, "No terminus class nor terminus setting was provided for indirection #{self.name}" + end end + @terminus_class + end - # Specify the terminus class to use. - def terminus_class=(klass) - validate_terminus_class(klass) - @terminus_class = klass - end + def reset_terminus_class + @terminus_class = nil + end - # This is used by terminus_class= and cache=. - def validate_terminus_class(terminus_class) - unless terminus_class and terminus_class.to_s != "" - raise ArgumentError, "Invalid terminus name %s" % terminus_class.inspect - end - unless Puppet::Indirector::Terminus.terminus_class(self.name, terminus_class) - raise ArgumentError, "Could not find terminus %s for indirection %s" % [terminus_class, self.name] - end + # Specify the terminus class to use. + def terminus_class=(klass) + validate_terminus_class(klass) + @terminus_class = klass + end + + # This is used by terminus_class= and cache=. + def validate_terminus_class(terminus_class) + raise ArgumentError, "Invalid terminus name #{terminus_class.inspect}" unless terminus_class and terminus_class.to_s != "" + unless Puppet::Indirector::Terminus.terminus_class(self.name, terminus_class) + raise ArgumentError, "Could not find terminus #{terminus_class} for indirection #{self.name}" end + end - # Expire a cached object, if one is cached. Note that we don't actually - # remove it, we expire it and write it back out to disk. This way people - # can still use the expired object if they want. - def expire(key, *args) - request = request(:expire, key, *args) + # Expire a cached object, if one is cached. Note that we don't actually + # remove it, we expire it and write it back out to disk. This way people + # can still use the expired object if they want. + def expire(key, *args) + request = request(:expire, key, *args) - return nil unless cache? + return nil unless cache? - return nil unless instance = cache.find(request(:find, key, *args)) + return nil unless instance = cache.find(request(:find, key, *args)) - Puppet.info "Expiring the %s cache of %s" % [self.name, instance.name] + Puppet.info "Expiring the #{self.name} cache of #{instance.name}" - # Set an expiration date in the past - instance.expiration = Time.now - 60 + # Set an expiration date in the past + instance.expiration = Time.now - 60 - cache.save(request(:save, instance, *args)) - end + cache.save(request(:save, instance, *args)) + end - # Search for an instance in the appropriate terminus, caching the - # results if caching is configured.. - def find(key, *args) - request = request(:find, key, *args) - terminus = prepare(request) + # Search for an instance in the appropriate terminus, caching the + # results if caching is configured.. + def find(key, *args) + request = request(:find, key, *args) + terminus = prepare(request) - begin - if result = find_in_cache(request) - return result - end - rescue => detail - puts detail.backtrace if Puppet[:trace] - Puppet.err "Cached %s for %s failed: %s" % [self.name, request.key, detail] - end + begin + if result = find_in_cache(request) + return result + end + rescue => detail + puts detail.backtrace if Puppet[:trace] + Puppet.err "Cached #{self.name} for #{request.key} failed: #{detail}" + end - # Otherwise, return the result from the terminus, caching if appropriate. - if ! request.ignore_terminus? and result = terminus.find(request) - result.expiration ||= self.expiration - if cache? and request.use_cache? - Puppet.info "Caching %s for %s" % [self.name, request.key] - cache.save request(:save, result, *args) - end + # Otherwise, return the result from the terminus, caching if appropriate. + if ! request.ignore_terminus? and result = terminus.find(request) + result.expiration ||= self.expiration + if cache? and request.use_cache? + Puppet.info "Caching #{self.name} for #{request.key}" + cache.save request(:save, result, *args) + end - return terminus.respond_to?(:filter) ? terminus.filter(result) : result - end - - return nil + return terminus.respond_to?(:filter) ? terminus.filter(result) : result end - def find_in_cache(request) - # See if our instance is in the cache and up to date. - return nil unless cache? and ! request.ignore_cache? and cached = cache.find(request) - if cached.expired? - Puppet.info "Not using expired %s for %s from cache; expired at %s" % [self.name, request.key, cached.expiration] - return nil - end + nil + end - Puppet.debug "Using cached %s for %s" % [self.name, request.key] - return cached + def find_in_cache(request) + # See if our instance is in the cache and up to date. + return nil unless cache? and ! request.ignore_cache? and cached = cache.find(request) + if cached.expired? + Puppet.info "Not using expired #{self.name} for #{request.key} from cache; expired at #{cached.expiration}" + return nil end - # Remove something via the terminus. - def destroy(key, *args) - request = request(:destroy, key, *args) - terminus = prepare(request) + Puppet.debug "Using cached #{self.name} for #{request.key}" + cached + end - result = terminus.destroy(request) + # Remove something via the terminus. + def destroy(key, *args) + request = request(:destroy, key, *args) + terminus = prepare(request) - if cache? and cached = cache.find(request(:find, key, *args)) - # Reuse the existing request, since it's equivalent. - cache.destroy(request) - end + result = terminus.destroy(request) - result + if cache? and cached = cache.find(request(:find, key, *args)) + # Reuse the existing request, since it's equivalent. + cache.destroy(request) end - # Search for more than one instance. Should always return an array. - def search(key, *args) - request = request(:search, key, *args) - terminus = prepare(request) + result + end - if result = terminus.search(request) - raise Puppet::DevError, "Search results from terminus %s are not an array" % terminus.name unless result.is_a?(Array) - result.each do |instance| - instance.expiration ||= self.expiration - end - return result - end + # Search for more than one instance. Should always return an array. + def search(key, *args) + request = request(:search, key, *args) + terminus = prepare(request) + + if result = terminus.search(request) + raise Puppet::DevError, "Search results from terminus #{terminus.name} are not an array" unless result.is_a?(Array) + result.each do |instance| + instance.expiration ||= self.expiration + end + return result end + end - # Save the instance in the appropriate terminus. This method is - # normally an instance method on the indirected class. - def save(instance, *args) - request = request(:save, instance, *args) - terminus = prepare(request) + # Save the instance in the appropriate terminus. This method is + # normally an instance method on the indirected class. + def save(key, instance = nil) + request = request(:save, key, instance) + terminus = prepare(request) - result = terminus.save(request) + result = terminus.save(request) - # If caching is enabled, save our document there - cache.save(request) if cache? + # If caching is enabled, save our document there + cache.save(request) if cache? - result - end + result + end - private + private - # Check authorization if there's a hook available; fail if there is one - # and it returns false. - def check_authorization(request, terminus) - # At this point, we're assuming authorization makes no sense without - # client information. - return unless request.node + # Check authorization if there's a hook available; fail if there is one + # and it returns false. + def check_authorization(request, terminus) + # At this point, we're assuming authorization makes no sense without + # client information. + return unless request.node - # This is only to authorize via a terminus-specific authorization hook. - return unless terminus.respond_to?(:authorized?) + # This is only to authorize via a terminus-specific authorization hook. + return unless terminus.respond_to?(:authorized?) - unless terminus.authorized?(request) - msg = "Not authorized to call %s on %s" % [request.method, request.to_s] - unless request.options.empty? - msg += " with %s" % request.options.inspect - end - raise ArgumentError, msg - end + unless terminus.authorized?(request) + msg = "Not authorized to call #{request.method} on #{request}" + msg += " with #{request.options.inspect}" unless request.options.empty? + raise ArgumentError, msg end + end - # Setup a request, pick the appropriate terminus, check the request's authorization, and return it. - def prepare(request) - # Pick our terminus. - if respond_to?(:select_terminus) - unless terminus_name = select_terminus(request) - raise ArgumentError, "Could not determine appropriate terminus for %s" % request - end - else - terminus_name = terminus_class - end + # Setup a request, pick the appropriate terminus, check the request's authorization, and return it. + def prepare(request) + # Pick our terminus. + if respond_to?(:select_terminus) + unless terminus_name = select_terminus(request) + raise ArgumentError, "Could not determine appropriate terminus for #{request}" + end + else + terminus_name = terminus_class + end - dest_terminus = terminus(terminus_name) - check_authorization(request, dest_terminus) + dest_terminus = terminus(terminus_name) + check_authorization(request, dest_terminus) - return dest_terminus - end + dest_terminus + end - # Create a new terminus instance. - def make_terminus(terminus_class) - # Load our terminus class. - unless klass = Puppet::Indirector::Terminus.terminus_class(self.name, terminus_class) - raise ArgumentError, "Could not find terminus %s for indirection %s" % [terminus_class, self.name] - end - return klass.new + # Create a new terminus instance. + def make_terminus(terminus_class) + # Load our terminus class. + unless klass = Puppet::Indirector::Terminus.terminus_class(self.name, terminus_class) + raise ArgumentError, "Could not find terminus #{terminus_class} for indirection #{self.name}" end + klass.new + end - # Cache our terminus instances indefinitely, but make it easy to clean them up. - cached_attr(:termini) { Hash.new } + # Cache our terminus instances indefinitely, but make it easy to clean them up. + cached_attr(:termini) { Hash.new } end