oxidized 0.21.0 → 0.22.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +5 -5
- data/.github/no-response.yml +13 -0
- data/.gitignore +3 -0
- data/.rubocop.yml +37 -0
- data/.rubocop_todo.yml +714 -0
- data/.travis.yml +7 -1
- data/CHANGELOG.md +341 -243
- data/Dockerfile +44 -16
- data/LICENSE +201 -0
- data/README.md +114 -82
- data/Rakefile +19 -0
- data/TODO.md +29 -23
- data/bin/oxidized +1 -2
- data/docs/Configuration.md +71 -31
- data/docs/Creating-Models.md +78 -0
- data/docs/Hooks.md +145 -41
- data/docs/Model-Notes/AireOS.md +12 -0
- data/docs/Model-Notes/ArbOS.md +12 -0
- data/docs/Model-Notes/Comware.md +14 -0
- data/docs/Model-Notes/EOS.md +9 -0
- data/docs/Model-Notes/JunOS.md +34 -0
- data/docs/Model-Notes/Netgear.md +68 -0
- data/docs/Model-Notes/README.md +19 -0
- data/docs/{VRP-Huawei.md → Model-Notes/VRP-Huawei.md} +10 -2
- data/docs/Model-Notes/XGS4600-Zyxel.md +39 -0
- data/docs/Outputs.md +27 -28
- data/docs/Ruby-API.md +38 -18
- data/docs/Sources.md +78 -16
- data/docs/Supported-OS-Types.md +171 -148
- data/extra/oxidized.logrotate +7 -0
- data/extra/oxidized.service +1 -1
- data/extra/rest_client.rb +4 -5
- data/extra/syslog.rb +16 -16
- data/lib/oxidized/cli.rb +3 -3
- data/lib/oxidized/config.rb +7 -4
- data/lib/oxidized/core.rb +3 -3
- data/lib/oxidized/hook.rb +64 -65
- data/lib/oxidized/hook/awssns.rb +2 -3
- data/lib/oxidized/hook/ciscosparkdiff.rb +49 -0
- data/lib/oxidized/hook/exec.rb +5 -5
- data/lib/oxidized/hook/githubrepo.rb +20 -14
- data/lib/oxidized/hook/slackdiff.rb +38 -19
- data/lib/oxidized/hook/xmppdiff.rb +58 -0
- data/lib/oxidized/input/cli.rb +5 -6
- data/lib/oxidized/input/ftp.rb +8 -7
- data/lib/oxidized/input/http.rb +39 -0
- data/lib/oxidized/input/ssh.rb +24 -22
- data/lib/oxidized/input/telnet.rb +38 -32
- data/lib/oxidized/jobs.rb +3 -4
- data/lib/oxidized/manager.rb +9 -4
- data/lib/oxidized/model/acos.rb +15 -16
- data/lib/oxidized/model/acsw.rb +3 -8
- data/lib/oxidized/model/aen.rb +1 -2
- data/lib/oxidized/model/aireos.rb +3 -5
- data/lib/oxidized/model/alteonos.rb +16 -18
- data/lib/oxidized/model/alvarion.rb +0 -4
- data/lib/oxidized/model/aos.rb +2 -4
- data/lib/oxidized/model/aos7.rb +2 -3
- data/lib/oxidized/model/aosw.rb +13 -15
- data/lib/oxidized/model/apc_aos.rb +0 -3
- data/lib/oxidized/model/arbos.rb +26 -0
- data/lib/oxidized/model/aricentiss.rb +51 -0
- data/lib/oxidized/model/asa.rb +33 -35
- data/lib/oxidized/model/asyncos.rb +41 -44
- data/lib/oxidized/model/audiocodes.rb +4 -8
- data/lib/oxidized/model/awplus.rb +84 -0
- data/lib/oxidized/model/boss.rb +6 -5
- data/lib/oxidized/model/br6910.rb +43 -45
- data/lib/oxidized/model/c4cmts.rb +3 -5
- data/lib/oxidized/model/cambium.rb +22 -0
- data/lib/oxidized/model/catos.rb +0 -2
- data/lib/oxidized/model/cisconga.rb +1 -3
- data/lib/oxidized/model/ciscosma.rb +37 -40
- data/lib/oxidized/model/ciscosmb.rb +7 -4
- data/lib/oxidized/model/comnetms.rb +43 -0
- data/lib/oxidized/model/comware.rb +9 -9
- data/lib/oxidized/model/coriant8600.rb +3 -5
- data/lib/oxidized/model/coriantgroove.rb +3 -5
- data/lib/oxidized/model/corianttmos.rb +1 -3
- data/lib/oxidized/model/cumulus.rb +26 -32
- data/lib/oxidized/model/datacom.rb +0 -2
- data/lib/oxidized/model/dcnos.rb +46 -0
- data/lib/oxidized/model/dlink.rb +1 -1
- data/lib/oxidized/model/dnos.rb +9 -5
- data/lib/oxidized/model/edgecos.rb +45 -0
- data/lib/oxidized/model/edgeos.rb +5 -3
- data/lib/oxidized/model/edgeswitch.rb +1 -3
- data/lib/oxidized/model/enterasys.rb +1 -3
- data/lib/oxidized/model/eos.rb +6 -8
- data/lib/oxidized/model/fabricos.rb +3 -5
- data/lib/oxidized/model/firewareos.rb +2 -5
- data/lib/oxidized/model/fortios.rb +21 -17
- data/lib/oxidized/model/ftos.rb +2 -4
- data/lib/oxidized/model/fujitsupy.rb +2 -4
- data/lib/oxidized/model/gaiaos.rb +6 -10
- data/lib/oxidized/model/gcombnps.rb +82 -0
- data/lib/oxidized/model/hatteras.rb +8 -5
- data/lib/oxidized/model/hirschmann.rb +8 -10
- data/lib/oxidized/model/hpebladesystem.rb +19 -17
- data/lib/oxidized/model/hpemsa.rb +0 -3
- data/lib/oxidized/model/ios.rb +54 -55
- data/lib/oxidized/model/iosxe.rb +5 -0
- data/lib/oxidized/model/iosxr.rb +1 -3
- data/lib/oxidized/model/ipos.rb +1 -3
- data/lib/oxidized/model/ironware.rb +12 -15
- data/lib/oxidized/model/isam.rb +4 -5
- data/lib/oxidized/model/junos.rb +8 -7
- data/lib/oxidized/model/masteros.rb +1 -3
- data/lib/oxidized/model/mlnxos.rb +3 -4
- data/lib/oxidized/model/model.rb +15 -7
- data/lib/oxidized/model/mtrlrfs.rb +1 -4
- data/lib/oxidized/model/ndms.rb +24 -0
- data/lib/oxidized/model/netgear.rb +3 -4
- data/lib/oxidized/model/netscaler.rb +0 -2
- data/lib/oxidized/model/nos.rb +1 -3
- data/lib/oxidized/model/nxos.rb +13 -3
- data/lib/oxidized/model/oneos.rb +6 -8
- data/lib/oxidized/model/openbsd.rb +76 -0
- data/lib/oxidized/model/opengear.rb +3 -5
- data/lib/oxidized/model/openwrt.rb +77 -0
- data/lib/oxidized/model/opnsense.rb +19 -0
- data/lib/oxidized/model/outputs.rb +1 -3
- data/lib/oxidized/model/panos.rb +1 -2
- data/lib/oxidized/model/pfsense.rb +9 -5
- data/lib/oxidized/model/planet.rb +8 -12
- data/lib/oxidized/model/powerconnect.rb +6 -9
- data/lib/oxidized/model/procurve.rb +18 -4
- data/lib/oxidized/model/quantaos.rb +3 -5
- data/lib/oxidized/model/routeros.rb +3 -2
- data/lib/oxidized/model/saos.rb +0 -1
- data/lib/oxidized/model/screenos.rb +3 -5
- data/lib/oxidized/model/sgos.rb +2 -3
- data/lib/oxidized/model/siklu.rb +0 -2
- data/lib/oxidized/model/slxos.rb +59 -0
- data/lib/oxidized/model/sros.rb +117 -0
- data/lib/oxidized/model/stoneos.rb +32 -0
- data/lib/oxidized/model/supermicro.rb +6 -41
- data/lib/oxidized/model/timos.rb +6 -114
- data/lib/oxidized/model/tmos.rb +1 -3
- data/lib/oxidized/model/tplink.rb +7 -11
- data/lib/oxidized/model/trango.rb +6 -7
- data/lib/oxidized/model/ucs.rb +0 -1
- data/lib/oxidized/model/voltaire.rb +3 -6
- data/lib/oxidized/model/voss.rb +1 -2
- data/lib/oxidized/model/vrp.rb +4 -5
- data/lib/oxidized/model/vyatta.rb +6 -4
- data/lib/oxidized/model/weos.rb +1 -3
- data/lib/oxidized/model/xos.rb +6 -5
- data/lib/oxidized/model/zhoneolt.rb +2 -2
- data/lib/oxidized/model/zynos.rb +1 -3
- data/lib/oxidized/model/zynoscli.rb +36 -0
- data/lib/oxidized/node.rb +11 -11
- data/lib/oxidized/node/stats.rb +15 -2
- data/lib/oxidized/nodes.rb +8 -8
- data/lib/oxidized/output/file.rb +41 -42
- data/lib/oxidized/output/git.rb +113 -115
- data/lib/oxidized/output/gitcrypt.rb +241 -242
- data/lib/oxidized/output/http.rb +23 -27
- data/lib/oxidized/output/output.rb +1 -2
- data/lib/oxidized/source/csv.rb +44 -45
- data/lib/oxidized/source/http.rb +52 -49
- data/lib/oxidized/source/source.rb +6 -7
- data/lib/oxidized/source/sql.rb +55 -51
- data/lib/oxidized/string.rb +3 -4
- data/lib/oxidized/version.rb +17 -1
- data/lib/oxidized/worker.rb +12 -3
- data/oxidized.gemspec +19 -13
- metadata +139 -51
- data/.ruby-version +0 -1
- data/Gemfile.lock +0 -44
data/extra/oxidized.service
CHANGED
data/extra/rest_client.rb
CHANGED
@@ -6,10 +6,10 @@ module Oxidized
|
|
6
6
|
require 'asetus'
|
7
7
|
|
8
8
|
class Config
|
9
|
-
Root
|
9
|
+
Root = Root = ENV['OXIDIZED_HOME'] || File.join(ENV['HOME'], '.config', 'oxidized')
|
10
10
|
end
|
11
11
|
|
12
|
-
CFGS = Asetus.new :name=>'oxidized', :load=>false, :key_to_s=>true
|
12
|
+
CFGS = Asetus.new :name => 'oxidized', :load => false, :key_to_s => true
|
13
13
|
CFGS.default.rest = '127.0.0.1:8888'
|
14
14
|
|
15
15
|
begin
|
@@ -28,13 +28,13 @@ module Oxidized
|
|
28
28
|
PATH = URI(restcfg).path
|
29
29
|
|
30
30
|
class << self
|
31
|
-
def next opt={}, host=HOST, port=PORT
|
31
|
+
def next opt = {}, host = HOST, port = PORT
|
32
32
|
web = new host, port
|
33
33
|
web.next opt
|
34
34
|
end
|
35
35
|
end
|
36
36
|
|
37
|
-
def initialize host=HOST, port=PORT
|
37
|
+
def initialize host = HOST, port = PORT
|
38
38
|
@web = Net::HTTP.new host, port
|
39
39
|
end
|
40
40
|
|
@@ -42,6 +42,5 @@ module Oxidized
|
|
42
42
|
data = JSON.dump opt
|
43
43
|
@web.put PATH + '/node/next/' + opt[:name].to_s, data
|
44
44
|
end
|
45
|
-
|
46
45
|
end
|
47
46
|
end
|
data/extra/syslog.rb
CHANGED
@@ -27,13 +27,12 @@ require 'resolv'
|
|
27
27
|
require_relative 'rest_client'
|
28
28
|
|
29
29
|
module Oxidized
|
30
|
-
|
31
30
|
require 'asetus'
|
32
31
|
class Config
|
33
|
-
Root
|
32
|
+
Root = File.join ENV['HOME'], '.config', 'oxidized'
|
34
33
|
end
|
35
34
|
|
36
|
-
CFGS = Asetus.new :name=>'oxidized', :load=>false, :key_to_s=>true
|
35
|
+
CFGS = Asetus.new :name => 'oxidized', :load => false, :key_to_s => true
|
37
36
|
CFGS.default.syslogd.port = 514
|
38
37
|
CFGS.default.syslogd.file = 'messages'
|
39
38
|
CFGS.default.syslogd.resolve = true
|
@@ -43,7 +42,7 @@ module Oxidized
|
|
43
42
|
rescue => error
|
44
43
|
raise InvalidConfig, "Error loading config: #{error.message}"
|
45
44
|
ensure
|
46
|
-
CFG = CFGS.cfg
|
45
|
+
CFG = CFGS.cfg # convenienence, instead of Config.cfg.password, CFG.password
|
47
46
|
end
|
48
47
|
|
49
48
|
class SyslogMonitor
|
@@ -59,12 +58,13 @@ module Oxidized
|
|
59
58
|
}
|
60
59
|
|
61
60
|
class << self
|
62
|
-
def udp port=Oxidized::CFG.syslogd.port, listen=0
|
61
|
+
def udp port = Oxidized::CFG.syslogd.port, listen = 0
|
63
62
|
io = UDPSocket.new
|
64
63
|
io.bind listen, port
|
65
64
|
new io, :udp
|
66
65
|
end
|
67
|
-
|
66
|
+
|
67
|
+
def file syslog_file = Oxidized::CFG.syslogd.file
|
68
68
|
io = open syslog_file, 'r'
|
69
69
|
io.seek 0, IO::SEEK_END
|
70
70
|
new io, :file
|
@@ -73,7 +73,7 @@ module Oxidized
|
|
73
73
|
|
74
74
|
private
|
75
75
|
|
76
|
-
def initialize io, mode
|
76
|
+
def initialize io, mode = :udp
|
77
77
|
@mode = mode
|
78
78
|
run io
|
79
79
|
end
|
@@ -84,24 +84,24 @@ module Oxidized
|
|
84
84
|
|
85
85
|
def ios ip, log, i
|
86
86
|
# TODO: we need to fetch 'ip/name' in mode == :file here
|
87
|
-
user = log[i+5]
|
87
|
+
user = log[i + 5]
|
88
88
|
from = log[-1][1..-2]
|
89
|
-
rest(
|
90
|
-
|
89
|
+
rest(:user => user, :from => from, :model => 'ios', :ip => ip,
|
90
|
+
:name => getname(ip))
|
91
91
|
end
|
92
92
|
|
93
93
|
def jnpr ip, log, i
|
94
94
|
# TODO: we need to fetch 'ip/name' in mode == :file here
|
95
|
-
user = log[i+2][1..-2]
|
96
|
-
msg = log[(i+6)..-1].join(' ')[10..-2]
|
95
|
+
user = log[i + 2][1..-2]
|
96
|
+
msg = log[(i + 6)..-1].join(' ')[10..-2]
|
97
97
|
msg = nil if msg == 'none'
|
98
|
-
rest(
|
99
|
-
|
98
|
+
rest(:user => user, :msg => msg, :model => 'jnpr', :ip => ip,
|
99
|
+
:name => getname(ip))
|
100
100
|
end
|
101
101
|
|
102
102
|
def handle_log log, ip
|
103
103
|
log = log.to_s.split ' '
|
104
|
-
if i = log.find_index { |e| e.match(
|
104
|
+
if i = log.find_index { |e| e.match(MSG[:ios]) }
|
105
105
|
ios ip, log, i
|
106
106
|
elsif i = log.index(MSG[:junos])
|
107
107
|
jnpr ip, log, i
|
@@ -140,4 +140,4 @@ module Oxidized
|
|
140
140
|
end
|
141
141
|
|
142
142
|
Oxidized::SyslogMonitor.udp
|
143
|
-
#Oxidized::SyslogMonitor.file '/var/log/poop'
|
143
|
+
# Oxidized::SyslogMonitor.file '/var/log/poop'
|
data/lib/oxidized/cli.rb
CHANGED
@@ -40,11 +40,11 @@ module Oxidized
|
|
40
40
|
end
|
41
41
|
|
42
42
|
def parse_opts
|
43
|
-
opts = Slop.new(:help=>true) do
|
43
|
+
opts = Slop.new(:help => true) do
|
44
44
|
on 'd', 'debug', 'turn on debugging'
|
45
45
|
on 'daemonize', 'Daemonize/fork the process'
|
46
46
|
on 'v', 'version', 'show version' do
|
47
|
-
puts Oxidized::
|
47
|
+
puts Oxidized::VERSION_FULL
|
48
48
|
Kernel.exit
|
49
49
|
end
|
50
50
|
end
|
@@ -62,7 +62,7 @@ module Oxidized
|
|
62
62
|
def write_pid
|
63
63
|
if pidfile?
|
64
64
|
begin
|
65
|
-
File.open(pidfile, ::File::CREAT | ::File::EXCL | ::File::WRONLY){|f| f.write("#{Process.pid}") }
|
65
|
+
File.open(pidfile, ::File::CREAT | ::File::EXCL | ::File::WRONLY) { |f| f.write("#{Process.pid}") }
|
66
66
|
at_exit { File.delete(pidfile) if File.exists?(pidfile) }
|
67
67
|
rescue Errno::EEXIST
|
68
68
|
check_pid
|
data/lib/oxidized/config.rb
CHANGED
@@ -13,13 +13,14 @@ module Oxidized
|
|
13
13
|
HookDir = File.join Directory, %w(lib oxidized hook)
|
14
14
|
Sleep = 1
|
15
15
|
|
16
|
-
def self.load(cmd_opts={})
|
16
|
+
def self.load(cmd_opts = {})
|
17
17
|
asetus = Asetus.new(name: 'oxidized', load: false, key_to_s: true)
|
18
18
|
Oxidized.asetus = asetus
|
19
19
|
|
20
20
|
asetus.default.username = 'username'
|
21
21
|
asetus.default.password = 'password'
|
22
22
|
asetus.default.model = 'junos'
|
23
|
+
asetus.default.resolve_dns = true # if false, don't resolve DNS to IP
|
23
24
|
asetus.default.interval = 3600
|
24
25
|
asetus.default.use_syslog = false
|
25
26
|
asetus.default.debug = false
|
@@ -34,9 +35,11 @@ module Oxidized
|
|
34
35
|
asetus.default.models = {} # model level configuration
|
35
36
|
asetus.default.pid = File.join(Oxidized::Config::Root, 'pid')
|
36
37
|
|
37
|
-
asetus.default.input.default
|
38
|
-
asetus.default.input.debug
|
39
|
-
asetus.default.input.ssh.secure
|
38
|
+
asetus.default.input.default = 'ssh, telnet'
|
39
|
+
asetus.default.input.debug = false # or String for session log file
|
40
|
+
asetus.default.input.ssh.secure = false # complain about changed certs
|
41
|
+
asetus.default.input.ftp.passive = true # ftp passive mode
|
42
|
+
asetus.default.input.utf8_encoded = true # configuration is utf8 encoded or ascii-8bit
|
40
43
|
|
41
44
|
asetus.default.output.default = 'file' # file, git
|
42
45
|
asetus.default.source.default = 'csv' # csv, sql
|
data/lib/oxidized/core.rb
CHANGED
@@ -11,9 +11,9 @@ module Oxidized
|
|
11
11
|
def initialize args
|
12
12
|
Oxidized.mgr = Manager.new
|
13
13
|
Oxidized.Hooks = HookManager.from_config(Oxidized.config)
|
14
|
-
nodes
|
14
|
+
nodes = Nodes.new
|
15
15
|
raise NoNodesFound, 'source returns no usable nodes' if nodes.size == 0
|
16
|
-
@worker
|
16
|
+
@worker = Worker.new nodes
|
17
17
|
trap('HUP') { nodes.load }
|
18
18
|
if Oxidized.config.rest?
|
19
19
|
begin
|
@@ -22,7 +22,7 @@ module Oxidized
|
|
22
22
|
raise OxidizedError, 'oxidized-web not found: sudo gem install oxidized-web - \
|
23
23
|
or disable web support by setting "rest: false" in your configuration'
|
24
24
|
end
|
25
|
-
@rest
|
25
|
+
@rest = API::Web.new nodes, Oxidized.config.rest
|
26
26
|
@rest.run
|
27
27
|
end
|
28
28
|
run
|
data/lib/oxidized/hook.rb
CHANGED
@@ -1,89 +1,88 @@
|
|
1
1
|
module Oxidized
|
2
|
-
class HookManager
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
2
|
+
class HookManager
|
3
|
+
class << self
|
4
|
+
def from_config cfg
|
5
|
+
mgr = new
|
6
|
+
cfg.hooks.each do |name, h_cfg|
|
7
|
+
h_cfg.events.each do |event|
|
8
|
+
mgr.register event.to_sym, name, h_cfg.type, h_cfg
|
9
|
+
end
|
9
10
|
end
|
11
|
+
mgr
|
10
12
|
end
|
11
|
-
mgr
|
12
13
|
end
|
13
|
-
end
|
14
|
-
|
15
|
-
# HookContext is passed to each hook. It can contain anything related to the
|
16
|
-
# event in question. At least it contains the event name
|
17
|
-
class HookContext < OpenStruct; end
|
18
14
|
|
19
|
-
|
20
|
-
|
15
|
+
# HookContext is passed to each hook. It can contain anything related to the
|
16
|
+
# event in question. At least it contains the event name
|
17
|
+
class HookContext < OpenStruct; end
|
21
18
|
|
22
|
-
|
23
|
-
:
|
24
|
-
:node_fail,
|
25
|
-
:post_store,
|
26
|
-
:nodes_done
|
27
|
-
]
|
28
|
-
attr_reader :registered_hooks
|
19
|
+
# RegisteredHook is a container for a Hook instance
|
20
|
+
class RegisteredHook < Struct.new(:name, :hook); end
|
29
21
|
|
30
|
-
|
31
|
-
|
32
|
-
|
22
|
+
Events = [
|
23
|
+
:node_success,
|
24
|
+
:node_fail,
|
25
|
+
:post_store,
|
26
|
+
:nodes_done
|
27
|
+
]
|
28
|
+
attr_reader :registered_hooks
|
33
29
|
|
34
|
-
|
35
|
-
|
36
|
-
raise ArgumentError,
|
37
|
-
"unknown event #{event}, available: #{Events.join ','}"
|
30
|
+
def initialize
|
31
|
+
@registered_hooks = Hash.new { |h, k| h[k] = [] }
|
38
32
|
end
|
39
33
|
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
end
|
34
|
+
def register event, name, hook_type, cfg
|
35
|
+
unless Events.include? event
|
36
|
+
raise ArgumentError,
|
37
|
+
"unknown event #{event}, available: #{Events.join ','}"
|
38
|
+
end
|
46
39
|
|
47
|
-
|
40
|
+
Oxidized.mgr.add_hook hook_type
|
41
|
+
begin
|
42
|
+
hook = Oxidized.mgr.hook.fetch(hook_type).new
|
43
|
+
rescue KeyError
|
44
|
+
raise KeyError, "cannot find hook #{hook_type.inspect}"
|
45
|
+
end
|
48
46
|
|
49
|
-
|
50
|
-
Oxidized.logger.debug "Hook #{name.inspect} registered #{hook.class} for event #{event.inspect}"
|
51
|
-
end
|
47
|
+
hook.cfg = cfg
|
52
48
|
|
53
|
-
|
54
|
-
|
55
|
-
|
49
|
+
@registered_hooks[event] << RegisteredHook.new(name, hook)
|
50
|
+
Oxidized.logger.debug "Hook #{name.inspect} registered #{hook.class} for event #{event.inspect}"
|
51
|
+
end
|
56
52
|
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
53
|
+
def handle event, ctx_params = {}
|
54
|
+
ctx = HookContext.new ctx_params
|
55
|
+
ctx.event = event
|
56
|
+
|
57
|
+
@registered_hooks[event].each do |r_hook|
|
58
|
+
begin
|
59
|
+
r_hook.hook.run_hook ctx
|
60
|
+
rescue => e
|
61
|
+
Oxidized.logger.error "Hook #{r_hook.name} (#{r_hook.hook}) failed " +
|
62
|
+
"(#{e.inspect}) for event #{event.inspect}"
|
63
|
+
end
|
63
64
|
end
|
64
65
|
end
|
65
66
|
end
|
66
|
-
end
|
67
67
|
|
68
|
-
# Hook abstract base class
|
69
|
-
class Hook
|
70
|
-
|
68
|
+
# Hook abstract base class
|
69
|
+
class Hook
|
70
|
+
attr_reader :cfg
|
71
71
|
|
72
|
-
|
73
|
-
|
72
|
+
def initialize
|
73
|
+
end
|
74
74
|
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
75
|
+
def cfg=(cfg)
|
76
|
+
@cfg = cfg
|
77
|
+
validate_cfg! if self.respond_to? :validate_cfg!
|
78
|
+
end
|
79
79
|
|
80
|
-
|
81
|
-
|
82
|
-
|
80
|
+
def run_hook ctx
|
81
|
+
raise NotImplementedError
|
82
|
+
end
|
83
83
|
|
84
|
-
|
85
|
-
|
84
|
+
def log(msg, level = :info)
|
85
|
+
Oxidized.logger.send(level, "#{self.class.name}: #{msg}")
|
86
|
+
end
|
86
87
|
end
|
87
|
-
|
88
|
-
end
|
89
88
|
end
|
data/lib/oxidized/hook/awssns.rb
CHANGED
@@ -0,0 +1,49 @@
|
|
1
|
+
require 'cisco_spark'
|
2
|
+
|
3
|
+
# defaults to posting a diff, if messageformat is supplied them a message will be posted too
|
4
|
+
# diffenable defaults to true
|
5
|
+
# Modified from slackdiff
|
6
|
+
|
7
|
+
class CiscoSparkDiff < Oxidized::Hook
|
8
|
+
def validate_cfg!
|
9
|
+
raise KeyError, 'hook.accesskey is required' unless cfg.has_key?('accesskey')
|
10
|
+
raise KeyError, 'hook.space is required' unless cfg.has_key?('space')
|
11
|
+
end
|
12
|
+
|
13
|
+
def run_hook(ctx)
|
14
|
+
return unless ctx.node
|
15
|
+
return unless ctx.event.to_s == "post_store"
|
16
|
+
log "Connecting to Cisco Spark"
|
17
|
+
CiscoSpark.configure do |config|
|
18
|
+
config.api_key = cfg.accesskey
|
19
|
+
config.proxy = cfg.proxy if cfg.has_key?('proxy')
|
20
|
+
end
|
21
|
+
space = cfg.space
|
22
|
+
client = CiscoSpark::Room.new(id: space)
|
23
|
+
client.fetch
|
24
|
+
log "Connected"
|
25
|
+
diffenable = true
|
26
|
+
if cfg.has_key?('diff') == true
|
27
|
+
if cfg.diff == false
|
28
|
+
diffenable = false
|
29
|
+
end
|
30
|
+
end
|
31
|
+
if diffenable == true
|
32
|
+
gitoutput = ctx.node.output.new
|
33
|
+
diff = gitoutput.get_diff ctx.node, ctx.node.group, ctx.commitref, nil
|
34
|
+
title = ctx.node.name.to_s
|
35
|
+
log "Posting diff as snippet to #{cfg.space}"
|
36
|
+
message = CiscoSpark::Message.new(text: 'Device ' + title + ' modified:' + "\n" + diff[:patch].lines.to_a[4..-1].join)
|
37
|
+
room = CiscoSpark::Room.new(id: space)
|
38
|
+
room.send_message(message)
|
39
|
+
end
|
40
|
+
if cfg.has_key?('message') == true
|
41
|
+
log cfg.message
|
42
|
+
msg = cfg.message % { :node => ctx.node.name.to_s, :group => ctx.node.group.to_s, :commitref => ctx.commitref, :model => ctx.node.model.class.name.to_s.downcase }
|
43
|
+
log msg
|
44
|
+
log "Posting message to #{cfg.space}"
|
45
|
+
client.chat_postMessage(channel: cfg.channel, text: msg, as_user: true)
|
46
|
+
end
|
47
|
+
log "Finished"
|
48
|
+
end
|
49
|
+
end
|
data/lib/oxidized/hook/exec.rb
CHANGED
@@ -23,10 +23,9 @@ class Exec < Oxidized::Hook
|
|
23
23
|
@cmd = cfg.cmd
|
24
24
|
raise "invalid cmd value" unless @cmd.is_a?(String) || @cmd.is_a?(Array)
|
25
25
|
end
|
26
|
-
|
27
26
|
rescue RuntimeError => e
|
28
27
|
raise ArgumentError,
|
29
|
-
|
28
|
+
"#{self.class.name}: configuration invalid: #{e.message}"
|
30
29
|
end
|
31
30
|
|
32
31
|
def run_hook ctx
|
@@ -45,7 +44,7 @@ class Exec < Oxidized::Hook
|
|
45
44
|
def run_cmd! env
|
46
45
|
pid, status = nil, nil
|
47
46
|
Timeout.timeout(@timeout) do
|
48
|
-
pid = spawn env, @cmd
|
47
|
+
pid = spawn env, @cmd, :unsetenv_others => true
|
49
48
|
pid, status = wait2 pid
|
50
49
|
unless status.exitstatus.zero?
|
51
50
|
msg = "#{@cmd.inspect} failed with exit value #{status.exitstatus}"
|
@@ -53,11 +52,11 @@ class Exec < Oxidized::Hook
|
|
53
52
|
raise msg
|
54
53
|
end
|
55
54
|
end
|
56
|
-
rescue
|
55
|
+
rescue Timeout::Error
|
57
56
|
kill "TERM", pid
|
58
57
|
msg = "#{@cmd} timed out"
|
59
58
|
log msg, :error
|
60
|
-
raise
|
59
|
+
raise Timeout::Error, msg
|
61
60
|
end
|
62
61
|
|
63
62
|
def make_env ctx
|
@@ -71,6 +70,7 @@ class Exec < Oxidized::Hook
|
|
71
70
|
"OX_NODE_FROM" => ctx.node.from.to_s,
|
72
71
|
"OX_NODE_MSG" => ctx.node.msg.to_s,
|
73
72
|
"OX_NODE_GROUP" => ctx.node.group.to_s,
|
73
|
+
"OX_NODE_MODEL" => ctx.node.model.class.name,
|
74
74
|
"OX_REPO_COMMITREF" => ctx.commitref.to_s,
|
75
75
|
"OX_REPO_NAME" => ctx.node.repo.to_s,
|
76
76
|
)
|