lib/oxidized/input/telnet.rb
module Oxidized
require 'net/telnet'
require 'oxidized/input/cli'
class Telnet < Input
RESCUE_FAIL = {}.freeze
include Input::CLI
attr_reader :telnet
def connect(node)
@node = node
@timeout = Oxidized.config.timeout
@node.model.cfg['telnet'].each { |cb| instance_exec(&cb) }
@log = File.open(Oxidized::Config::LOG + "/#{@node.ip}-telnet", 'w') if Oxidized.config.input.debug?
port = vars(:telnet_port) || 23
telnet_opts = {
'Host' => @node.ip,
'Port' => port.to_i,
'Timeout' => @timeout,
'Model' => @node.model,
'Log' => @log
}
@telnet = Net::Telnet.new telnet_opts
begin
login
rescue Timeout::Error
raise PromptUndetect, ['unable to detect prompt:', @node.prompt].join(' ')
end
connected?
end
def connected?
@telnet && (not @telnet.sock.closed?)
end
def cmd(cmd_str, expect = @node.prompt)
Oxidized.logger.debug "Telnet: #{cmd_str} @#{@node.name}"
return send(cmd_str + "\r\n") unless expect
# create a string to be passed to oxidized_expect and modified _there_
# default to a single space so it shouldn't be coerced to nil by any models.
out = String(' ')
@telnet.puts(cmd_str)
@telnet.oxidized_expect(timeout: @timeout, expect: expect, out: out)
out
end
def send(data)
@telnet.write data
end
def output
@telnet.output
end
private
def expect(regex)
@telnet.oxidized_expect expect: regex, timeout: @timeout
end
def disconnect
disconnect_cli
@telnet.close
rescue Errno::ECONNRESET
# This exception is intented and therefore not handled here
ensure
@log.close if Oxidized.config.input.debug?
(@telnet.close rescue true) unless @telnet.sock.closed?
end
end
end
module Net
class Telnet
## how to do this, without redefining the whole damn thing
## FIXME: we also need output (not sure I'm going to support this)
attr_reader :output
def oxidized_expect(options) ## rubocop:disable Metrics/PerceivedComplexity
model = @options["Model"]
@log = @options["Log"]
expects = [options[:expect]].flatten
time_out = options[:timeout] || @options["Timeout"] || Oxidized.config.timeout?
Timeout.timeout(time_out) do
line = ""
rest = ""
buf = ""
loop do
c = @sock.readpartial(1024 * 1024)
@output = c
c = rest + c
if Integer(c.rindex(/#{IAC}#{SE}/no) || 0) <
Integer(c.rindex(/#{IAC}#{SB}/no) || 0)
buf = preprocess(c[0...c.rindex(/#{IAC}#{SB}/no)])
rest = c[c.rindex(/#{IAC}#{SB}/no)..-1]
elsif (pt = c.rindex(/#{IAC}[^#{IAC}#{AO}#{AYT}#{DM}#{IP}#{NOP}]?\z/no) ||
c.rindex(/\r\z/no))
buf = preprocess(c[0...pt])
rest = c[pt..-1]
else
buf = preprocess(c)
rest = ''
end
if Oxidized.config.input.debug?
@log.print buf
@log.flush
end
line += buf
line = model.expects line
# match is a regexp object. we need to return that for logins to work.
match = expects.find { |re| line.match re }
# stomp on the out string object if we have one. (thus we were called by cmd?)
options[:out]&.replace(line)
return match if match
end
end
end
end
end