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
|
#
# Author:: Seth Chisamore <schisamo@chef.io>
# Copyright:: 2011-2018, Chef Software, Inc <legal@chef.io>
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
require_relative "../resource"
require_relative "../dist"
class Chef
class Resource
class ChefHandler < Chef::Resource
resource_name :chef_handler
provides(:chef_handler) { true }
description "Use the chef_handler resource to install or uninstall reporting/exception handlers."
introduced "14.0"
property :class_name, String,
description: "The name of the handler class. This can be module name-spaced.",
name_property: true
property :source, String,
description: "The full path to the handler file. Can also be a gem path if the handler ships as part of a Ruby gem."
property :arguments, [Array, Hash],
description: "Arguments to pass the handler's class initializer.",
default: lazy { [] }
property :type, Hash,
description: "The type of handler to register as, i.e. :report, :exception or both.",
default: { report: true, exception: true }
# supports means a different thing in chef-land so we renamed it but
# wanted to make sure we didn't break the world
alias_method :supports, :type
# This action needs to find an rb file that presumably contains the indicated class in it and the
# load that file. It then instantiates that class by name and registers it as a handler.
action :enable do
description "Enables the handler for the current #{Chef::Dist::PRODUCT} run on the current node"
class_name = new_resource.class_name
new_resource.type.each do |type, enable|
next unless enable
unregister_handler(type, class_name)
end
handler = nil
require new_resource.source unless new_resource.source.nil?
_, klass = get_class(class_name)
handler = klass.send(:new, *collect_args(new_resource.arguments))
new_resource.type.each do |type, enable|
next unless enable
register_handler(type, handler)
end
end
action :disable do
description "Disables the handler for the current #{Chef::Dist::PRODUCT} run on the current node"
new_resource.type.each_key do |type|
unregister_handler(type, new_resource.class_name)
end
end
action_class do
# Registers a handler in Chef::Config.
#
# @param handler_type [Symbol] such as :report or :exception.
# @param handler [Chef::Handler] handler to register.
def register_handler(handler_type, handler)
Chef::Log.info("Enabling #{handler.class.name} as a #{handler_type} handler.")
Chef::Config.send("#{handler_type}_handlers") << handler
end
# Removes all handlers that match the given class name in Chef::Config.
#
# @param handler_type [Symbol] such as :report or :exception.
# @param class_full_name [String] such as 'Chef::Handler::ErrorReport'.
#
# @return [void]
def unregister_handler(handler_type, class_full_name)
Chef::Config.send("#{handler_type}_handlers").delete_if do |v|
# avoid a bit of log spam
if v.class.name == class_full_name
Chef::Log.info("Disabling #{class_full_name} as a #{handler_type} handler.")
true
end
end
end
# Walks down the namespace heirarchy to return the class object for the given class name.
# If the class is not available, NameError is thrown.
#
# @param class_full_name [String] full class name such as 'Chef::Handler::Foo' or 'MyHandler'.
#
# @return [Array] parent class and child class.
def get_class(class_full_name)
ancestors = class_full_name.split("::")
class_name = ancestors.pop
# We need to search the ancestors only for the first/uppermost namespace of the class, so we
# need to enable the #const_get inherit paramenter only when we are searching in Kernel scope
# (see COOK-4117).
parent = ancestors.inject(Kernel) { |scope, const_name| scope.const_get(const_name, scope === Kernel) }
child = parent.const_get(class_name, parent === Kernel)
[parent, child]
end
def collect_args(resource_args = [])
if resource_args.is_a? Array
resource_args
else
[resource_args]
end
end
end
end
end
end
|