=begin rdoc
Abstracts out the searchable path for a resource such as a
template file, key, or clouds.rb
NOTE: this is currently _only_ implemented on templates
(PoolParty::Resources::File) and not yet working on clouds.rb or key files.
These will eventually be refactored to use this class.
=end
module SearchablePaths
def self.included(mod)
mod.extend(ClassMethods)
end
module ClassMethods
# Specify that a particular class has methods for searchable paths.
#
# Options:
# * :dirs: array of directories to look in *under* the search paths. (default: ["/"])
# * :dir: set the directory to look in *under* the search paths. Use either dir or dirs, not both. (default: +/+)
# * :paths: overwrite all default paths and set the paths to this array exactly
# * :append_paths: append these paths to any existing paths
# * :prepend_paths: prepend these paths to any existing paths
def has_searchable_paths(opts={})
class_eval do
@searchable_paths_dirs = [opts[:dir]] if opts[:dir]
@searchable_paths_dirs = opts[:dirs] if opts[:dirs]
@paths_override = opts[:paths] if opts[:paths]
@paths_prepend = opts[:prepend_paths] || []
@paths_append = opts[:append_paths] || []
end
extend SearchablePaths::SingletonMethods
include SearchablePaths::InstanceMethods
end
def searchable_paths_dir; @searchable_paths_dirs.first; end
def searchable_paths_dirs
@searchable_paths_dirs && @searchable_paths_dirs.size > 0 ? @searchable_paths_dirs : ["/"]
end
# These are the default search paths in order:
#
# * current working directory (Dir.pwd)
def default_paths
[
Dir.pwd
]
end
# returns the full set of valid searchable paths, given the options
def searchable_paths
return @paths_override if @paths_override && @paths_override.size > 0
@searchable_paths ||= @paths_prepend + default_paths + @paths_append
end
end
module SingletonMethods
end
module InstanceMethods
# Searches for +filepath+ in the searchable_paths iff +filepath+
# doesn't exist. e.g. +filepath+ is interpreted *first* as an absolute
# path, if +filepath+ doesn't exist verbatim then it looks for the file
# in the searchable_paths.
#
# Returns +nil+ if the file cannot be found.
def search_in_known_locations(filepath, additional_search_paths=[])
return filepath if File.exists?(filepath) # return the file if its an absolute path
additional_search_paths.each do |path|
full_path = File.expand_path(path / filepath)
return full_path if File.exists?(full_path)
end
self.class.searchable_paths.each do |path|
self.class.searchable_paths_dirs.each do |dir|
next if path.nil?
full_path = File.expand_path(path / dir / filepath)
return full_path if File.exists?(full_path)
end
end
nil
end
alias_method :find_file, :search_in_known_locations
end
end