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
|
module Oxidized
class HookManager
class << self
def from_config cfg
mgr = new
cfg.hooks.each do |name,h_cfg|
h_cfg.events.each do |event|
mgr.register event.to_sym, name, h_cfg.type, h_cfg
end
end
mgr
end
end
# HookContext is passed to each hook. It can contain anything related to the
# event in question. At least it contains the event name
class HookContext < OpenStruct; end
# RegisteredHook is a container for a Hook instance
class RegisteredHook < Struct.new(:name, :hook); end
Events = [
:node_success,
:node_fail,
:post_store,
]
attr_reader :registered_hooks
def initialize
@registered_hooks = Hash.new {|h,k| h[k] = []}
end
def register event, name, hook_type, cfg
unless Events.include? event
raise ArgumentError,
"unknown event #{event}, available: #{Events.join ','}"
end
Oxidized.mgr.add_hook hook_type
begin
hook = Oxidized.mgr.hook.fetch(hook_type).new
rescue KeyError
raise KeyError, "cannot find hook #{hook_type.inspect}"
end
hook.cfg = cfg
@registered_hooks[event] << RegisteredHook.new(name, hook)
Log.debug "Hook #{name.inspect} registered #{hook.class} for event #{event.inspect}"
end
def handle event, ctx_params={}
ctx = HookContext.new ctx_params
ctx.event = event
@registered_hooks[event].each do |r_hook|
begin
r_hook.hook.run_hook ctx
rescue => e
Log.error "Hook #{r_hook.name} (#{r_hook.hook}) failed " +
"(#{e.inspect}) for event #{event.inspect}"
end
end
end
end
# Hook abstract base class
class Hook
attr_accessor :cfg
def initialize
end
def cfg=(cfg)
@cfg = cfg
validate_cfg! if self.respond_to? :validate_cfg!
end
def run_hook ctx
raise NotImplementedError
end
def log(msg, level=:info)
Log.send(level, "#{self.class.name}: #{msg}")
end
end
end
|