xref: /openbmc/openbmc-test-automation/lib/gen_robot_ssh.py (revision 3982974c06feee77be9f9874b7e4dac3e736f659)
1#!/usr/bin/env python
2
3r"""
4This module provides many valuable ssh functions such as sprint_connection,
5execute_ssh_command, etc.
6"""
7
8import sys
9import re
10import socket
11import paramiko
12import exceptions
13
14import gen_print as gp
15
16from robot.libraries.BuiltIn import BuiltIn
17from SSHLibrary import SSHLibrary
18sshlib = SSHLibrary()
19
20
21def sprint_connection(connection,
22                      indent=0):
23    r"""
24    sprint data from the connection object to a string and return it.
25
26    connection                      A connection object which is created by
27                                    the SSHlibrary open_connection() function.
28    indent                          The number of characters to indent the
29                                    output.
30    """
31
32    buffer = gp.sindent("", indent)
33    buffer += "connection:\n"
34    indent += 2
35    buffer += gp.sprint_varx("index", connection.index, 0, indent)
36    buffer += gp.sprint_varx("host", connection.host, 0, indent)
37    buffer += gp.sprint_varx("alias", connection.alias, 0, indent)
38    buffer += gp.sprint_varx("port", connection.port, 0, indent)
39    buffer += gp.sprint_varx("timeout", connection.timeout, 0, indent)
40    buffer += gp.sprint_varx("newline", connection.newline, 0, indent)
41    buffer += gp.sprint_varx("prompt", connection.prompt, 0, indent)
42    buffer += gp.sprint_varx("term_type", connection.term_type, 0, indent)
43    buffer += gp.sprint_varx("width", connection.width, 0, indent)
44    buffer += gp.sprint_varx("height", connection.height, 0, indent)
45    buffer += gp.sprint_varx("path_separator", connection.path_separator, 0,
46                             indent)
47    buffer += gp.sprint_varx("encoding", connection.encoding, 0, indent)
48
49    return buffer
50
51
52def sprint_connections(connections=None,
53                       indent=0):
54    r"""
55    sprint data from the connections list to a string and return it.
56
57    connections                     A list of connection objects which are
58                                    created by the SSHlibrary open_connection
59                                    function.  If this value is null, this
60                                    function will populate with a call to the
61                                    SSHlibrary get_connections() function.
62    indent                          The number of characters to indent the
63                                    output.
64    """
65
66    if connections is None:
67        connections = sshlib.get_connections()
68
69    buffer = ""
70    for connection in connections:
71        buffer += sprint_connection(connection, indent)
72
73    return buffer
74
75
76def find_connection(open_connection_args={}):
77    r"""
78    Find connection that matches the given connection arguments and return
79    connection object.  Return False if no matching connection is found.
80
81    Description of argument(s):
82    open_connection_args            A dictionary of arg names and values which
83                                    are legal to pass to the SSHLibrary
84                                    open_connection function as parms/args.
85                                    For a match to occur, the value for each
86                                    item in open_connection_args must match
87                                    the corresponding value in the connection
88                                    being examined.
89    """
90
91    global sshlib
92
93    for connection in sshlib.get_connections():
94        # Create connection_dict from connection object.
95        connection_dict = dict((key, str(value)) for key, value in
96                               connection._config.iteritems())
97        if dict(connection_dict, **open_connection_args) == connection_dict:
98            return connection
99
100    return False
101
102
103def login_ssh(login_args={},
104              max_login_attempts=5):
105    r"""
106    Login on the latest open SSH connection.  Retry on failure up to
107    max_login_attempts.
108
109    The caller is responsible for making sure there is an open SSH connection.
110
111    Description of argument(s):
112    login_args                      A dictionary containing the key/value
113                                    pairs which are acceptable to the
114                                    SSHLibrary login function as parms/args.
115                                    At a minimum, this should contain a
116                                    'username' and a 'password' entry.
117    max_login_attempts              The max number of times to try logging in
118                                    (in the event of login failures).
119    """
120
121    gp.lprint_executing()
122
123    global sshlib
124
125    # Get connection data for debug output.
126    connection = sshlib.get_connection()
127    gp.lprintn(sprint_connection(connection))
128    for login_attempt_num in range(1, max_login_attempts + 1):
129        gp.lprint_timen("Logging in to " + connection.host + ".")
130        gp.lprint_var(login_attempt_num)
131        try:
132            out_buf = sshlib.login(**login_args)
133        except Exception as login_exception:
134            # Login will sometimes fail if the connection is new.
135            except_type, except_value, except_traceback = sys.exc_info()
136            gp.lprint_var(except_type)
137            gp.lprint_varx("except_value", str(except_value))
138            if except_type is paramiko.ssh_exception.SSHException and\
139                    re.match(r"No existing session", str(except_value)):
140                continue
141            else:
142                # We don't tolerate any other error so break from loop and
143                # re-raise exception.
144                break
145        # If we get to this point, the login has worked and we can return.
146        gp.lpvar(out_buf)
147        return
148
149    # If we get to this point, the login has failed on all attempts so the
150    # exception will be raised again.
151    raise(login_exception)
152
153
154def execute_ssh_command(cmd_buf,
155                        open_connection_args={},
156                        login_args={},
157                        print_out=0,
158                        print_err=0,
159                        ignore_err=1,
160                        fork=0,
161                        quiet=None,
162                        test_mode=None):
163    r"""
164    Run the given command in an SSH session and return the stdout, stderr and
165    the return code.
166
167    If there is no open SSH connection, this function will connect and login.
168    Likewise, if the caller has not yet logged in to the connection, this
169    function will do the login.
170
171    NOTE: There is special handling when open_connection_args['alias'] equals
172    "device_connection".
173    - A write, rather than an execute_command, is done.
174    - Only stdout is returned (no stderr or rc).
175    - print_err, ignore_err and fork are not supported.
176
177    Description of arguments:
178    cmd_buf                         The command string to be run in an SSH
179                                    session.
180    open_connection_args            A dictionary of arg names and values which
181                                    are legal to pass to the SSHLibrary
182                                    open_connection function as parms/args.
183                                    At a minimum, this should contain a 'host'
184                                    entry.
185    login_args                      A dictionary containing the key/value
186                                    pairs which are acceptable to the
187                                    SSHLibrary login function as parms/args.
188                                    At a minimum, this should contain a
189                                    'username' and a 'password' entry.
190    print_out                       If this is set, this function will print
191                                    the stdout/stderr generated by the shell
192                                    command.
193    print_err                       If show_err is set, this function will
194                                    print a standardized error report if the
195                                    shell command returns non-zero.
196    ignore_err                      Indicates that errors encountered on the
197                                    sshlib.execute_command are to be ignored.
198    fork                            Indicates that sshlib.start is to be used
199                                    rather than sshlib.execute_command.
200    quiet                           Indicates whether this function should run
201                                    the pissuing() function which prints an
202                                    "Issuing: <cmd string>" to stdout.  This
203                                    defaults to the global quiet value.
204    test_mode                       If test_mode is set, this function will
205                                    not actually run the command.  This
206                                    defaults to the global test_mode value.
207    """
208
209    gp.lprint_executing()
210
211    # Obtain default values.
212    quiet = int(gp.get_var_value(quiet, 0))
213    test_mode = int(gp.get_var_value(test_mode, 0))
214
215    if not quiet:
216        gp.pissuing(cmd_buf, test_mode)
217    gp.lpissuing(cmd_buf, test_mode)
218
219    if test_mode:
220        return "", "", 0
221
222    global sshlib
223
224    # Look for existing SSH connection.
225    # Prepare a search connection dictionary.
226    search_connection_args = open_connection_args.copy()
227    # Remove keys that don't work well for searches.
228    search_connection_args.pop("timeout", None)
229    connection = find_connection(search_connection_args)
230    if connection:
231        gp.lprint_timen("Found the following existing connection:")
232        gp.lprintn(sprint_connection(connection))
233        if connection.alias == "":
234            index_or_alias = connection.index
235        else:
236            index_or_alias = connection.alias
237        gp.lprint_timen("Switching to existing connection: \"" +
238                        str(index_or_alias) + "\".")
239        sshlib.switch_connection(index_or_alias)
240    else:
241        gp.lprint_timen("Connecting to " + open_connection_args['host'] + ".")
242        cix = sshlib.open_connection(**open_connection_args)
243        login_ssh(login_args)
244
245    max_exec_cmd_attempts = 2
246    for exec_cmd_attempt_num in range(1, max_exec_cmd_attempts + 1):
247        gp.lprint_var(exec_cmd_attempt_num)
248        try:
249            if fork:
250                sshlib.start_command(cmd_buf)
251            else:
252                if open_connection_args['alias'] == "device_connection":
253                    stdout = sshlib.write(cmd_buf)
254                    stderr = ""
255                    rc = 0
256                else:
257                    stdout, stderr, rc = \
258                        sshlib.execute_command(cmd_buf,
259                                               return_stdout=True,
260                                               return_stderr=True,
261                                               return_rc=True)
262        except Exception as execute_exception:
263            except_type, except_value, except_traceback = sys.exc_info()
264            gp.lprint_var(except_type)
265            gp.lprint_varx("except_value", str(except_value))
266
267            if except_type is exceptions.AssertionError and\
268               re.match(r"Connection not open", str(except_value)):
269                login_ssh(login_args)
270                # Now we must continue to next loop iteration to retry the
271                # execute_command.
272                continue
273            if (except_type is paramiko.ssh_exception.SSHException and
274                re.match(r"SSH session not active", str(except_value))) or\
275               (except_type is socket.error and
276                re.match(r"\[Errno 104\] Connection reset by peer",
277                         str(except_value))):
278                # Close and re-open a connection.
279                # Note: close_connection() doesn't appear to get rid of the
280                # connection.  It merely closes it.  Since there is a concern
281                # about over-consumption of resources, we use
282                # close_all_connections() which also gets rid of all
283                # connections.
284                gp.lprint_timen("Closing all connections.")
285                sshlib.close_all_connections()
286                gp.lprint_timen("Connecting to " +
287                                open_connection_args['host'] + ".")
288                cix = sshlib.open_connection(**open_connection_args)
289                login_ssh(login_args)
290                continue
291
292            # We do not handle any other RuntimeErrors so we will raise the
293            # exception again.
294            sshlib.close_all_connections()
295            raise(execute_exception)
296
297        # If we get to this point, the command was executed.
298        break
299
300    if fork:
301        return
302
303    if rc != 0 and print_err:
304        gp.print_var(rc, 1)
305        if not print_out:
306            gp.print_var(stderr)
307            gp.print_var(stdout)
308
309    if print_out:
310        gp.printn(stderr + stdout)
311
312    if not ignore_err:
313        message = gp.sprint_error("The prior SSH" +
314                                  " command returned a non-zero return" +
315                                  " code:\n" + gp.sprint_var(rc, 1) + stderr +
316                                  "\n")
317        BuiltIn().should_be_equal(rc, 0, message)
318
319    if open_connection_args['alias'] == "device_connection":
320        return stdout
321    return stdout, stderr, rc
322