oxidized 0.21.0 → 0.22.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (170) hide show
  1. checksums.yaml +5 -5
  2. data/.github/no-response.yml +13 -0
  3. data/.gitignore +3 -0
  4. data/.rubocop.yml +37 -0
  5. data/.rubocop_todo.yml +714 -0
  6. data/.travis.yml +7 -1
  7. data/CHANGELOG.md +341 -243
  8. data/Dockerfile +44 -16
  9. data/LICENSE +201 -0
  10. data/README.md +114 -82
  11. data/Rakefile +19 -0
  12. data/TODO.md +29 -23
  13. data/bin/oxidized +1 -2
  14. data/docs/Configuration.md +71 -31
  15. data/docs/Creating-Models.md +78 -0
  16. data/docs/Hooks.md +145 -41
  17. data/docs/Model-Notes/AireOS.md +12 -0
  18. data/docs/Model-Notes/ArbOS.md +12 -0
  19. data/docs/Model-Notes/Comware.md +14 -0
  20. data/docs/Model-Notes/EOS.md +9 -0
  21. data/docs/Model-Notes/JunOS.md +34 -0
  22. data/docs/Model-Notes/Netgear.md +68 -0
  23. data/docs/Model-Notes/README.md +19 -0
  24. data/docs/{VRP-Huawei.md → Model-Notes/VRP-Huawei.md} +10 -2
  25. data/docs/Model-Notes/XGS4600-Zyxel.md +39 -0
  26. data/docs/Outputs.md +27 -28
  27. data/docs/Ruby-API.md +38 -18
  28. data/docs/Sources.md +78 -16
  29. data/docs/Supported-OS-Types.md +171 -148
  30. data/extra/oxidized.logrotate +7 -0
  31. data/extra/oxidized.service +1 -1
  32. data/extra/rest_client.rb +4 -5
  33. data/extra/syslog.rb +16 -16
  34. data/lib/oxidized/cli.rb +3 -3
  35. data/lib/oxidized/config.rb +7 -4
  36. data/lib/oxidized/core.rb +3 -3
  37. data/lib/oxidized/hook.rb +64 -65
  38. data/lib/oxidized/hook/awssns.rb +2 -3
  39. data/lib/oxidized/hook/ciscosparkdiff.rb +49 -0
  40. data/lib/oxidized/hook/exec.rb +5 -5
  41. data/lib/oxidized/hook/githubrepo.rb +20 -14
  42. data/lib/oxidized/hook/slackdiff.rb +38 -19
  43. data/lib/oxidized/hook/xmppdiff.rb +58 -0
  44. data/lib/oxidized/input/cli.rb +5 -6
  45. data/lib/oxidized/input/ftp.rb +8 -7
  46. data/lib/oxidized/input/http.rb +39 -0
  47. data/lib/oxidized/input/ssh.rb +24 -22
  48. data/lib/oxidized/input/telnet.rb +38 -32
  49. data/lib/oxidized/jobs.rb +3 -4
  50. data/lib/oxidized/manager.rb +9 -4
  51. data/lib/oxidized/model/acos.rb +15 -16
  52. data/lib/oxidized/model/acsw.rb +3 -8
  53. data/lib/oxidized/model/aen.rb +1 -2
  54. data/lib/oxidized/model/aireos.rb +3 -5
  55. data/lib/oxidized/model/alteonos.rb +16 -18
  56. data/lib/oxidized/model/alvarion.rb +0 -4
  57. data/lib/oxidized/model/aos.rb +2 -4
  58. data/lib/oxidized/model/aos7.rb +2 -3
  59. data/lib/oxidized/model/aosw.rb +13 -15
  60. data/lib/oxidized/model/apc_aos.rb +0 -3
  61. data/lib/oxidized/model/arbos.rb +26 -0
  62. data/lib/oxidized/model/aricentiss.rb +51 -0
  63. data/lib/oxidized/model/asa.rb +33 -35
  64. data/lib/oxidized/model/asyncos.rb +41 -44
  65. data/lib/oxidized/model/audiocodes.rb +4 -8
  66. data/lib/oxidized/model/awplus.rb +84 -0
  67. data/lib/oxidized/model/boss.rb +6 -5
  68. data/lib/oxidized/model/br6910.rb +43 -45
  69. data/lib/oxidized/model/c4cmts.rb +3 -5
  70. data/lib/oxidized/model/cambium.rb +22 -0
  71. data/lib/oxidized/model/catos.rb +0 -2
  72. data/lib/oxidized/model/cisconga.rb +1 -3
  73. data/lib/oxidized/model/ciscosma.rb +37 -40
  74. data/lib/oxidized/model/ciscosmb.rb +7 -4
  75. data/lib/oxidized/model/comnetms.rb +43 -0
  76. data/lib/oxidized/model/comware.rb +9 -9
  77. data/lib/oxidized/model/coriant8600.rb +3 -5
  78. data/lib/oxidized/model/coriantgroove.rb +3 -5
  79. data/lib/oxidized/model/corianttmos.rb +1 -3
  80. data/lib/oxidized/model/cumulus.rb +26 -32
  81. data/lib/oxidized/model/datacom.rb +0 -2
  82. data/lib/oxidized/model/dcnos.rb +46 -0
  83. data/lib/oxidized/model/dlink.rb +1 -1
  84. data/lib/oxidized/model/dnos.rb +9 -5
  85. data/lib/oxidized/model/edgecos.rb +45 -0
  86. data/lib/oxidized/model/edgeos.rb +5 -3
  87. data/lib/oxidized/model/edgeswitch.rb +1 -3
  88. data/lib/oxidized/model/enterasys.rb +1 -3
  89. data/lib/oxidized/model/eos.rb +6 -8
  90. data/lib/oxidized/model/fabricos.rb +3 -5
  91. data/lib/oxidized/model/firewareos.rb +2 -5
  92. data/lib/oxidized/model/fortios.rb +21 -17
  93. data/lib/oxidized/model/ftos.rb +2 -4
  94. data/lib/oxidized/model/fujitsupy.rb +2 -4
  95. data/lib/oxidized/model/gaiaos.rb +6 -10
  96. data/lib/oxidized/model/gcombnps.rb +82 -0
  97. data/lib/oxidized/model/hatteras.rb +8 -5
  98. data/lib/oxidized/model/hirschmann.rb +8 -10
  99. data/lib/oxidized/model/hpebladesystem.rb +19 -17
  100. data/lib/oxidized/model/hpemsa.rb +0 -3
  101. data/lib/oxidized/model/ios.rb +54 -55
  102. data/lib/oxidized/model/iosxe.rb +5 -0
  103. data/lib/oxidized/model/iosxr.rb +1 -3
  104. data/lib/oxidized/model/ipos.rb +1 -3
  105. data/lib/oxidized/model/ironware.rb +12 -15
  106. data/lib/oxidized/model/isam.rb +4 -5
  107. data/lib/oxidized/model/junos.rb +8 -7
  108. data/lib/oxidized/model/masteros.rb +1 -3
  109. data/lib/oxidized/model/mlnxos.rb +3 -4
  110. data/lib/oxidized/model/model.rb +15 -7
  111. data/lib/oxidized/model/mtrlrfs.rb +1 -4
  112. data/lib/oxidized/model/ndms.rb +24 -0
  113. data/lib/oxidized/model/netgear.rb +3 -4
  114. data/lib/oxidized/model/netscaler.rb +0 -2
  115. data/lib/oxidized/model/nos.rb +1 -3
  116. data/lib/oxidized/model/nxos.rb +13 -3
  117. data/lib/oxidized/model/oneos.rb +6 -8
  118. data/lib/oxidized/model/openbsd.rb +76 -0
  119. data/lib/oxidized/model/opengear.rb +3 -5
  120. data/lib/oxidized/model/openwrt.rb +77 -0
  121. data/lib/oxidized/model/opnsense.rb +19 -0
  122. data/lib/oxidized/model/outputs.rb +1 -3
  123. data/lib/oxidized/model/panos.rb +1 -2
  124. data/lib/oxidized/model/pfsense.rb +9 -5
  125. data/lib/oxidized/model/planet.rb +8 -12
  126. data/lib/oxidized/model/powerconnect.rb +6 -9
  127. data/lib/oxidized/model/procurve.rb +18 -4
  128. data/lib/oxidized/model/quantaos.rb +3 -5
  129. data/lib/oxidized/model/routeros.rb +3 -2
  130. data/lib/oxidized/model/saos.rb +0 -1
  131. data/lib/oxidized/model/screenos.rb +3 -5
  132. data/lib/oxidized/model/sgos.rb +2 -3
  133. data/lib/oxidized/model/siklu.rb +0 -2
  134. data/lib/oxidized/model/slxos.rb +59 -0
  135. data/lib/oxidized/model/sros.rb +117 -0
  136. data/lib/oxidized/model/stoneos.rb +32 -0
  137. data/lib/oxidized/model/supermicro.rb +6 -41
  138. data/lib/oxidized/model/timos.rb +6 -114
  139. data/lib/oxidized/model/tmos.rb +1 -3
  140. data/lib/oxidized/model/tplink.rb +7 -11
  141. data/lib/oxidized/model/trango.rb +6 -7
  142. data/lib/oxidized/model/ucs.rb +0 -1
  143. data/lib/oxidized/model/voltaire.rb +3 -6
  144. data/lib/oxidized/model/voss.rb +1 -2
  145. data/lib/oxidized/model/vrp.rb +4 -5
  146. data/lib/oxidized/model/vyatta.rb +6 -4
  147. data/lib/oxidized/model/weos.rb +1 -3
  148. data/lib/oxidized/model/xos.rb +6 -5
  149. data/lib/oxidized/model/zhoneolt.rb +2 -2
  150. data/lib/oxidized/model/zynos.rb +1 -3
  151. data/lib/oxidized/model/zynoscli.rb +36 -0
  152. data/lib/oxidized/node.rb +11 -11
  153. data/lib/oxidized/node/stats.rb +15 -2
  154. data/lib/oxidized/nodes.rb +8 -8
  155. data/lib/oxidized/output/file.rb +41 -42
  156. data/lib/oxidized/output/git.rb +113 -115
  157. data/lib/oxidized/output/gitcrypt.rb +241 -242
  158. data/lib/oxidized/output/http.rb +23 -27
  159. data/lib/oxidized/output/output.rb +1 -2
  160. data/lib/oxidized/source/csv.rb +44 -45
  161. data/lib/oxidized/source/http.rb +52 -49
  162. data/lib/oxidized/source/source.rb +6 -7
  163. data/lib/oxidized/source/sql.rb +55 -51
  164. data/lib/oxidized/string.rb +3 -4
  165. data/lib/oxidized/version.rb +17 -1
  166. data/lib/oxidized/worker.rb +12 -3
  167. data/oxidized.gemspec +19 -13
  168. metadata +139 -51
  169. data/.ruby-version +0 -1
  170. data/Gemfile.lock +0 -44
@@ -0,0 +1,7 @@
1
+ /var/log/oxidized/*.log {
2
+ weekly
3
+ rotate 3
4
+ size 10M
5
+ compress
6
+ delaycompress
7
+ }
@@ -10,7 +10,7 @@ Wants=network-online.target
10
10
  [Service]
11
11
  ExecStart=/usr/local/bin/oxidized
12
12
  User=oxidized
13
- KillSignal=SIGINT
13
+ KillSignal=SIGKILL
14
14
 
15
15
  [Install]
16
16
  WantedBy=multi-user.target
@@ -6,10 +6,10 @@ module Oxidized
6
6
  require 'asetus'
7
7
 
8
8
  class Config
9
- Root = Root = ENV['OXIDIZED_HOME'] || File.join(ENV['HOME'], '.config', 'oxidized')
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
@@ -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 = File.join ENV['HOME'], '.config', 'oxidized'
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 # convenienence, instead of Config.cfg.password, CFG.password
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
- def file syslog_file=Oxidized::CFG.syslogd.file
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=:udp
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( :user => user, :from => from, :model => 'ios', :ip => ip,
90
- :name => getname(ip) )
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( :user => user, :msg => msg, :model => 'jnpr', :ip => ip,
99
- :name => getname(ip) )
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( MSG[:ios] ) }
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'
@@ -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::VERSION
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
@@ -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 = 'ssh, telnet'
38
- asetus.default.input.debug = false # or String for session log file
39
- asetus.default.input.ssh.secure = false # complain about changed certs
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
@@ -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 = Nodes.new
14
+ nodes = Nodes.new
15
15
  raise NoNodesFound, 'source returns no usable nodes' if nodes.size == 0
16
- @worker = Worker.new nodes
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 = API::Web.new nodes, Oxidized.config.rest
25
+ @rest = API::Web.new nodes, Oxidized.config.rest
26
26
  @rest.run
27
27
  end
28
28
  run
@@ -1,89 +1,88 @@
1
1
  module Oxidized
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
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
- # RegisteredHook is a container for a Hook instance
20
- class RegisteredHook < Struct.new(:name, :hook); end
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
- Events = [
23
- :node_success,
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
- def initialize
31
- @registered_hooks = Hash.new {|h,k| h[k] = []}
32
- end
22
+ Events = [
23
+ :node_success,
24
+ :node_fail,
25
+ :post_store,
26
+ :nodes_done
27
+ ]
28
+ attr_reader :registered_hooks
33
29
 
34
- def register event, name, hook_type, cfg
35
- unless Events.include? event
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
- 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
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
- hook.cfg = cfg
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
- @registered_hooks[event] << RegisteredHook.new(name, hook)
50
- Oxidized.logger.debug "Hook #{name.inspect} registered #{hook.class} for event #{event.inspect}"
51
- end
47
+ hook.cfg = cfg
52
48
 
53
- def handle event, ctx_params={}
54
- ctx = HookContext.new ctx_params
55
- ctx.event = event
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
- @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}"
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
- attr_reader :cfg
68
+ # Hook abstract base class
69
+ class Hook
70
+ attr_reader :cfg
71
71
 
72
- def initialize
73
- end
72
+ def initialize
73
+ end
74
74
 
75
- def cfg=(cfg)
76
- @cfg = cfg
77
- validate_cfg! if self.respond_to? :validate_cfg!
78
- end
75
+ def cfg=(cfg)
76
+ @cfg = cfg
77
+ validate_cfg! if self.respond_to? :validate_cfg!
78
+ end
79
79
 
80
- def run_hook ctx
81
- raise NotImplementedError
82
- end
80
+ def run_hook ctx
81
+ raise NotImplementedError
82
+ end
83
83
 
84
- def log(msg, level=:info)
85
- Oxidized.logger.send(level, "#{self.class.name}: #{msg}")
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
@@ -19,9 +19,8 @@ class AwsSns < Oxidized::Hook
19
19
  :node => ctx.node.name.to_s
20
20
  )
21
21
  end
22
- topic.publish({
22
+ topic.publish(
23
23
  message: message.to_json
24
- })
24
+ )
25
25
  end
26
-
27
26
  end
@@ -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
@@ -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
- "#{self.class.name}: configuration invalid: #{e.message}"
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 , :unsetenv_others => true
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 TimeoutError
55
+ rescue Timeout::Error
57
56
  kill "TERM", pid
58
57
  msg = "#{@cmd} timed out"
59
58
  log msg, :error
60
- raise TimeoutError, msg
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
  )