blob: 5a4dbe1a43e2340bc1ba44f983cd303838e514ae [file] [log] [blame]
Michael Walshbd1bb602017-06-30 17:01:25 -05001#!/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
Michael Walsh2575cd62017-11-01 17:03:45 -050010import socket
Michael Walshbd1bb602017-06-30 17:01:25 -050011import paramiko
12import exceptions
13
14import gen_print as gp
15
16from robot.libraries.BuiltIn import BuiltIn
17from SSHLibrary import SSHLibrary
18sshlib = SSHLibrary()
19
20
Michael Walshbd1bb602017-06-30 17:01:25 -050021def sprint_connection(connection,
22 indent=0):
Michael Walshbd1bb602017-06-30 17:01:25 -050023 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
Michael Walshbd1bb602017-06-30 17:01:25 -050051
Michael Walshbd1bb602017-06-30 17:01:25 -050052def sprint_connections(connections=None,
53 indent=0):
Michael Walshbd1bb602017-06-30 17:01:25 -050054 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
Michael Walshbd1bb602017-06-30 17:01:25 -050075
Michael Walshbd1bb602017-06-30 17:01:25 -050076def find_connection(open_connection_args={}):
Michael Walshbd1bb602017-06-30 17:01:25 -050077 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
Michael Walshbd1bb602017-06-30 17:01:25 -0500102
Michael Walshbd1bb602017-06-30 17:01:25 -0500103def login_ssh(login_args={},
104 max_login_attempts=5):
Michael Walshbd1bb602017-06-30 17:01:25 -0500105 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
Michael Walsha03a3312017-12-01 16:47:44 -0600121 gp.lprint_executing()
122
Michael Walshbd1bb602017-06-30 17:01:25 -0500123 global sshlib
124
125 # Get connection data for debug output.
126 connection = sshlib.get_connection()
Michael Walsha03a3312017-12-01 16:47:44 -0600127 gp.lprintn(sprint_connection(connection))
Michael Walshbd1bb602017-06-30 17:01:25 -0500128 for login_attempt_num in range(1, max_login_attempts + 1):
Michael Walsha03a3312017-12-01 16:47:44 -0600129 gp.lprint_timen("Logging in to " + connection.host + ".")
130 gp.lprint_var(login_attempt_num)
Michael Walshbd1bb602017-06-30 17:01:25 -0500131 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()
Michael Walsha03a3312017-12-01 16:47:44 -0600136 gp.lprint_var(except_type)
137 gp.lprint_varx("except_value", str(except_value))
Michael Walshbd1bb602017-06-30 17:01:25 -0500138 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.
Michael Walsha03a3312017-12-01 16:47:44 -0600146 gp.lpvar(out_buf)
Michael Walshbd1bb602017-06-30 17:01:25 -0500147 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
Michael Walshbd1bb602017-06-30 17:01:25 -0500153
Michael Walshbd1bb602017-06-30 17:01:25 -0500154def 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):
Michael Walshbd1bb602017-06-30 17:01:25 -0500163 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
Michael Walsh8243ac92018-05-02 13:47:07 -0500171 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
Michael Walshbd1bb602017-06-30 17:01:25 -0500177 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
Michael Walsha03a3312017-12-01 16:47:44 -0600209 gp.lprint_executing()
Michael Walshbd1bb602017-06-30 17:01:25 -0500210
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)
Michael Walsha03a3312017-12-01 16:47:44 -0600217 gp.lpissuing(cmd_buf, test_mode)
Michael Walshbd1bb602017-06-30 17:01:25 -0500218
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:
Michael Walsha03a3312017-12-01 16:47:44 -0600231 gp.lprint_timen("Found the following existing connection:")
232 gp.lprintn(sprint_connection(connection))
Michael Walshbd1bb602017-06-30 17:01:25 -0500233 if connection.alias == "":
234 index_or_alias = connection.index
235 else:
236 index_or_alias = connection.alias
Michael Walsha03a3312017-12-01 16:47:44 -0600237 gp.lprint_timen("Switching to existing connection: \"" +
Michael Walshbd1bb602017-06-30 17:01:25 -0500238 str(index_or_alias) + "\".")
239 sshlib.switch_connection(index_or_alias)
240 else:
Michael Walsha03a3312017-12-01 16:47:44 -0600241 gp.lprint_timen("Connecting to " + open_connection_args['host'] + ".")
Michael Walshbd1bb602017-06-30 17:01:25 -0500242 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):
Michael Walsha03a3312017-12-01 16:47:44 -0600247 gp.lprint_var(exec_cmd_attempt_num)
Michael Walshbd1bb602017-06-30 17:01:25 -0500248 try:
249 if fork:
250 sshlib.start_command(cmd_buf)
251 else:
Michael Walsh8243ac92018-05-02 13:47:07 -0500252 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)
Michael Walshbd1bb602017-06-30 17:01:25 -0500262 except Exception as execute_exception:
263 except_type, except_value, except_traceback = sys.exc_info()
Michael Walsha03a3312017-12-01 16:47:44 -0600264 gp.lprint_var(except_type)
265 gp.lprint_varx("except_value", str(except_value))
Michael Walshbd1bb602017-06-30 17:01:25 -0500266
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
Michael Walsh2575cd62017-11-01 17:03:45 -0500273 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))):
Michael Walshbd1bb602017-06-30 17:01:25 -0500278 # Close and re-open a connection.
Michael Walsh2575cd62017-11-01 17:03:45 -0500279 # 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.
Michael Walsha03a3312017-12-01 16:47:44 -0600284 gp.lprint_timen("Closing all connections.")
Michael Walsh2575cd62017-11-01 17:03:45 -0500285 sshlib.close_all_connections()
Michael Walsha03a3312017-12-01 16:47:44 -0600286 gp.lprint_timen("Connecting to " +
Michael Walshbd1bb602017-06-30 17:01:25 -0500287 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.
Michael Walshe77585a2017-12-14 11:02:28 -0600294 sshlib.close_all_connections()
Michael Walshbd1bb602017-06-30 17:01:25 -0500295 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
Michael Walsh8243ac92018-05-02 13:47:07 -0500319 if open_connection_args['alias'] == "device_connection":
320 return stdout
Michael Walshbd1bb602017-06-30 17:01:25 -0500321 return stdout, stderr, rc