# Lint as: python2, python3 # Copyright (c) 2018 The Chromium OS Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. import socket import telnetlib from autotest_lib.client.common_lib import error SHORT_TIMEOUT = 2 LONG_TIMEOUT = 30 def _ascii_string(uc_string): """Returns ascii string given unicode string. @param uc_string: Unicode string """ return str(uc_string).encode('ASCII') class TelnetHelper(object): """Helper class to run basic string commands on a telnet host.""" def __init__(self, tx_cmd_separator="\n", rx_cmd_separator="\n", prompt=""): self._tn = None self._tx_cmd_separator = tx_cmd_separator self._rx_cmd_separator = rx_cmd_separator self._prompt = prompt def open(self, hostname, port=22): """Opens telnet connection to attenuator host. @param hostname: Valid hostname @param port: Optional port number, defaults to 22 """ if self._tn: self._tn.close() self._tn = telnetlib.Telnet() try: self._tn.open(hostname, port, LONG_TIMEOUT) except socket.timeout as e: raise error.TestError("Timed out while opening telnet connection") def is_open(self): """Returns true if telnet connection is open.""" return bool(self._tn) def close(self): """Closes telnet connection.""" if self._tn: self._tn.close() self._tn = None def cmd(self, cmd_str, wait_ret=True): """Run command on attenuator. @param cmd_str: Command to run @param wait_ret: Wait for command output or not @returns command output """ if not isinstance(cmd_str, str): raise error.TestError("Invalid command string %s" % cmd_str) if not self.is_open(): raise error.TestError("Telnet connection not open for commands") cmd_str.strip(self._tx_cmd_separator) try: self._tn.read_until(_ascii_string(self._prompt), SHORT_TIMEOUT) except EOFError as e: raise error.TestError("Connection closed. EOFError (%s)" % e) try: self._tn.write(_ascii_string(cmd_str + self._tx_cmd_separator)) except socket.error as e: raise error.TestError("Connection closed. Socket error (%s)." % e) if wait_ret is False: return None try: match_channel_idx, _, ret_text = self._tn.expect( [_ascii_string("\S+" + self._rx_cmd_separator)], SHORT_TIMEOUT) except EOFError as e: raise error.TestError("Connection closed. EOFError (%s)" % e) if match_channel_idx == -1: raise error.TestError("Telnet command failed to return valid data. " "Data returned: %s" % ret_text) ret_text = ret_text.decode() ret_text = ret_text.strip() return ret_text