oxidized 0.20.0 → 0.28.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (222) hide show
  1. checksums.yaml +5 -5
  2. data/.codeclimate.yml +4 -0
  3. data/.github/PULL_REQUEST_TEMPLATE.md +12 -0
  4. data/.github/no-response.yml +13 -0
  5. data/.github/workflows/publishdocker.yml +13 -0
  6. data/.gitignore +4 -0
  7. data/.rubocop.yml +73 -0
  8. data/.rubocop_todo.yml +120 -0
  9. data/.travis.yml +6 -1
  10. data/CHANGELOG.md +693 -243
  11. data/Dockerfile +27 -19
  12. data/LICENSE +201 -0
  13. data/README.md +234 -913
  14. data/Rakefile +48 -7
  15. data/TODO.md +29 -23
  16. data/bin/console +1 -1
  17. data/bin/oxidized +6 -5
  18. data/docs/Configuration.md +313 -0
  19. data/docs/Creating-Models.md +140 -0
  20. data/docs/Hooks.md +274 -0
  21. data/docs/Model-Notes/AireOS.md +11 -0
  22. data/docs/Model-Notes/ArbOS.md +11 -0
  23. data/docs/Model-Notes/Comware.md +13 -0
  24. data/docs/Model-Notes/Cumulus.md +40 -0
  25. data/docs/Model-Notes/EOS.md +12 -0
  26. data/docs/Model-Notes/IOS.md +29 -0
  27. data/docs/Model-Notes/JunOS.md +33 -0
  28. data/docs/Model-Notes/LinuxGeneric.md +24 -0
  29. data/docs/Model-Notes/Netgear.md +87 -0
  30. data/docs/Model-Notes/Nokia.md +9 -0
  31. data/docs/Model-Notes/README.md +24 -0
  32. data/docs/Model-Notes/SmartAX-Huawei.md +35 -0
  33. data/docs/Model-Notes/VRP-Huawei.md +34 -0
  34. data/docs/Model-Notes/Viptela.md +12 -0
  35. data/docs/Model-Notes/XGS4600-Zyxel.md +36 -0
  36. data/docs/Outputs.md +190 -0
  37. data/docs/Ruby-API.md +199 -0
  38. data/docs/Sources.md +171 -0
  39. data/docs/Supported-OS-Types.md +227 -0
  40. data/docs/Troubleshooting.md +66 -0
  41. data/extra/nagios_check_failing_nodes.rb +9 -2
  42. data/extra/oxidized-report-git-commits +21 -40
  43. data/extra/oxidized-ubuntu.haproxy +45 -0
  44. data/extra/oxidized.logrotate +7 -0
  45. data/extra/oxidized.service +13 -0
  46. data/extra/rest_client.rb +7 -10
  47. data/extra/syslog.rb +47 -42
  48. data/lib/oxidized/cli.rb +41 -31
  49. data/lib/oxidized/config/vars.rb +9 -14
  50. data/lib/oxidized/config.rb +20 -13
  51. data/lib/oxidized/core.rb +8 -10
  52. data/lib/oxidized/hook/awssns.rb +6 -7
  53. data/lib/oxidized/hook/ciscosparkdiff.rb +43 -0
  54. data/lib/oxidized/hook/exec.rb +19 -24
  55. data/lib/oxidized/hook/githubrepo.rb +17 -17
  56. data/lib/oxidized/hook/noophook.rb +1 -1
  57. data/lib/oxidized/hook/slackdiff.rb +32 -19
  58. data/lib/oxidized/hook/xmppdiff.rb +59 -0
  59. data/lib/oxidized/hook.rb +63 -64
  60. data/lib/oxidized/input/cli.rb +22 -12
  61. data/lib/oxidized/input/exec.rb +28 -0
  62. data/lib/oxidized/input/ftp.rb +16 -15
  63. data/lib/oxidized/input/http.rb +72 -0
  64. data/lib/oxidized/input/input.rb +6 -6
  65. data/lib/oxidized/input/ssh.rb +64 -56
  66. data/lib/oxidized/input/telnet.rb +59 -102
  67. data/lib/oxidized/input/tftp.rb +9 -10
  68. data/lib/oxidized/jobs.rb +9 -10
  69. data/lib/oxidized/manager.rb +42 -44
  70. data/lib/oxidized/model/acos.rb +19 -20
  71. data/lib/oxidized/model/acsw.rb +62 -0
  72. data/lib/oxidized/model/adtran.rb +26 -0
  73. data/lib/oxidized/model/aen.rb +19 -0
  74. data/lib/oxidized/model/aireos.rb +9 -10
  75. data/lib/oxidized/model/airfiber.rb +22 -0
  76. data/lib/oxidized/model/alteonos.rb +58 -0
  77. data/lib/oxidized/model/alvarion.rb +0 -4
  78. data/lib/oxidized/model/aos.rb +11 -5
  79. data/lib/oxidized/model/aos7.rb +6 -7
  80. data/lib/oxidized/model/aosw.rb +30 -27
  81. data/lib/oxidized/model/apc_aos.rb +2 -5
  82. data/lib/oxidized/model/arbos.rb +26 -0
  83. data/lib/oxidized/model/aricentiss.rb +49 -0
  84. data/lib/oxidized/model/asa.rb +61 -22
  85. data/lib/oxidized/model/asyncos.rb +46 -0
  86. data/lib/oxidized/model/audiocodes.rb +28 -0
  87. data/lib/oxidized/model/audiocodesmp.rb +28 -0
  88. data/lib/oxidized/model/awplus.rb +84 -0
  89. data/lib/oxidized/model/axos.rb +16 -0
  90. data/lib/oxidized/model/boss.rb +77 -0
  91. data/lib/oxidized/model/br6910.rb +42 -45
  92. data/lib/oxidized/model/c4cmts.rb +6 -10
  93. data/lib/oxidized/model/cambium.rb +23 -0
  94. data/lib/oxidized/model/casa.rb +1 -1
  95. data/lib/oxidized/model/catos.rb +1 -3
  96. data/lib/oxidized/model/cisconga.rb +1 -3
  97. data/lib/oxidized/model/ciscosma.rb +42 -0
  98. data/lib/oxidized/model/ciscosmb.rb +30 -10
  99. data/lib/oxidized/model/ciscovpn3k.rb +11 -0
  100. data/lib/oxidized/model/cnos.rb +33 -0
  101. data/lib/oxidized/model/comnetms.rb +43 -0
  102. data/lib/oxidized/model/comtrol.rb +41 -0
  103. data/lib/oxidized/model/comware.rb +28 -16
  104. data/lib/oxidized/model/coriant8600.rb +3 -5
  105. data/lib/oxidized/model/coriantgroove.rb +26 -0
  106. data/lib/oxidized/model/corianttmos.rb +1 -3
  107. data/lib/oxidized/model/cumulus.rb +60 -49
  108. data/lib/oxidized/model/datacom.rb +1 -4
  109. data/lib/oxidized/model/dcnos.rb +46 -0
  110. data/lib/oxidized/model/dellx.rb +76 -0
  111. data/lib/oxidized/model/dlink.rb +5 -4
  112. data/lib/oxidized/model/dnos.rb +11 -5
  113. data/lib/oxidized/model/eciapollo.rb +34 -0
  114. data/lib/oxidized/model/edgecos.rb +49 -0
  115. data/lib/oxidized/model/edgeos.rb +12 -5
  116. data/lib/oxidized/model/edgeswitch.rb +2 -4
  117. data/lib/oxidized/model/enterasys.rb +28 -0
  118. data/lib/oxidized/model/eos.rb +8 -8
  119. data/lib/oxidized/model/fabricos.rb +4 -6
  120. data/lib/oxidized/model/fastiron.rb +66 -0
  121. data/lib/oxidized/model/fiberdriver.rb +2 -2
  122. data/lib/oxidized/model/firebrick.rb +31 -0
  123. data/lib/oxidized/model/firelinuxos.rb +41 -0
  124. data/lib/oxidized/model/firewareos.rb +3 -6
  125. data/lib/oxidized/model/fortios.rb +31 -19
  126. data/lib/oxidized/model/ftos.rb +8 -5
  127. data/lib/oxidized/model/fujitsupy.rb +5 -7
  128. data/lib/oxidized/model/gaiaos.rb +7 -11
  129. data/lib/oxidized/model/gcombnps.rb +84 -0
  130. data/lib/oxidized/model/grandstream.rb +9 -0
  131. data/lib/oxidized/model/hatteras.rb +9 -6
  132. data/lib/oxidized/model/hirschmann.rb +39 -0
  133. data/lib/oxidized/model/hpebladesystem.rb +20 -18
  134. data/lib/oxidized/model/hpemsa.rb +10 -0
  135. data/lib/oxidized/model/hpmsm.rb +84 -0
  136. data/lib/oxidized/model/ibos.rb +55 -0
  137. data/lib/oxidized/model/icotera.rb +27 -0
  138. data/lib/oxidized/model/ios.rb +63 -70
  139. data/lib/oxidized/model/iosxe.rb +5 -0
  140. data/lib/oxidized/model/iosxr.rb +2 -3
  141. data/lib/oxidized/model/ipos.rb +10 -6
  142. data/lib/oxidized/model/ironware.rb +20 -19
  143. data/lib/oxidized/model/isam.rb +5 -6
  144. data/lib/oxidized/model/junos.rb +9 -11
  145. data/lib/oxidized/model/linuxgeneric.rb +74 -0
  146. data/lib/oxidized/model/masteros.rb +3 -6
  147. data/lib/oxidized/model/mlnxos.rb +9 -10
  148. data/lib/oxidized/model/model.rb +72 -46
  149. data/lib/oxidized/model/mtrlrfs.rb +1 -4
  150. data/lib/oxidized/model/ndms.rb +23 -0
  151. data/lib/oxidized/model/netgear.rb +35 -15
  152. data/lib/oxidized/model/netonix.rb +2 -2
  153. data/lib/oxidized/model/netscaler.rb +6 -3
  154. data/lib/oxidized/model/nos.rb +5 -7
  155. data/lib/oxidized/model/nsxconfig.rb +22 -0
  156. data/lib/oxidized/model/nsxfirewall.rb +22 -0
  157. data/lib/oxidized/model/nxos.rb +13 -3
  158. data/lib/oxidized/model/oneos.rb +15 -9
  159. data/lib/oxidized/model/openbsd.rb +63 -0
  160. data/lib/oxidized/model/opengear.rb +3 -5
  161. data/lib/oxidized/model/openwrt.rb +78 -0
  162. data/lib/oxidized/model/opnsense.rb +19 -0
  163. data/lib/oxidized/model/os10.rb +46 -0
  164. data/lib/oxidized/model/outputs.rb +5 -7
  165. data/lib/oxidized/model/panos.rb +11 -12
  166. data/lib/oxidized/model/pfsense.rb +11 -6
  167. data/lib/oxidized/model/planet.rb +14 -17
  168. data/lib/oxidized/model/powerconnect.rb +24 -19
  169. data/lib/oxidized/model/procurve.rb +43 -11
  170. data/lib/oxidized/model/purityos.rb +12 -0
  171. data/lib/oxidized/model/qtech.rb +41 -0
  172. data/lib/oxidized/model/quantaos.rb +4 -6
  173. data/lib/oxidized/model/raisecom.rb +19 -0
  174. data/lib/oxidized/model/routeros.rb +26 -8
  175. data/lib/oxidized/model/saos.rb +1 -2
  176. data/lib/oxidized/model/screenos.rb +8 -11
  177. data/lib/oxidized/model/sgos.rb +45 -0
  178. data/lib/oxidized/model/siklu.rb +1 -3
  179. data/lib/oxidized/model/slxos.rb +59 -0
  180. data/lib/oxidized/model/smartax.rb +25 -0
  181. data/lib/oxidized/model/sonicos.rb +51 -0
  182. data/lib/oxidized/model/speedtouch.rb +34 -0
  183. data/lib/oxidized/model/sros.rb +96 -0
  184. data/lib/oxidized/model/stoneos.rb +32 -0
  185. data/lib/oxidized/model/supermicro.rb +6 -41
  186. data/lib/oxidized/model/tdre.rb +30 -0
  187. data/lib/oxidized/model/telco.rb +24 -0
  188. data/lib/oxidized/model/timos.rb +6 -114
  189. data/lib/oxidized/model/tmos.rb +6 -3
  190. data/lib/oxidized/model/tplink.rb +11 -11
  191. data/lib/oxidized/model/trango.rb +21 -42
  192. data/lib/oxidized/model/ucs.rb +30 -0
  193. data/lib/oxidized/model/viptela.rb +29 -0
  194. data/lib/oxidized/model/voltaire.rb +9 -12
  195. data/lib/oxidized/model/voss.rb +17 -6
  196. data/lib/oxidized/model/vrp.rb +11 -6
  197. data/lib/oxidized/model/vyatta.rb +8 -6
  198. data/lib/oxidized/model/weos.rb +20 -0
  199. data/lib/oxidized/model/xos.rb +20 -8
  200. data/lib/oxidized/model/zhoneolt.rb +2 -2
  201. data/lib/oxidized/model/zynos.rb +1 -3
  202. data/lib/oxidized/model/zynoscli.rb +36 -0
  203. data/lib/oxidized/model/zynosgs.rb +38 -0
  204. data/lib/oxidized/node/stats.rb +33 -8
  205. data/lib/oxidized/node.rb +86 -95
  206. data/lib/oxidized/nodes.rb +48 -44
  207. data/lib/oxidized/output/file.rb +32 -37
  208. data/lib/oxidized/output/git.rb +138 -153
  209. data/lib/oxidized/output/gitcrypt.rb +228 -242
  210. data/lib/oxidized/output/http.rb +35 -34
  211. data/lib/oxidized/output/output.rb +2 -3
  212. data/lib/oxidized/source/csv.rb +50 -44
  213. data/lib/oxidized/source/http.rb +58 -58
  214. data/lib/oxidized/source/source.rb +9 -10
  215. data/lib/oxidized/source/sql.rb +47 -45
  216. data/lib/oxidized/string.rb +18 -14
  217. data/lib/oxidized/version.rb +17 -1
  218. data/lib/oxidized/worker.rb +72 -33
  219. data/oxidized.gemspec +20 -19
  220. metadata +180 -36
  221. data/.ruby-version +0 -1
  222. data/Gemfile.lock +0 -44
@@ -5,61 +5,42 @@ module Oxidized
5
5
  require 'oxidized/input/cli'
6
6
  class SSH < Input
7
7
  RescueFail = {
8
- :debug => [
9
- Net::SSH::Disconnect,
8
+ debug: [
9
+ Net::SSH::Disconnect
10
10
  ],
11
- :warn => [
11
+ warn: [
12
12
  RuntimeError,
13
- Net::SSH::AuthenticationFailed,
14
- ],
15
- }
13
+ Net::SSH::AuthenticationFailed
14
+ ]
15
+ }.freeze
16
16
  include Input::CLI
17
17
  class NoShell < OxidizedError; end
18
18
 
19
- def connect node
19
+ def connect(node)
20
20
  @node = node
21
21
  @output = ''
22
22
  @pty_options = { term: "vt100" }
23
23
  @node.model.cfg['ssh'].each { |cb| instance_exec(&cb) }
24
- secure = Oxidized.config.input.ssh.secure
25
24
  @log = File.open(Oxidized::Config::Log + "/#{@node.ip}-ssh", 'w') if Oxidized.config.input.debug?
26
- port = vars(:ssh_port) || 22
27
- if proxy_host = vars(:ssh_proxy)
28
- proxy_command = "ssh "
29
- proxy_command += "-o StrictHostKeyChecking=no " unless secure
30
- proxy_command += "#{proxy_host} -W %h:%p"
31
- proxy = Net::SSH::Proxy::Command.new(proxy_command)
32
- end
33
- ssh_opts = {
34
- :port => port.to_i,
35
- :password => @node.auth[:password], :timeout => Oxidized.config.timeout,
36
- :paranoid => secure,
37
- :auth_methods => %w(none publickey password keyboard-interactive),
38
- :number_of_password_prompts => 0,
39
- :proxy => proxy,
40
- }
41
- ssh_opts[:keys] = vars(:ssh_keys).is_a?(Array) ? vars(:ssh_keys) : [vars(:ssh_keys)] if vars(:ssh_keys)
42
- ssh_opts[:kex] = vars(:ssh_kex).split(/,\s*/) if vars(:ssh_kex)
43
- ssh_opts[:encryption] = vars(:ssh_encryption).split(/,\s*/) if vars(:ssh_encryption)
44
25
 
45
26
  Oxidized.logger.debug "lib/oxidized/input/ssh.rb: Connecting to #{@node.name}"
46
- @ssh = Net::SSH.start(@node.ip, @node.auth[:username], ssh_opts)
27
+ @ssh = Net::SSH.start(@node.ip, @node.auth[:username], make_ssh_opts)
47
28
  unless @exec
48
29
  shell_open @ssh
49
30
  begin
50
31
  login
51
32
  rescue Timeout::Error
52
- raise PromptUndetect, [ @output, 'not matching configured prompt', @node.prompt ].join(' ')
33
+ raise PromptUndetect, [@output, 'not matching configured prompt', @node.prompt].join(' ')
53
34
  end
54
35
  end
55
36
  connected?
56
37
  end
57
38
 
58
39
  def connected?
59
- @ssh and not @ssh.closed?
40
+ @ssh && (not @ssh.closed?)
60
41
  end
61
42
 
62
- def cmd cmd, expect=node.prompt
43
+ def cmd(cmd, expect = node.prompt)
63
44
  Oxidized.logger.debug "lib/oxidized/input/ssh.rb #{cmd} @ #{node.name} with expect: #{expect.inspect}"
64
45
  if @exec
65
46
  @ssh.exec! cmd
@@ -68,15 +49,13 @@ module Oxidized
68
49
  end
69
50
  end
70
51
 
71
- def send data
52
+ def send(data)
72
53
  @ses.send_data data
73
54
  end
74
55
 
75
- def output
76
- @output
77
- end
56
+ attr_reader :output
78
57
 
79
- def pty_options hash
58
+ def pty_options(hash)
80
59
  @pty_options = @pty_options.merge hash
81
60
  end
82
61
 
@@ -85,25 +64,26 @@ module Oxidized
85
64
  def disconnect
86
65
  disconnect_cli
87
66
  # if disconnect does not disconnect us, give up after timeout
88
- Timeout::timeout(Oxidized.config.timeout) { @ssh.loop }
67
+ Timeout.timeout(Oxidized.config.timeout) { @ssh.loop }
89
68
  rescue Errno::ECONNRESET, Net::SSH::Disconnect, IOError
90
69
  ensure
91
70
  @log.close if Oxidized.config.input.debug?
92
71
  (@ssh.close rescue true) unless @ssh.closed?
93
72
  end
94
73
 
95
- def shell_open ssh
74
+ def shell_open(ssh)
96
75
  @ses = ssh.open_channel do |ch|
97
76
  ch.on_data do |_ch, data|
98
77
  if Oxidized.config.input.debug?
99
78
  @log.print data
100
- @log.fsync
79
+ @log.flush
101
80
  end
102
81
  @output << data
103
82
  @output = @node.model.expects @output
104
83
  end
105
- ch.request_pty (@pty_options) do |_ch, success_pty|
84
+ ch.request_pty(@pty_options) do |_ch, success_pty|
106
85
  raise NoShell, "Can't get PTY" unless success_pty
86
+
107
87
  ch.send_channel_request 'shell' do |_ch, success_shell|
108
88
  raise NoShell, "Can't get shell" unless success_shell
109
89
  end
@@ -111,22 +91,10 @@ module Oxidized
111
91
  end
112
92
  end
113
93
 
114
- # some models have SSH auth or terminal auth based on version of code
115
- # if SSH is configured for terminal auth, we'll still try to detect prompt
116
- def login
117
- if @username
118
- match = expect username, @node.prompt
119
- if match == username
120
- cmd @node.auth[:username], password
121
- cmd @node.auth[:password]
122
- end
123
- else
124
- expect @node.prompt
125
- end
126
- end
94
+ def exec(state = nil)
95
+ return nil if vars(:ssh_no_exec)
127
96
 
128
- def exec state=nil
129
- state == nil ? @exec : (@exec=state) unless vars :ssh_no_exec
97
+ state.nil? ? @exec : (@exec = state)
130
98
  end
131
99
 
132
100
  def cmd_shell(cmd, expect_re)
@@ -137,18 +105,58 @@ module Oxidized
137
105
  @output
138
106
  end
139
107
 
140
- def expect *regexps
108
+ def expect(*regexps)
141
109
  regexps = [regexps].flatten
142
110
  Oxidized.logger.debug "lib/oxidized/input/ssh.rb: expecting #{regexps.inspect} at #{node.name}"
143
- Timeout::timeout(Oxidized.config.timeout) do
111
+ Timeout.timeout(Oxidized.config.timeout) do
144
112
  @ssh.loop(0.1) do
145
113
  sleep 0.1
146
114
  match = regexps.find { |regexp| @output.match regexp }
147
115
  return match if match
116
+
148
117
  true
149
118
  end
150
119
  end
151
120
  end
152
121
 
122
+ def make_ssh_opts
123
+ secure = Oxidized.config.input.ssh.secure?
124
+ ssh_opts = {
125
+ number_of_password_prompts: 0,
126
+ keepalive: vars(:ssh_no_keepalive) ? false : true,
127
+ verify_host_key: secure ? :always : :never,
128
+ password: @node.auth[:password],
129
+ timeout: Oxidized.config.timeout,
130
+ port: (vars(:ssh_port) || 22).to_i
131
+ }
132
+
133
+ auth_methods = vars(:auth_methods) || %w[none publickey password]
134
+ ssh_opts[:auth_methods] = auth_methods
135
+ Oxidized.logger.debug "AUTH METHODS::#{auth_methods}"
136
+
137
+ if (proxy_host = vars(:ssh_proxy))
138
+ proxy_command = "ssh "
139
+ proxy_command += "-o StrictHostKeyChecking=no " unless secure
140
+ if (proxy_port = vars(:ssh_proxy_port))
141
+ proxy_command += "-p #{proxy_port} "
142
+ end
143
+ proxy_command += "#{proxy_host} -W %h:%p"
144
+ proxy = Net::SSH::Proxy::Command.new(proxy_command)
145
+ ssh_opts[:proxy] = proxy
146
+ end
147
+
148
+ ssh_opts[:keys] = [vars(:ssh_keys)].flatten if vars(:ssh_keys)
149
+ ssh_opts[:kex] = vars(:ssh_kex).split(/,\s*/) if vars(:ssh_kex)
150
+ ssh_opts[:encryption] = vars(:ssh_encryption).split(/,\s*/) if vars(:ssh_encryption)
151
+ ssh_opts[:host_key] = vars(:ssh_host_key).split(/,\s*/) if vars(:ssh_host_key)
152
+ ssh_opts[:hmac] = vars(:ssh_hmac).split(/,\s*/) if vars(:ssh_hmac)
153
+
154
+ if Oxidized.config.input.debug?
155
+ ssh_opts[:logger] = Oxidized.logger
156
+ ssh_opts[:verbose] = Logger::DEBUG
157
+ end
158
+
159
+ ssh_opts
160
+ end
153
161
  end
154
162
  end
@@ -2,48 +2,49 @@ module Oxidized
2
2
  require 'net/telnet'
3
3
  require 'oxidized/input/cli'
4
4
  class Telnet < Input
5
- RescueFail = {}
5
+ RescueFail = {}.freeze
6
6
  include Input::CLI
7
7
  attr_reader :telnet
8
8
 
9
- def connect node
9
+ def connect(node)
10
10
  @node = node
11
11
  @timeout = Oxidized.config.timeout
12
12
  @node.model.cfg['telnet'].each { |cb| instance_exec(&cb) }
13
+ @log = File.open(Oxidized::Config::Log + "/#{@node.ip}-telnet", 'w') if Oxidized.config.input.debug?
13
14
  port = vars(:telnet_port) || 23
14
15
 
15
- opt = { 'Host' => @node.ip,
16
- 'Port' => port.to_i,
17
- 'Timeout' => @timeout,
18
- 'Model' => @node.model }
19
- opt['Output_log'] = Oxidized::Config::Log + "/#{@node.ip}-telnet" if Oxidized.config.input.debug?
16
+ telnet_opts = {
17
+ 'Host' => @node.ip,
18
+ 'Port' => port.to_i,
19
+ 'Timeout' => @timeout,
20
+ 'Model' => @node.model,
21
+ 'Log' => @log
22
+ }
20
23
 
21
- @telnet = Net::Telnet.new opt
22
- if @node.auth[:username] and @node.auth[:username].length > 0
23
- expect username
24
- @telnet.puts @node.auth[:username]
25
- end
26
- expect password
27
- @telnet.puts @node.auth[:password]
24
+ @telnet = Net::Telnet.new telnet_opts
28
25
  begin
29
- expect @node.prompt
26
+ login
30
27
  rescue Timeout::Error
31
- raise PromptUndetect, [ 'unable to detect prompt:', @node.prompt ].join(' ')
28
+ raise PromptUndetect, ['unable to detect prompt:', @node.prompt].join(' ')
32
29
  end
30
+ connected?
33
31
  end
34
32
 
35
33
  def connected?
36
- @telnet and not @telnet.sock.closed?
34
+ @telnet && (not @telnet.sock.closed?)
37
35
  end
38
36
 
39
- def cmd cmd, expect=@node.prompt
40
- Oxidized.logger.debug "Telnet: #{cmd} @#{@node.name}"
41
- args = { 'String' => cmd }
42
- args.merge!({ 'Match' => expect, 'Timeout' => @timeout }) if expect
37
+ def cmd(cmd_str, expect = @node.prompt)
38
+ return send(cmd_str + "\r\n") unless expect
39
+
40
+ Oxidized.logger.debug "Telnet: #{cmd_str} @#{@node.name}"
41
+ args = { 'String' => cmd_str,
42
+ 'Match' => expect,
43
+ 'Timeout' => @timeout }
43
44
  @telnet.cmd args
44
45
  end
45
46
 
46
- def send data
47
+ def send(data)
47
48
  @telnet.write data
48
49
  end
49
50
 
@@ -53,106 +54,62 @@ module Oxidized
53
54
 
54
55
  private
55
56
 
56
- def expect re
57
- @telnet.waitfor 'Match' => re, 'Timeout' => @timeout
57
+ def expect(regex)
58
+ @telnet.oxidized_expect expect: regex, timeout: @timeout
58
59
  end
59
60
 
60
61
  def disconnect
61
- begin
62
- disconnect_cli
63
- @telnet.close
64
- rescue Errno::ECONNRESET
65
- end
62
+ disconnect_cli
63
+ @telnet.close
64
+ rescue Errno::ECONNRESET
65
+ ensure
66
+ @log.close if Oxidized.config.input.debug?
67
+ (@telnet.close rescue true) unless @telnet.sock.closed?
66
68
  end
67
-
68
69
  end
69
70
  end
70
71
 
71
-
72
72
  class Net::Telnet
73
- ## FIXME: we just need 'line = model.expects line' to handle pager
74
73
  ## how to do this, without redefining the whole damn thing
75
74
  ## FIXME: we also need output (not sure I'm going to support this)
76
75
  attr_reader :output
77
- def waitfor(options) # :yield: recvdata
78
- time_out = @options["Timeout"]
79
- waittime = @options["Waittime"]
80
- fail_eof = @options["FailEOF"]
76
+ def oxidized_expect(options)
81
77
  model = @options["Model"]
78
+ @log = @options["Log"]
82
79
 
83
- if options.kind_of?(Hash)
84
- prompt = if options.has_key?("Match")
85
- options["Match"]
86
- elsif options.has_key?("Prompt")
87
- options["Prompt"]
88
- elsif options.has_key?("String")
89
- Regexp.new( Regexp.quote(options["String"]) )
90
- end
91
- time_out = options["Timeout"] if options.has_key?("Timeout")
92
- waittime = options["Waittime"] if options.has_key?("Waittime")
93
- fail_eof = options["FailEOF"] if options.has_key?("FailEOF")
94
- else
95
- prompt = options
96
- end
80
+ expects = [options[:expect]].flatten
81
+ time_out = options[:timeout] || @options["Timeout"] || Oxidized.config.timeout?
97
82
 
98
- if time_out == false
99
- time_out = nil
100
- end
101
-
102
- line = ''
103
- buf = ''
104
- rest = ''
105
- until(prompt === line and not IO::select([@sock], nil, nil, waittime))
106
- unless IO::select([@sock], nil, nil, time_out)
107
- raise TimeoutError, "timed out while waiting for more data"
108
- end
109
- begin
83
+ Timeout.timeout(time_out) do
84
+ line = ""
85
+ rest = ""
86
+ buf = ""
87
+ loop do
110
88
  c = @sock.readpartial(1024 * 1024)
111
89
  @output = c
112
- @dumplog.log_dump('<', c) if @options.has_key?("Dump_log")
113
- if @options["Telnetmode"]
114
- c = rest + c
115
- if Integer(c.rindex(/#{IAC}#{SE}/no) || 0) <
116
- Integer(c.rindex(/#{IAC}#{SB}/no) || 0)
117
- buf = preprocess(c[0 ... c.rindex(/#{IAC}#{SB}/no)])
118
- rest = c[c.rindex(/#{IAC}#{SB}/no) .. -1]
119
- elsif pt = c.rindex(/#{IAC}[^#{IAC}#{AO}#{AYT}#{DM}#{IP}#{NOP}]?\z/no) ||
120
- c.rindex(/\r\z/no)
121
- buf = preprocess(c[0 ... pt])
122
- rest = c[pt .. -1]
123
- else
124
- buf = preprocess(c)
125
- rest = ''
126
- end
127
- else
128
- # Not Telnetmode.
129
- #
130
- # We cannot use preprocess() on this data, because that
131
- # method makes some Telnetmode-specific assumptions.
132
- buf = rest + c
133
- rest = ''
134
- unless @options["Binmode"]
135
- if pt = buf.rindex(/\r\z/no)
136
- buf = buf[0 ... pt]
137
- rest = buf[pt .. -1]
138
- end
139
- buf.gsub!(/#{EOL}/no, "\n")
140
- end
90
+ c = rest + c
91
+
92
+ if Integer(c.rindex(/#{IAC}#{SE}/no) || 0) <
93
+ Integer(c.rindex(/#{IAC}#{SB}/no) || 0)
94
+ buf = preprocess(c[0...c.rindex(/#{IAC}#{SB}/no)])
95
+ rest = c[c.rindex(/#{IAC}#{SB}/no)..-1]
96
+ elsif (pt = c.rindex(/#{IAC}[^#{IAC}#{AO}#{AYT}#{DM}#{IP}#{NOP}]?\z/no) ||
97
+ c.rindex(/\r\z/no))
98
+ buf = preprocess(c[0...pt])
99
+ rest = c[pt..-1]
100
+ else
101
+ buf = preprocess(c)
102
+ rest = ''
103
+ end
104
+ if Oxidized.config.input.debug?
105
+ @log.print buf
106
+ @log.flush
141
107
  end
142
- @log.print(buf) if @options.has_key?("Output_log")
143
108
  line += buf
144
109
  line = model.expects line
145
- line = yield line if block_given?
146
- yield buf if block_given?
147
- rescue EOFError # End of file reached
148
- raise if fail_eof
149
- if line == ''
150
- line = nil
151
- yield nil if block_given?
152
- end
153
- break
110
+ match = expects.find { |re| line.match re }
111
+ return match if match
154
112
  end
155
113
  end
156
- line
157
114
  end
158
115
  end
@@ -1,41 +1,40 @@
1
1
  module Oxidized
2
2
  require 'stringio'
3
3
  require_relative 'cli'
4
-
4
+
5
5
  begin
6
6
  require 'net/tftp'
7
7
  rescue LoadError
8
8
  raise OxidizedError, 'net/tftp not found: sudo gem install net-tftp'
9
9
  end
10
-
10
+
11
11
  class TFTP < Input
12
-
13
12
  include Input::CLI
14
-
13
+
15
14
  # TFTP utilizes UDP, there is not a connection. We simply specify an IP and send/receive data.
16
- def connect node
17
- @node = node
15
+ def connect(node)
16
+ @node = node
18
17
 
19
18
  @node.model.cfg['tftp'].each { |cb| instance_exec(&cb) }
20
19
  @log = File.open(Oxidized::Config::Log + "/#{@node.ip}-tftp", 'w') if Oxidized.config.input.debug?
21
20
  @tftp = Net::TFTP.new @node.ip
22
21
  end
23
22
 
24
- def cmd file
23
+ def cmd(file)
25
24
  Oxidized.logger.debug "TFTP: #{file} @ #{@node.name}"
26
25
  config = StringIO.new
27
26
  @tftp.getbinary file, config
28
27
  config.rewind
29
28
  config.read
30
29
  end
31
-
30
+
32
31
  private
33
-
32
+
34
33
  def disconnect
35
34
  # TFTP uses UDP, there is no connection to close
35
+ true
36
36
  ensure
37
37
  @log.close if Oxidized.config.input.debug?
38
38
  end
39
-
40
39
  end
41
40
  end
data/lib/oxidized/jobs.rb CHANGED
@@ -4,11 +4,11 @@ module Oxidized
4
4
  MAX_INTER_JOB_GAP = 300 # add job if more than X from last job started
5
5
  attr_accessor :interval, :max, :want
6
6
 
7
- def initialize max, interval, nodes
7
+ def initialize(max, interval, nodes)
8
8
  @max = max
9
- # Set interval to 1 if interval is 0 (=disabled) so we don't break
9
+ # Set interval to 1 if interval is 0 (=disabled) so we don't break
10
10
  # the 'ceil' function
11
- @interval = interval == 0 ? 1 : interval
11
+ @interval = interval.zero? ? 1 : interval
12
12
  @nodes = nodes
13
13
  @last = Time.now.utc
14
14
  @durations = Array.new @nodes.size, AVERAGE_DURATION
@@ -16,19 +16,19 @@ module Oxidized
16
16
  super()
17
17
  end
18
18
 
19
- def push arg
19
+ def push(arg)
20
20
  @last = Time.now.utc
21
21
  super
22
22
  end
23
23
 
24
- def duration last
24
+ def duration(last)
25
25
  if @durations.size > @nodes.size
26
26
  @durations.slice! @nodes.size...@durations.size
27
27
  elsif @durations.size < @nodes.size
28
28
  @durations.fill AVERAGE_DURATION, @durations.size...@nodes.size
29
29
  end
30
30
  @durations.push(last).shift
31
- @duration = @durations.inject(:+).to_f / @nodes.size #rolling average
31
+ @duration = @durations.inject(:+).to_f / @nodes.size # rolling average
32
32
  new_count
33
33
  end
34
34
 
@@ -44,10 +44,9 @@ module Oxidized
44
44
  # and b) we want less threads running than the total amount of nodes
45
45
  # and c) there is more than MAX_INTER_JOB_GAP since last one was started
46
46
  # then we want one more thread (rationale is to fix hanging thread causing HOLB)
47
- if @want <= size and @want < @nodes.size
48
- @want +=1 if (Time.now.utc - @last) > MAX_INTER_JOB_GAP
49
- end
50
- end
47
+ return unless @want <= size && @want < @nodes.size
51
48
 
49
+ @want += 1 if (Time.now.utc - @last) > MAX_INTER_JOB_GAP
50
+ end
52
51
  end
53
52
  end
@@ -5,62 +5,60 @@ module Oxidized
5
5
  require 'oxidized/source/source'
6
6
  class Manager
7
7
  class << self
8
- def load dir, file
9
- begin
10
- require File.join dir, file+'.rb'
11
- klass = nil
12
- [Oxidized, Object].each do |mod|
13
- klass = mod.constants.find { |const| const.to_s.downcase == file.downcase }
14
- klass = mod.constants.find { |const| const.to_s.downcase == 'oxidized'+ file.downcase } unless klass
15
- klass = mod.const_get klass if klass
16
- break if klass
17
- end
18
- i = klass.new
19
- i.setup if i.respond_to? :setup
20
- { file => klass }
21
- rescue LoadError
22
- {}
8
+ def load(dir, file)
9
+ require File.join dir, file + '.rb'
10
+ klass = nil
11
+ [Oxidized, Object].each do |mod|
12
+ klass = mod.constants.find { |const| const.to_s.casecmp(file).zero? }
13
+ klass ||= mod.constants.find { |const| const.to_s.downcase == 'oxidized' + file.downcase }
14
+ klass = mod.const_get klass if klass
15
+ break if klass
23
16
  end
17
+ i = klass.new
18
+ i.setup if i.respond_to? :setup
19
+ { file => klass }
20
+ rescue LoadError
21
+ false
24
22
  end
25
23
  end
26
- attr_reader :input, :output, :model, :source, :hook
24
+
25
+ attr_reader :input, :output, :source, :model, :hook
27
26
  def initialize
28
27
  @input = {}
29
28
  @output = {}
30
- @model = {}
31
29
  @source = {}
32
- @hook = {}
30
+ @model = {}
31
+ @hook = {}
32
+ end
33
+
34
+ def add_input(name)
35
+ loader @input, Config::InputDir, "input", name
33
36
  end
34
- def add_input method
35
- method = Manager.load Config::InputDir, method
36
- return false if method.empty?
37
- @input.merge! method
37
+
38
+ def add_output(name)
39
+ loader @output, Config::OutputDir, "output", name
38
40
  end
39
- def add_output method
40
- method = Manager.load Config::OutputDir, method
41
- return false if method.empty?
42
- @output.merge! method
41
+
42
+ def add_source(name)
43
+ loader @source, Config::SourceDir, "source", name
43
44
  end
44
- def add_model _model
45
- name = _model
46
- _model = Manager.load File.join(Config::Root, 'model'), name
47
- _model = Manager.load Config::ModelDir, name if _model.empty?
48
- return false if _model.empty?
49
- @model.merge! _model
45
+
46
+ def add_model(name)
47
+ loader @model, Config::ModelDir, "model", name
50
48
  end
51
- def add_source _source
52
- return nil if @source.key? _source
53
- _source = Manager.load Config::SourceDir, _source
54
- return false if _source.empty?
55
- @source.merge! _source
49
+
50
+ def add_hook(name)
51
+ loader @hook, Config::HookDir, "hook", name
56
52
  end
57
- def add_hook _hook
58
- return nil if @hook.key? _hook
59
- name = _hook
60
- _hook = Manager.load File.join(Config::Root, 'hook'), name
61
- _hook = Manager.load Config::HookDir, name if _hook.empty?
62
- return false if _hook.empty?
63
- @hook.merge! _hook
53
+
54
+ private
55
+
56
+ # if local version of file exists, load it, else load global - return falsy value if nothing loaded
57
+ def loader(hash, global_dir, local_dir, name)
58
+ dir = File.join(Config::Root, local_dir)
59
+ map = Manager.load(dir, name) if File.exist? File.join(dir, name + ".rb")
60
+ map ||= Manager.load(global_dir, name)
61
+ hash.merge!(map) if map
64
62
  end
65
63
  end
66
64
  end