1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
|
# Autoload paths, either based on names or all at once.
class Puppet::Util::Autoload
include Puppet::Util
@autoloaders = {}
@loaded = []
class << self
attr_reader :autoloaders
private :autoloaders
end
# Send [], []=, and :clear to the @autloaders hash
Puppet::Util.classproxy self, :autoloaders, "[]", "[]="
# List all loaded files.
def self.list_loaded
@loaded.sort { |a,b| a[0] <=> b[0] }.collect do |path, hash|
"%s: %s" % [path, hash[:file]]
end
end
# Has a given path been loaded? This is used for testing whether a
# changed file should be loaded or just ignored. This is only
# used in network/client/master, when downloading plugins, to
# see if a given plugin is currently loaded and thus should be
# reloaded.
def self.loaded?(path)
path = path.to_s.sub(/\.rb$/, '')
@loaded.include?(path)
end
# Save the fact that a given path has been loaded. This is so
# we can load downloaded plugins if they've already been loaded
# into memory.
def self.loaded(file)
@loaded << file unless @loaded.include?(file)
end
attr_accessor :object, :path, :objwarn, :wrap
def initialize(obj, path, options = {})
@path = path.to_s
if @path !~ /^\w/
raise ArgumentError, "Autoload paths cannot be fully qualified"
end
@object = obj
self.class[obj] = self
options.each do |opt, value|
opt = opt.intern if opt.is_a? String
begin
self.send(opt.to_s + "=", value)
rescue NoMethodError
raise ArgumentError, "%s is not a valid option" % opt
end
end
unless defined? @wrap
@wrap = true
end
end
# Load a single plugin by name. We use 'load' here so we can reload a
# given plugin.
def load(name)
path = name.to_s + ".rb"
eachdir do |dir|
file = File.join(dir, path)
next unless FileTest.exists?(file)
begin
Kernel.load file, @wrap
name = symbolize(name)
loaded name, file
return true
rescue LoadError => detail
# I have no idea what's going on here, but different versions
# of ruby are raising different errors on missing files.
unless detail.to_s =~ /^no such file/i
warn "Could not autoload %s: %s" % [name, detail]
if Puppet[:trace]
puts detail.backtrace
end
end
return false
end
end
return false
end
# Mark the named object as loaded. Note that this supports unqualified
# queries, while we store the result as a qualified query in the class.
def loaded(name, file)
self.class.loaded(File.join(@path, name.to_s))
end
# Indicate whether the specfied plugin has been loaded.
def loaded?(name)
self.class.loaded?(File.join(@path, name.to_s))
end
# Load all instances that we can. This uses require, rather than load,
# so that already-loaded files don't get reloaded unnecessarily.
def loadall
# Load every instance of everything we can find.
eachdir do |dir|
Dir.glob("#{dir}/*.rb").each do |file|
name = File.basename(file).sub(".rb", '').intern
next if loaded?(name)
rubypath = File.join(@path, name.to_s)
begin
Kernel.require rubypath
loaded(name, file)
rescue => detail
if Puppet[:trace]
puts detail.backtrace
end
warn "Could not autoload %s: %s" % [file.inspect, detail]
end
end
end
end
private
# Yield each subdir in turn.
def eachdir
searchpath.each do |dir|
subdir = File.join(dir, @path)
yield subdir if FileTest.directory?(subdir)
end
end
# The list of directories to search through for loadable plugins.
def searchpath
# JJM: Search for optional lib directories in each module bundle.
module_lib_dirs = Puppet[:modulepath].split(":").collect do |d|
Dir.glob("%s/*/lib" % d).select do |f|
FileTest.directory?(f)
end
end.flatten
[module_lib_dirs, Puppet[:libdir], $:].flatten
end
end
|