blob: 7af985d8e0852fd959fe264291c536efe46b8c5b [file] [log] [blame]
koder aka kdanilove06762a2015-03-22 23:32:09 +02001import re
koder aka kdanilov3a6633e2015-03-26 18:20:00 +02002import time
koder aka kdanilov416b87a2015-05-12 00:26:04 +03003import errno
koder aka kdanilov4af1c1d2015-05-18 15:48:58 +03004import random
koder aka kdanilov652cd802015-04-13 12:21:07 +03005import socket
koder aka kdanilov0c598a12015-04-21 03:01:40 +03006import shutil
koder aka kdanilove06762a2015-03-22 23:32:09 +02007import logging
8import os.path
koder aka kdanilov3a6633e2015-03-26 18:20:00 +02009import getpass
koder aka kdanilovf86d7af2015-05-06 04:01:54 +030010import StringIO
koder aka kdanilov652cd802015-04-13 12:21:07 +030011import threading
koder aka kdanilov0c598a12015-04-21 03:01:40 +030012import subprocess
koder aka kdanilov652cd802015-04-13 12:21:07 +030013
koder aka kdanilov3a6633e2015-03-26 18:20:00 +020014import paramiko
koder aka kdanilove06762a2015-03-22 23:32:09 +020015
koder aka kdanilove06762a2015-03-22 23:32:09 +020016
koder aka kdanilovcff7b2e2015-04-18 20:48:15 +030017logger = logging.getLogger("wally")
koder aka kdanilove06762a2015-03-22 23:32:09 +020018
19
koder aka kdanilov0c598a12015-04-21 03:01:40 +030020class Local(object):
21 "placeholder for local node"
22 @classmethod
23 def open_sftp(cls):
koder aka kdanilovafd98742015-04-24 01:27:22 +030024 return cls()
koder aka kdanilov0c598a12015-04-21 03:01:40 +030025
26 @classmethod
27 def mkdir(cls, remotepath, mode=None):
28 os.mkdir(remotepath)
29 if mode is not None:
30 os.chmod(remotepath, mode)
31
32 @classmethod
33 def put(cls, localfile, remfile):
koder aka kdanilov4d4771c2015-04-23 01:32:02 +030034 dirname = os.path.dirname(remfile)
35 if not os.path.exists(dirname):
36 os.makedirs(dirname)
koder aka kdanilov0c598a12015-04-21 03:01:40 +030037 shutil.copyfile(localfile, remfile)
38
39 @classmethod
40 def chmod(cls, path, mode):
41 os.chmod(path, mode)
42
43 @classmethod
44 def copytree(cls, src, dst):
45 shutil.copytree(src, dst)
46
47 @classmethod
48 def remove(cls, path):
49 os.unlink(path)
50
51 @classmethod
52 def close(cls):
53 pass
54
55 @classmethod
56 def open(cls, *args, **kwarhgs):
57 return open(*args, **kwarhgs)
58
koder aka kdanilove2de58c2015-04-24 22:59:36 +030059 @classmethod
60 def stat(cls, path):
61 return os.stat(path)
62
koder aka kdanilov783b4542015-04-23 18:57:04 +030063 def __enter__(self):
64 return self
65
66 def __exit__(self, x, y, z):
67 return False
68
koder aka kdanilov0c598a12015-04-21 03:01:40 +030069
koder aka kdanilovf86d7af2015-05-06 04:01:54 +030070NODE_KEYS = {}
71
72
koder aka kdanilov416b87a2015-05-12 00:26:04 +030073def exists(sftp, path):
74 """os.path.exists for paramiko's SCP object
75 """
76 try:
77 sftp.stat(path)
78 return True
79 except IOError as e:
80 if e.errno == errno.ENOENT:
81 return False
82 raise
83
84
koder aka kdanilovf86d7af2015-05-06 04:01:54 +030085def set_key_for_node(host_port, key):
86 sio = StringIO.StringIO(key)
87 NODE_KEYS[host_port] = paramiko.RSAKey.from_private_key(sio)
88 sio.close()
89
90
koder aka kdanilovbb5fe072015-05-21 02:50:23 +030091def ssh_connect(creds, conn_timeout=60, reuse_conn=None):
koder aka kdanilov0c598a12015-04-21 03:01:40 +030092 if creds == 'local':
koder aka kdanilov416b87a2015-05-12 00:26:04 +030093 return Local()
koder aka kdanilov0c598a12015-04-21 03:01:40 +030094
koder aka kdanilov46d4f392015-04-24 11:35:00 +030095 tcp_timeout = 15
96 banner_timeout = 30
97
koder aka kdanilovbb5fe072015-05-21 02:50:23 +030098 if reuse_conn is None:
99 ssh = paramiko.SSHClient()
100 ssh.load_host_keys('/dev/null')
101 ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
102 ssh.known_hosts = None
103 else:
104 ssh = reuse_conn
koder aka kdanilov168f6092015-04-19 02:33:38 +0300105
koder aka kdanilov6b1341a2015-04-21 22:44:21 +0300106 etime = time.time() + conn_timeout
107
108 while True:
koder aka kdanilov3a6633e2015-03-26 18:20:00 +0200109 try:
koder aka kdanilov46d4f392015-04-24 11:35:00 +0300110 tleft = etime - time.time()
111 c_tcp_timeout = min(tcp_timeout, tleft)
112 c_banner_timeout = min(banner_timeout, tleft)
113
koder aka kdanilov3a6633e2015-03-26 18:20:00 +0200114 if creds.passwd is not None:
115 ssh.connect(creds.host,
koder aka kdanilov46d4f392015-04-24 11:35:00 +0300116 timeout=c_tcp_timeout,
koder aka kdanilova4a570f2015-04-23 22:11:40 +0300117 username=creds.user,
koder aka kdanilov3a6633e2015-03-26 18:20:00 +0200118 password=creds.passwd,
119 port=creds.port,
120 allow_agent=False,
koder aka kdanilov46d4f392015-04-24 11:35:00 +0300121 look_for_keys=False,
122 banner_timeout=c_banner_timeout)
123 elif creds.key_file is not None:
koder aka kdanilov3a6633e2015-03-26 18:20:00 +0200124 ssh.connect(creds.host,
koder aka kdanilova4a570f2015-04-23 22:11:40 +0300125 username=creds.user,
koder aka kdanilov46d4f392015-04-24 11:35:00 +0300126 timeout=c_tcp_timeout,
koder aka kdanilov3a6633e2015-03-26 18:20:00 +0200127 key_filename=creds.key_file,
128 look_for_keys=False,
koder aka kdanilov46d4f392015-04-24 11:35:00 +0300129 port=creds.port,
130 banner_timeout=c_banner_timeout)
koder aka kdanilovf86d7af2015-05-06 04:01:54 +0300131 elif (creds.host, creds.port) in NODE_KEYS:
132 ssh.connect(creds.host,
133 username=creds.user,
134 timeout=c_tcp_timeout,
135 pkey=NODE_KEYS[(creds.host, creds.port)],
136 look_for_keys=False,
137 port=creds.port,
138 banner_timeout=c_banner_timeout)
koder aka kdanilov46d4f392015-04-24 11:35:00 +0300139 else:
140 key_file = os.path.expanduser('~/.ssh/id_rsa')
141 ssh.connect(creds.host,
142 username=creds.user,
143 timeout=c_tcp_timeout,
144 key_filename=key_file,
145 look_for_keys=False,
146 port=creds.port,
147 banner_timeout=c_banner_timeout)
koder aka kdanilov3a6633e2015-03-26 18:20:00 +0200148 return ssh
koder aka kdanilov3a6633e2015-03-26 18:20:00 +0200149 except paramiko.PasswordRequiredException:
150 raise
koder aka kdanilov46d4f392015-04-24 11:35:00 +0300151 except (socket.error, paramiko.SSHException):
koder aka kdanilov6b1341a2015-04-21 22:44:21 +0300152 if time.time() > etime:
koder aka kdanilov3a6633e2015-03-26 18:20:00 +0200153 raise
koder aka kdanilovcff7b2e2015-04-18 20:48:15 +0300154 time.sleep(1)
koder aka kdanilov3a6633e2015-03-26 18:20:00 +0200155
156
koder aka kdanilov4d4771c2015-04-23 01:32:02 +0300157def save_to_remote(sftp, path, content):
158 with sftp.open(path, "wb") as fd:
159 fd.write(content)
160
161
162def read_from_remote(sftp, path):
163 with sftp.open(path, "rb") as fd:
164 return fd.read()
165
166
koder aka kdanilove06762a2015-03-22 23:32:09 +0200167def normalize_dirpath(dirpath):
168 while dirpath.endswith("/"):
169 dirpath = dirpath[:-1]
170 return dirpath
171
172
koder aka kdanilov2c473092015-03-29 17:12:13 +0300173ALL_RWX_MODE = ((1 << 9) - 1)
174
175
176def ssh_mkdir(sftp, remotepath, mode=ALL_RWX_MODE, intermediate=False):
koder aka kdanilove06762a2015-03-22 23:32:09 +0200177 remotepath = normalize_dirpath(remotepath)
178 if intermediate:
179 try:
180 sftp.mkdir(remotepath, mode=mode)
koder aka kdanilov4d4771c2015-04-23 01:32:02 +0300181 except (IOError, OSError):
koder aka kdanilov168f6092015-04-19 02:33:38 +0300182 upper_dir = remotepath.rsplit("/", 1)[0]
183
184 if upper_dir == '' or upper_dir == '/':
185 raise
186
187 ssh_mkdir(sftp, upper_dir, mode=mode, intermediate=True)
koder aka kdanilove06762a2015-03-22 23:32:09 +0200188 return sftp.mkdir(remotepath, mode=mode)
189 else:
190 sftp.mkdir(remotepath, mode=mode)
191
192
193def ssh_copy_file(sftp, localfile, remfile, preserve_perm=True):
194 sftp.put(localfile, remfile)
195 if preserve_perm:
koder aka kdanilov2c473092015-03-29 17:12:13 +0300196 sftp.chmod(remfile, os.stat(localfile).st_mode & ALL_RWX_MODE)
koder aka kdanilove06762a2015-03-22 23:32:09 +0200197
198
199def put_dir_recursively(sftp, localpath, remotepath, preserve_perm=True):
200 "upload local directory to remote recursively"
201
202 # hack for localhost connection
203 if hasattr(sftp, "copytree"):
204 sftp.copytree(localpath, remotepath)
205 return
206
207 assert remotepath.startswith("/"), "%s must be absolute path" % remotepath
208
209 # normalize
210 localpath = normalize_dirpath(localpath)
211 remotepath = normalize_dirpath(remotepath)
212
213 try:
214 sftp.chdir(remotepath)
215 localsuffix = localpath.rsplit("/", 1)[1]
216 remotesuffix = remotepath.rsplit("/", 1)[1]
217 if localsuffix != remotesuffix:
218 remotepath = os.path.join(remotepath, localsuffix)
219 except IOError:
220 pass
221
222 for root, dirs, fls in os.walk(localpath):
223 prefix = os.path.commonprefix([localpath, root])
224 suffix = root.split(prefix, 1)[1]
225 if suffix.startswith("/"):
226 suffix = suffix[1:]
227
228 remroot = os.path.join(remotepath, suffix)
229
230 try:
231 sftp.chdir(remroot)
232 except IOError:
233 if preserve_perm:
koder aka kdanilov2c473092015-03-29 17:12:13 +0300234 mode = os.stat(root).st_mode & ALL_RWX_MODE
koder aka kdanilove06762a2015-03-22 23:32:09 +0200235 else:
koder aka kdanilov2c473092015-03-29 17:12:13 +0300236 mode = ALL_RWX_MODE
koder aka kdanilove06762a2015-03-22 23:32:09 +0200237 ssh_mkdir(sftp, remroot, mode=mode, intermediate=True)
238 sftp.chdir(remroot)
239
240 for f in fls:
241 remfile = os.path.join(remroot, f)
242 localfile = os.path.join(root, f)
243 ssh_copy_file(sftp, localfile, remfile, preserve_perm)
244
245
koder aka kdanilov4500a5f2015-04-17 16:55:17 +0300246def delete_file(conn, path):
247 sftp = conn.open_sftp()
248 sftp.remove(path)
249 sftp.close()
250
251
koder aka kdanilove06762a2015-03-22 23:32:09 +0200252def copy_paths(conn, paths):
253 sftp = conn.open_sftp()
254 try:
255 for src, dst in paths.items():
256 try:
257 if os.path.isfile(src):
258 ssh_copy_file(sftp, src, dst)
259 elif os.path.isdir(src):
260 put_dir_recursively(sftp, src, dst)
261 else:
262 templ = "Can't copy {0!r} - " + \
263 "it neither a file not a directory"
koder aka kdanilov168f6092015-04-19 02:33:38 +0300264 raise OSError(templ.format(src))
koder aka kdanilove06762a2015-03-22 23:32:09 +0200265 except Exception as exc:
266 tmpl = "Scp {0!r} => {1!r} failed - {2!r}"
koder aka kdanilov168f6092015-04-19 02:33:38 +0300267 raise OSError(tmpl.format(src, dst, exc))
koder aka kdanilove06762a2015-03-22 23:32:09 +0200268 finally:
269 sftp.close()
270
271
272class ConnCreds(object):
koder aka kdanilov2c473092015-03-29 17:12:13 +0300273 conn_uri_attrs = ("user", "passwd", "host", "port", "path")
274
koder aka kdanilove06762a2015-03-22 23:32:09 +0200275 def __init__(self):
koder aka kdanilov2c473092015-03-29 17:12:13 +0300276 for name in self.conn_uri_attrs:
koder aka kdanilove06762a2015-03-22 23:32:09 +0200277 setattr(self, name, None)
278
koder aka kdanilovcff7b2e2015-04-18 20:48:15 +0300279 def __str__(self):
280 return str(self.__dict__)
281
koder aka kdanilove06762a2015-03-22 23:32:09 +0200282
283uri_reg_exprs = []
284
285
286class URIsNamespace(object):
287 class ReParts(object):
288 user_rr = "[^:]*?"
koder aka kdanilov7e0f7cf2015-05-01 17:24:35 +0300289 host_rr = "[^:@]*?"
koder aka kdanilove06762a2015-03-22 23:32:09 +0200290 port_rr = "\\d+"
291 key_file_rr = "[^:@]*"
292 passwd_rr = ".*?"
293
294 re_dct = ReParts.__dict__
295
296 for attr_name, val in re_dct.items():
297 if attr_name.endswith('_rr'):
298 new_rr = "(?P<{0}>{1})".format(attr_name[:-3], val)
299 setattr(ReParts, attr_name, new_rr)
300
301 re_dct = ReParts.__dict__
302
303 templs = [
304 "^{host_rr}$",
koder aka kdanilov7e0f7cf2015-05-01 17:24:35 +0300305 "^{host_rr}:{port_rr}$",
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300306 "^{host_rr}::{key_file_rr}$",
307 "^{host_rr}:{port_rr}:{key_file_rr}$",
koder aka kdanilov7e0f7cf2015-05-01 17:24:35 +0300308 "^{user_rr}@{host_rr}$",
309 "^{user_rr}@{host_rr}:{port_rr}$",
koder aka kdanilove06762a2015-03-22 23:32:09 +0200310 "^{user_rr}@{host_rr}::{key_file_rr}$",
311 "^{user_rr}@{host_rr}:{port_rr}:{key_file_rr}$",
koder aka kdanilov7e0f7cf2015-05-01 17:24:35 +0300312 "^{user_rr}:{passwd_rr}@{host_rr}$",
313 "^{user_rr}:{passwd_rr}@{host_rr}:{port_rr}$",
koder aka kdanilove06762a2015-03-22 23:32:09 +0200314 ]
315
316 for templ in templs:
317 uri_reg_exprs.append(templ.format(**re_dct))
318
319
320def parse_ssh_uri(uri):
koder aka kdanilov7e0f7cf2015-05-01 17:24:35 +0300321 # user:passwd@ip_host:port
322 # user:passwd@ip_host
koder aka kdanilove06762a2015-03-22 23:32:09 +0200323 # user@ip_host:port
324 # user@ip_host
325 # ip_host:port
326 # ip_host
327 # user@ip_host:port:path_to_key_file
328 # user@ip_host::path_to_key_file
329 # ip_host:port:path_to_key_file
330 # ip_host::path_to_key_file
331
koder aka kdanilov4d4771c2015-04-23 01:32:02 +0300332 if uri.startswith("ssh://"):
333 uri = uri[len("ssh://"):]
334
koder aka kdanilove06762a2015-03-22 23:32:09 +0200335 res = ConnCreds()
336 res.port = "22"
337 res.key_file = None
338 res.passwd = None
koder aka kdanilova4a570f2015-04-23 22:11:40 +0300339 res.user = getpass.getuser()
koder aka kdanilove06762a2015-03-22 23:32:09 +0200340
341 for rr in uri_reg_exprs:
342 rrm = re.match(rr, uri)
343 if rrm is not None:
344 res.__dict__.update(rrm.groupdict())
345 return res
koder aka kdanilov652cd802015-04-13 12:21:07 +0300346
koder aka kdanilove06762a2015-03-22 23:32:09 +0200347 raise ValueError("Can't parse {0!r} as ssh uri value".format(uri))
348
349
koder aka kdanilovbb5fe072015-05-21 02:50:23 +0300350def reconnect(conn, uri, **params):
351 if uri == 'local':
352 return conn
353
354 creds = parse_ssh_uri(uri)
355 creds.port = int(creds.port)
356 return ssh_connect(creds, reuse_conn=conn, **params)
357
358
koder aka kdanilov168f6092015-04-19 02:33:38 +0300359def connect(uri, **params):
koder aka kdanilov0c598a12015-04-21 03:01:40 +0300360 if uri == 'local':
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300361 return Local()
koder aka kdanilov0c598a12015-04-21 03:01:40 +0300362
koder aka kdanilove06762a2015-03-22 23:32:09 +0200363 creds = parse_ssh_uri(uri)
364 creds.port = int(creds.port)
koder aka kdanilov168f6092015-04-19 02:33:38 +0300365 return ssh_connect(creds, **params)
koder aka kdanilove06762a2015-03-22 23:32:09 +0200366
367
koder aka kdanilov652cd802015-04-13 12:21:07 +0300368all_sessions_lock = threading.Lock()
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300369all_sessions = {}
370
371
koder aka kdanilov4af1c1d2015-05-18 15:48:58 +0300372class BGSSHTask(object):
373 def __init__(self, node, use_sudo):
374 self.node = node
375 self.pid = None
376 self.use_sudo = use_sudo
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300377
koder aka kdanilov4af1c1d2015-05-18 15:48:58 +0300378 def start(self, orig_cmd, **params):
379 uniq_name = 'test'
380 cmd = "screen -S {0} -d -m {1}".format(uniq_name, orig_cmd)
381 run_over_ssh(self.node.connection, cmd,
382 timeout=10, node=self.node.get_conn_id(),
383 **params)
384 processes = run_over_ssh(self.node.connection, "ps aux", nolog=True)
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300385
koder aka kdanilov4af1c1d2015-05-18 15:48:58 +0300386 for proc in processes.split("\n"):
387 if orig_cmd in proc and "SCREEN" not in proc:
388 self.pid = proc.split()[1]
389 break
390 else:
391 self.pid = -1
392
393 def check_running(self):
394 assert self.pid is not None
395 try:
396 run_over_ssh(self.node.connection,
397 "ls /proc/{0}".format(self.pid),
398 timeout=10, nolog=True)
399 return True
400 except OSError:
401 return False
koder aka kdanilov4af1c1d2015-05-18 15:48:58 +0300402
403 def kill(self, soft=True, use_sudo=True):
404 assert self.pid is not None
405 try:
406 if soft:
407 cmd = "kill {0}"
408 else:
409 cmd = "kill -9 {0}"
410
411 if self.use_sudo:
412 cmd = "sudo " + cmd
413
414 run_over_ssh(self.node.connection,
415 cmd.format(self.pid), nolog=True)
416 return True
417 except OSError:
418 return False
419
420 def wait(self, soft_timeout, timeout):
421 end_of_wait_time = timeout + time.time()
422 soft_end_of_wait_time = soft_timeout + time.time()
423 time_till_check = random.randint(5, 10)
424
koder aka kdanilovfd2cfa52015-05-20 03:17:42 +0300425 time_till_first_check = random.randint(2, 6)
426 time.sleep(time_till_first_check)
427 if not self.check_running():
428 return True
429
koder aka kdanilov4af1c1d2015-05-18 15:48:58 +0300430 while self.check_running() and time.time() < soft_end_of_wait_time:
431 time.sleep(soft_end_of_wait_time - time.time())
432
433 while end_of_wait_time > time.time():
434 time.sleep(time_till_check)
435 if not self.check_running():
436 break
437 else:
438 self.kill()
439 time.sleep(3)
440 if self.check_running():
441 self.kill(soft=False)
442 return False
443 return True
koder aka kdanilove06762a2015-03-22 23:32:09 +0200444
koder aka kdanilove06762a2015-03-22 23:32:09 +0200445
koder aka kdanilovcff7b2e2015-04-18 20:48:15 +0300446def run_over_ssh(conn, cmd, stdin_data=None, timeout=60,
447 nolog=False, node=None):
koder aka kdanilov652cd802015-04-13 12:21:07 +0300448 "should be replaces by normal implementation, with select"
koder aka kdanilov0c598a12015-04-21 03:01:40 +0300449
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300450 if isinstance(conn, Local):
koder aka kdanilov0c598a12015-04-21 03:01:40 +0300451 if not nolog:
452 logger.debug("SSH:local Exec {0!r}".format(cmd))
453 proc = subprocess.Popen(cmd, shell=True,
454 stdin=subprocess.PIPE,
455 stdout=subprocess.PIPE,
456 stderr=subprocess.STDOUT)
457
458 stdoutdata, _ = proc.communicate(input=stdin_data)
koder aka kdanilov0c598a12015-04-21 03:01:40 +0300459 if proc.returncode != 0:
460 templ = "SSH:{0} Cmd {1!r} failed with code {2}. Output: {3}"
461 raise OSError(templ.format(node, cmd, proc.returncode, stdoutdata))
462
463 return stdoutdata
464
koder aka kdanilov652cd802015-04-13 12:21:07 +0300465 transport = conn.get_transport()
466 session = transport.open_session()
koder aka kdanilove06762a2015-03-22 23:32:09 +0200467
koder aka kdanilov4500a5f2015-04-17 16:55:17 +0300468 if node is None:
469 node = ""
470
koder aka kdanilov652cd802015-04-13 12:21:07 +0300471 with all_sessions_lock:
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300472 all_sessions[id(session)] = session
koder aka kdanilove06762a2015-03-22 23:32:09 +0200473
koder aka kdanilov652cd802015-04-13 12:21:07 +0300474 try:
475 session.set_combine_stderr(True)
koder aka kdanilove06762a2015-03-22 23:32:09 +0200476
koder aka kdanilov652cd802015-04-13 12:21:07 +0300477 stime = time.time()
koder aka kdanilove06762a2015-03-22 23:32:09 +0200478
koder aka kdanilov652cd802015-04-13 12:21:07 +0300479 if not nolog:
koder aka kdanilov4500a5f2015-04-17 16:55:17 +0300480 logger.debug("SSH:{0} Exec {1!r}".format(node, cmd))
koder aka kdanilove06762a2015-03-22 23:32:09 +0200481
koder aka kdanilov652cd802015-04-13 12:21:07 +0300482 session.exec_command(cmd)
koder aka kdanilove06762a2015-03-22 23:32:09 +0200483
koder aka kdanilov652cd802015-04-13 12:21:07 +0300484 if stdin_data is not None:
485 session.sendall(stdin_data)
koder aka kdanilove06762a2015-03-22 23:32:09 +0200486
koder aka kdanilov652cd802015-04-13 12:21:07 +0300487 session.settimeout(1)
488 session.shutdown_write()
489 output = ""
490
491 while True:
492 try:
493 ndata = session.recv(1024)
494 output += ndata
495 if "" == ndata:
496 break
497 except socket.timeout:
498 pass
499
500 if time.time() - stime > timeout:
501 raise OSError(output + "\nExecution timeout")
502
503 code = session.recv_exit_status()
504 finally:
koder aka kdanilov4af1c1d2015-05-18 15:48:58 +0300505 found = False
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300506 with all_sessions_lock:
koder aka kdanilov4af1c1d2015-05-18 15:48:58 +0300507 if id(session) in all_sessions:
508 found = True
509 del all_sessions[id(session)]
510
511 if found:
512 session.close()
koder aka kdanilov652cd802015-04-13 12:21:07 +0300513
514 if code != 0:
koder aka kdanilov4500a5f2015-04-17 16:55:17 +0300515 templ = "SSH:{0} Cmd {1!r} failed with code {2}. Output: {3}"
516 raise OSError(templ.format(node, cmd, code, output))
koder aka kdanilov652cd802015-04-13 12:21:07 +0300517
518 return output
519
520
521def close_all_sessions():
522 with all_sessions_lock:
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300523 for session in all_sessions.values():
koder aka kdanilov652cd802015-04-13 12:21:07 +0300524 try:
525 session.sendall('\x03')
526 session.close()
527 except:
528 pass
koder aka kdanilov416b87a2015-05-12 00:26:04 +0300529 all_sessions.clear()