~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/transport/ssh.py

  • Committer: Patch Queue Manager
  • Date: 2016-04-21 04:10:52 UTC
  • mfrom: (6616.1.1 fix-en-user-guide)
  • Revision ID: pqm@pqm.ubuntu.com-20160421041052-clcye7ns1qcl2n7w
(richard-wilbur) Ensure build of English use guide always uses English text
 even when user's locale specifies a different language. (Jelmer Vernooij)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005 Robey Pointer <robey@lag.net>
 
1
# Copyright (C) 2006-2011 Robey Pointer <robey@lag.net>
2
2
# Copyright (C) 2005, 2006, 2007 Canonical Ltd
3
3
#
4
4
# This program is free software; you can redistribute it and/or modify
13
13
#
14
14
# You should have received a copy of the GNU General Public License
15
15
# along with this program; if not, write to the Free Software
16
 
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 
16
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
17
 
18
18
"""Foundation SSH support for SFTP and smart server."""
19
19
 
 
20
from __future__ import absolute_import
 
21
 
20
22
import errno
21
23
import getpass
 
24
import logging
22
25
import os
23
26
import socket
24
27
import subprocess
25
28
import sys
 
29
from binascii import hexlify
26
30
 
27
31
from bzrlib import (
28
32
    config,
93
97
            try:
94
98
                vendor = self._ssh_vendors[vendor_name]
95
99
            except KeyError:
96
 
                raise errors.UnknownSSH(vendor_name)
 
100
                vendor = self._get_vendor_from_path(vendor_name)
 
101
                if vendor is None:
 
102
                    raise errors.UnknownSSH(vendor_name)
 
103
                vendor.executable_path = vendor_name
97
104
            return vendor
98
105
        return None
99
106
 
109
116
            stdout = stderr = ''
110
117
        return stdout + stderr
111
118
 
112
 
    def _get_vendor_by_version_string(self, version, args):
 
119
    def _get_vendor_by_version_string(self, version, progname):
113
120
        """Return the vendor or None based on output from the subprocess.
114
121
 
115
122
        :param version: The output of 'ssh -V' like command.
122
129
        elif 'SSH Secure Shell' in version:
123
130
            trace.mutter('ssh implementation is SSH Corp.')
124
131
            vendor = SSHCorpSubprocessVendor()
125
 
        elif 'plink' in version and args[0] == 'plink':
 
132
        elif 'lsh' in version:
 
133
            trace.mutter('ssh implementation is GNU lsh.')
 
134
            vendor = LSHSubprocessVendor()
 
135
        # As plink user prompts are not handled currently, don't auto-detect
 
136
        # it by inspection below, but keep this vendor detection for if a path
 
137
        # is given in BZR_SSH. See https://bugs.launchpad.net/bugs/414743
 
138
        elif 'plink' in version and progname == 'plink':
126
139
            # Checking if "plink" was the executed argument as Windows
127
 
            # sometimes reports 'ssh -V' incorrectly with 'plink' in it's
 
140
            # sometimes reports 'ssh -V' incorrectly with 'plink' in its
128
141
            # version.  See https://bugs.launchpad.net/bzr/+bug/107155
129
142
            trace.mutter("ssh implementation is Putty's plink.")
130
143
            vendor = PLinkSubprocessVendor()
132
145
 
133
146
    def _get_vendor_by_inspection(self):
134
147
        """Return the vendor or None by checking for known SSH implementations."""
135
 
        for args in (['ssh', '-V'], ['plink', '-V']):
136
 
            version = self._get_ssh_version_string(args)
137
 
            vendor = self._get_vendor_by_version_string(version, args)
138
 
            if vendor is not None:
139
 
                return vendor
140
 
        return None
 
148
        version = self._get_ssh_version_string(['ssh', '-V'])
 
149
        return self._get_vendor_by_version_string(version, "ssh")
 
150
 
 
151
    def _get_vendor_from_path(self, path):
 
152
        """Return the vendor or None using the program at the given path"""
 
153
        version = self._get_ssh_version_string([path, '-V'])
 
154
        return self._get_vendor_by_version_string(version, 
 
155
            os.path.splitext(os.path.basename(path))[0])
141
156
 
142
157
    def get_vendor(self, environment=None):
143
158
        """Find out what version of SSH is on the system.
164
179
register_ssh_vendor = _ssh_vendor_manager.register_vendor
165
180
 
166
181
 
167
 
def _ignore_sigint():
 
182
def _ignore_signals():
168
183
    # TODO: This should possibly ignore SIGHUP as well, but bzr currently
169
184
    # doesn't handle it itself.
170
185
    # <https://launchpad.net/products/bzr/+bug/41433/+index>
171
186
    import signal
172
187
    signal.signal(signal.SIGINT, signal.SIG_IGN)
 
188
    # GZ 2010-02-19: Perhaps make this check if breakin is installed instead
 
189
    if signal.getsignal(signal.SIGQUIT) != signal.SIG_DFL:
 
190
        signal.signal(signal.SIGQUIT, signal.SIG_IGN)
173
191
 
174
192
 
175
193
class SocketAsChannelAdapter(object):
180
198
 
181
199
    def get_name(self):
182
200
        return "bzr SocketAsChannelAdapter"
183
 
    
 
201
 
184
202
    def send(self, data):
185
203
        return self.__socket.send(data)
186
204
 
211
229
 
212
230
    def connect_sftp(self, username, password, host, port):
213
231
        """Make an SSH connection, and return an SFTPClient.
214
 
        
 
232
 
215
233
        :param username: an ascii string
216
234
        :param password: an ascii string
217
235
        :param host: a host name as an ascii string
226
244
 
227
245
    def connect_ssh(self, username, password, host, port, command):
228
246
        """Make an SSH connection.
229
 
        
230
 
        :returns: something with a `close` method, and a `get_filelike_channels`
231
 
            method that returns a pair of (read, write) filelike objects.
 
247
 
 
248
        :returns: an SSHConnection.
232
249
        """
233
250
        raise NotImplementedError(self.connect_ssh)
234
251
 
257
274
register_ssh_vendor('loopback', LoopbackVendor())
258
275
 
259
276
 
260
 
class _ParamikoSSHConnection(object):
261
 
    def __init__(self, channel):
262
 
        self.channel = channel
263
 
 
264
 
    def get_filelike_channels(self):
265
 
        return self.channel.makefile('rb'), self.channel.makefile('wb')
266
 
 
267
 
    def close(self):
268
 
        return self.channel.close()
269
 
 
270
 
 
271
277
class ParamikoVendor(SSHVendor):
272
278
    """Vendor that uses paramiko."""
273
279
 
 
280
    def _hexify(self, s):
 
281
        return hexlify(s).upper()
 
282
 
274
283
    def _connect(self, username, password, host, port):
275
284
        global SYSTEM_HOSTKEYS, BZR_HOSTKEYS
276
285
 
284
293
            self._raise_connection_error(host, port=port, orig_error=e)
285
294
 
286
295
        server_key = t.get_remote_server_key()
287
 
        server_key_hex = paramiko.util.hexify(server_key.get_fingerprint())
 
296
        server_key_hex = self._hexify(server_key.get_fingerprint())
288
297
        keytype = server_key.get_name()
289
298
        if host in SYSTEM_HOSTKEYS and keytype in SYSTEM_HOSTKEYS[host]:
290
299
            our_server_key = SYSTEM_HOSTKEYS[host][keytype]
291
 
            our_server_key_hex = paramiko.util.hexify(
292
 
                our_server_key.get_fingerprint())
 
300
            our_server_key_hex = self._hexify(our_server_key.get_fingerprint())
293
301
        elif host in BZR_HOSTKEYS and keytype in BZR_HOSTKEYS[host]:
294
302
            our_server_key = BZR_HOSTKEYS[host][keytype]
295
 
            our_server_key_hex = paramiko.util.hexify(
296
 
                our_server_key.get_fingerprint())
 
303
            our_server_key_hex = self._hexify(our_server_key.get_fingerprint())
297
304
        else:
298
305
            trace.warning('Adding %s host key for %s: %s'
299
306
                          % (keytype, host, server_key_hex))
303
310
            else:
304
311
                BZR_HOSTKEYS.setdefault(host, {})[keytype] = server_key
305
312
            our_server_key = server_key
306
 
            our_server_key_hex = paramiko.util.hexify(
307
 
                our_server_key.get_fingerprint())
 
313
            our_server_key_hex = self._hexify(our_server_key.get_fingerprint())
308
314
            save_host_keys()
309
315
        if server_key != our_server_key:
310
316
            filename1 = os.path.expanduser('~/.ssh/known_hosts')
336
342
            self._raise_connection_error(host, port=port, orig_error=e,
337
343
                                         msg='Unable to invoke remote bzr')
338
344
 
 
345
_ssh_connection_errors = (EOFError, OSError, IOError, socket.error)
339
346
if paramiko is not None:
340
347
    vendor = ParamikoVendor()
341
348
    register_ssh_vendor('paramiko', vendor)
342
349
    register_ssh_vendor('none', vendor)
343
350
    register_default_ssh_vendor(vendor)
344
 
    _sftp_connection_errors = (EOFError, paramiko.SSHException)
 
351
    _ssh_connection_errors += (paramiko.SSHException,)
345
352
    del vendor
346
 
else:
347
 
    _sftp_connection_errors = (EOFError,)
348
353
 
349
354
 
350
355
class SubprocessVendor(SSHVendor):
351
356
    """Abstract base class for vendors that use pipes to a subprocess."""
352
357
 
 
358
    # In general stderr should be inherited from the parent process so prompts
 
359
    # are visible on the terminal. This can be overriden to another file for
 
360
    # tests, but beware of using PIPE which may hang due to not being read.
 
361
    _stderr_target = None
 
362
 
353
363
    def _connect(self, argv):
354
 
        proc = subprocess.Popen(argv,
355
 
                                stdin=subprocess.PIPE,
356
 
                                stdout=subprocess.PIPE,
 
364
        # Attempt to make a socketpair to use as stdin/stdout for the SSH
 
365
        # subprocess.  We prefer sockets to pipes because they support
 
366
        # non-blocking short reads, allowing us to optimistically read 64k (or
 
367
        # whatever) chunks.
 
368
        try:
 
369
            my_sock, subproc_sock = socket.socketpair()
 
370
            osutils.set_fd_cloexec(my_sock)
 
371
        except (AttributeError, socket.error):
 
372
            # This platform doesn't support socketpair(), so just use ordinary
 
373
            # pipes instead.
 
374
            stdin = stdout = subprocess.PIPE
 
375
            my_sock, subproc_sock = None, None
 
376
        else:
 
377
            stdin = stdout = subproc_sock
 
378
        proc = subprocess.Popen(argv, stdin=stdin, stdout=stdout,
 
379
                                stderr=self._stderr_target,
357
380
                                **os_specific_subprocess_params())
358
 
        return SSHSubprocess(proc)
 
381
        if subproc_sock is not None:
 
382
            subproc_sock.close()
 
383
        return SSHSubprocessConnection(proc, sock=my_sock)
359
384
 
360
385
    def connect_sftp(self, username, password, host, port):
361
386
        try:
363
388
                                                  subsystem='sftp')
364
389
            sock = self._connect(argv)
365
390
            return SFTPClient(SocketAsChannelAdapter(sock))
366
 
        except _sftp_connection_errors, e:
367
 
            self._raise_connection_error(host, port=port, orig_error=e)
368
 
        except (OSError, IOError), e:
369
 
            # If the machine is fast enough, ssh can actually exit
370
 
            # before we try and send it the sftp request, which
371
 
            # raises a Broken Pipe
372
 
            if e.errno not in (errno.EPIPE,):
373
 
                raise
 
391
        except _ssh_connection_errors, e:
374
392
            self._raise_connection_error(host, port=port, orig_error=e)
375
393
 
376
394
    def connect_ssh(self, username, password, host, port, command):
378
396
            argv = self._get_vendor_specific_argv(username, host, port,
379
397
                                                  command=command)
380
398
            return self._connect(argv)
381
 
        except (EOFError), e:
382
 
            self._raise_connection_error(host, port=port, orig_error=e)
383
 
        except (OSError, IOError), e:
384
 
            # If the machine is fast enough, ssh can actually exit
385
 
            # before we try and send it the sftp request, which
386
 
            # raises a Broken Pipe
387
 
            if e.errno not in (errno.EPIPE,):
388
 
                raise
 
399
        except _ssh_connection_errors, e:
389
400
            self._raise_connection_error(host, port=port, orig_error=e)
390
401
 
391
402
    def _get_vendor_specific_argv(self, username, host, port, subsystem=None,
392
403
                                  command=None):
393
404
        """Returns the argument list to run the subprocess with.
394
 
        
 
405
 
395
406
        Exactly one of 'subsystem' and 'command' must be specified.
396
407
        """
397
408
        raise NotImplementedError(self._get_vendor_specific_argv)
400
411
class OpenSSHSubprocessVendor(SubprocessVendor):
401
412
    """SSH vendor that uses the 'ssh' executable from OpenSSH."""
402
413
 
 
414
    executable_path = 'ssh'
 
415
 
403
416
    def _get_vendor_specific_argv(self, username, host, port, subsystem=None,
404
417
                                  command=None):
405
 
        args = ['ssh',
 
418
        args = [self.executable_path,
406
419
                '-oForwardX11=no', '-oForwardAgent=no',
407
 
                '-oClearAllForwardings=yes', '-oProtocol=2',
 
420
                '-oClearAllForwardings=yes',
408
421
                '-oNoHostAuthenticationForLocalhost=yes']
409
422
        if port is not None:
410
423
            args.extend(['-p', str(port)])
422
435
class SSHCorpSubprocessVendor(SubprocessVendor):
423
436
    """SSH vendor that uses the 'ssh' executable from SSH Corporation."""
424
437
 
 
438
    executable_path = 'ssh'
 
439
 
425
440
    def _get_vendor_specific_argv(self, username, host, port, subsystem=None,
426
441
                                  command=None):
427
 
        args = ['ssh', '-x']
 
442
        args = [self.executable_path, '-x']
428
443
        if port is not None:
429
444
            args.extend(['-p', str(port)])
430
445
        if username is not None:
435
450
            args.extend([host] + command)
436
451
        return args
437
452
 
438
 
register_ssh_vendor('ssh', SSHCorpSubprocessVendor())
 
453
register_ssh_vendor('sshcorp', SSHCorpSubprocessVendor())
 
454
 
 
455
 
 
456
class LSHSubprocessVendor(SubprocessVendor):
 
457
    """SSH vendor that uses the 'lsh' executable from GNU"""
 
458
 
 
459
    executable_path = 'lsh'
 
460
 
 
461
    def _get_vendor_specific_argv(self, username, host, port, subsystem=None,
 
462
                                  command=None):
 
463
        args = [self.executable_path]
 
464
        if port is not None:
 
465
            args.extend(['-p', str(port)])
 
466
        if username is not None:
 
467
            args.extend(['-l', username])
 
468
        if subsystem is not None:
 
469
            args.extend(['--subsystem', subsystem, host])
 
470
        else:
 
471
            args.extend([host] + command)
 
472
        return args
 
473
 
 
474
register_ssh_vendor('lsh', LSHSubprocessVendor())
439
475
 
440
476
 
441
477
class PLinkSubprocessVendor(SubprocessVendor):
442
478
    """SSH vendor that uses the 'plink' executable from Putty."""
443
479
 
 
480
    executable_path = 'plink'
 
481
 
444
482
    def _get_vendor_specific_argv(self, username, host, port, subsystem=None,
445
483
                                  command=None):
446
 
        args = ['plink', '-x', '-a', '-ssh', '-2', '-batch']
 
484
        args = [self.executable_path, '-x', '-a', '-ssh', '-2', '-batch']
447
485
        if port is not None:
448
486
            args.extend(['-P', str(port)])
449
487
        if username is not None:
458
496
 
459
497
 
460
498
def _paramiko_auth(username, password, host, port, paramiko_transport):
461
 
    # paramiko requires a username, but it might be none if nothing was supplied
462
 
    # use the local username, just in case.
463
 
    # We don't override username, because if we aren't using paramiko,
464
 
    # the username might be specified in ~/.ssh/config and we don't want to
465
 
    # force it to something else
466
 
    # Also, it would mess up the self.relpath() functionality
467
499
    auth = config.AuthenticationConfig()
 
500
    # paramiko requires a username, but it might be none if nothing was
 
501
    # supplied.  If so, use the local username.
468
502
    if username is None:
469
 
        username = auth.get_user('ssh', host, port=port)
470
 
        if username is None:
471
 
            # Default to local user
472
 
            username = getpass.getuser()
473
 
 
 
503
        username = auth.get_user('ssh', host, port=port,
 
504
                                 default=getpass.getuser())
474
505
    if _use_ssh_agent:
475
506
        agent = paramiko.Agent()
476
507
        for key in agent.get_keys():
477
508
            trace.mutter('Trying SSH agent key %s'
478
 
                         % paramiko.util.hexify(key.get_fingerprint()))
 
509
                         % self._hexify(key.get_fingerprint()))
479
510
            try:
480
511
                paramiko_transport.auth_publickey(username, key)
481
512
                return
488
519
    if _try_pkey_auth(paramiko_transport, paramiko.DSSKey, username, 'id_dsa'):
489
520
        return
490
521
 
 
522
    # If we have gotten this far, we are about to try for passwords, do an
 
523
    # auth_none check to see if it is even supported.
 
524
    supported_auth_types = []
 
525
    try:
 
526
        # Note that with paramiko <1.7.5 this logs an INFO message:
 
527
        #    Authentication type (none) not permitted.
 
528
        # So we explicitly disable the logging level for this action
 
529
        old_level = paramiko_transport.logger.level
 
530
        paramiko_transport.logger.setLevel(logging.WARNING)
 
531
        try:
 
532
            paramiko_transport.auth_none(username)
 
533
        finally:
 
534
            paramiko_transport.logger.setLevel(old_level)
 
535
    except paramiko.BadAuthenticationType, e:
 
536
        # Supported methods are in the exception
 
537
        supported_auth_types = e.allowed_types
 
538
    except paramiko.SSHException, e:
 
539
        # Don't know what happened, but just ignore it
 
540
        pass
 
541
    # We treat 'keyboard-interactive' and 'password' auth methods identically,
 
542
    # because Paramiko's auth_password method will automatically try
 
543
    # 'keyboard-interactive' auth (using the password as the response) if
 
544
    # 'password' auth is not available.  Apparently some Debian and Gentoo
 
545
    # OpenSSH servers require this.
 
546
    # XXX: It's possible for a server to require keyboard-interactive auth that
 
547
    # requires something other than a single password, but we currently don't
 
548
    # support that.
 
549
    if ('password' not in supported_auth_types and
 
550
        'keyboard-interactive' not in supported_auth_types):
 
551
        raise errors.ConnectionError('Unable to authenticate to SSH host as'
 
552
            '\n  %s@%s\nsupported auth types: %s'
 
553
            % (username, host, supported_auth_types))
 
554
 
491
555
    if password:
492
556
        try:
493
557
            paramiko_transport.auth_password(username, password)
497
561
 
498
562
    # give up and ask for a password
499
563
    password = auth.get_password('ssh', host, username, port=port)
500
 
    try:
501
 
        paramiko_transport.auth_password(username, password)
502
 
    except paramiko.SSHException, e:
503
 
        raise errors.ConnectionError(
504
 
            'Unable to authenticate to SSH host as %s@%s' % (username, host), e)
 
564
    # get_password can still return None, which means we should not prompt
 
565
    if password is not None:
 
566
        try:
 
567
            paramiko_transport.auth_password(username, password)
 
568
        except paramiko.SSHException, e:
 
569
            raise errors.ConnectionError(
 
570
                'Unable to authenticate to SSH host as'
 
571
                '\n  %s@%s\n' % (username, host), e)
 
572
    else:
 
573
        raise errors.ConnectionError('Unable to authenticate to SSH host as'
 
574
                                     '  %s@%s' % (username, host))
505
575
 
506
576
 
507
577
def _try_pkey_auth(paramiko_transport, pkey_class, username, filename):
512
582
        return True
513
583
    except paramiko.PasswordRequiredException:
514
584
        password = ui.ui_factory.get_password(
515
 
            prompt='SSH %(filename)s password', filename=filename)
 
585
            prompt=u'SSH %(filename)s password',
 
586
            filename=filename.decode(osutils._fs_enc))
516
587
        try:
517
588
            key = pkey_class.from_private_key_file(filename, password)
518
589
            paramiko_transport.auth_publickey(username, key)
569
640
def os_specific_subprocess_params():
570
641
    """Get O/S specific subprocess parameters."""
571
642
    if sys.platform == 'win32':
572
 
        # setting the process group and closing fds is not supported on 
 
643
        # setting the process group and closing fds is not supported on
573
644
        # win32
574
645
        return {}
575
646
    else:
576
 
        # We close fds other than the pipes as the child process does not need 
 
647
        # We close fds other than the pipes as the child process does not need
577
648
        # them to be open.
578
649
        #
579
650
        # We also set the child process to ignore SIGINT.  Normally the signal
581
652
        # this causes it to be seen only by bzr and not by ssh.  Python will
582
653
        # generate a KeyboardInterrupt in bzr, and we will then have a chance
583
654
        # to release locks or do other cleanup over ssh before the connection
584
 
        # goes away.  
 
655
        # goes away.
585
656
        # <https://launchpad.net/products/bzr/+bug/5987>
586
657
        #
587
658
        # Running it in a separate process group is not good because then it
588
659
        # can't get non-echoed input of a password or passphrase.
589
660
        # <https://launchpad.net/products/bzr/+bug/40508>
590
 
        return {'preexec_fn': _ignore_sigint,
 
661
        return {'preexec_fn': _ignore_signals,
591
662
                'close_fds': True,
592
663
                }
593
664
 
594
 
 
595
 
class SSHSubprocess(object):
596
 
    """A socket-like object that talks to an ssh subprocess via pipes."""
597
 
 
598
 
    def __init__(self, proc):
 
665
import weakref
 
666
_subproc_weakrefs = set()
 
667
 
 
668
def _close_ssh_proc(proc, sock):
 
669
    """Carefully close stdin/stdout and reap the SSH process.
 
670
 
 
671
    If the pipes are already closed and/or the process has already been
 
672
    wait()ed on, that's ok, and no error is raised.  The goal is to do our best
 
673
    to clean up (whether or not a clean up was already tried).
 
674
    """
 
675
    funcs = []
 
676
    for closeable in (proc.stdin, proc.stdout, sock):
 
677
        # We expect that either proc (a subprocess.Popen) will have stdin and
 
678
        # stdout streams to close, or that we will have been passed a socket to
 
679
        # close, with the option not in use being None.
 
680
        if closeable is not None:
 
681
            funcs.append(closeable.close)
 
682
    funcs.append(proc.wait)
 
683
    for func in funcs:
 
684
        try:
 
685
            func()
 
686
        except OSError:
 
687
            # It's ok for the pipe to already be closed, or the process to
 
688
            # already be finished.
 
689
            continue
 
690
 
 
691
 
 
692
class SSHConnection(object):
 
693
    """Abstract base class for SSH connections."""
 
694
 
 
695
    def get_sock_or_pipes(self):
 
696
        """Returns a (kind, io_object) pair.
 
697
 
 
698
        If kind == 'socket', then io_object is a socket.
 
699
 
 
700
        If kind == 'pipes', then io_object is a pair of file-like objects
 
701
        (read_from, write_to).
 
702
        """
 
703
        raise NotImplementedError(self.get_sock_or_pipes)
 
704
 
 
705
    def close(self):
 
706
        raise NotImplementedError(self.close)
 
707
 
 
708
 
 
709
class SSHSubprocessConnection(SSHConnection):
 
710
    """A connection to an ssh subprocess via pipes or a socket.
 
711
 
 
712
    This class is also socket-like enough to be used with
 
713
    SocketAsChannelAdapter (it has 'send' and 'recv' methods).
 
714
    """
 
715
 
 
716
    def __init__(self, proc, sock=None):
 
717
        """Constructor.
 
718
 
 
719
        :param proc: a subprocess.Popen
 
720
        :param sock: if proc.stdin/out is a socket from a socketpair, then sock
 
721
            should bzrlib's half of that socketpair.  If not passed, proc's
 
722
            stdin/out is assumed to be ordinary pipes.
 
723
        """
599
724
        self.proc = proc
 
725
        self._sock = sock
 
726
        # Add a weakref to proc that will attempt to do the same as self.close
 
727
        # to avoid leaving processes lingering indefinitely.
 
728
        def terminate(ref):
 
729
            _subproc_weakrefs.remove(ref)
 
730
            _close_ssh_proc(proc, sock)
 
731
        _subproc_weakrefs.add(weakref.ref(self, terminate))
600
732
 
601
733
    def send(self, data):
602
 
        return os.write(self.proc.stdin.fileno(), data)
 
734
        if self._sock is not None:
 
735
            return self._sock.send(data)
 
736
        else:
 
737
            return os.write(self.proc.stdin.fileno(), data)
603
738
 
604
739
    def recv(self, count):
605
 
        return os.read(self.proc.stdout.fileno(), count)
606
 
 
607
 
    def close(self):
608
 
        self.proc.stdin.close()
609
 
        self.proc.stdout.close()
610
 
        self.proc.wait()
611
 
 
612
 
    def get_filelike_channels(self):
613
 
        return (self.proc.stdout, self.proc.stdin)
 
740
        if self._sock is not None:
 
741
            return self._sock.recv(count)
 
742
        else:
 
743
            return os.read(self.proc.stdout.fileno(), count)
 
744
 
 
745
    def close(self):
 
746
        _close_ssh_proc(self.proc, self._sock)
 
747
 
 
748
    def get_sock_or_pipes(self):
 
749
        if self._sock is not None:
 
750
            return 'socket', self._sock
 
751
        else:
 
752
            return 'pipes', (self.proc.stdout, self.proc.stdin)
 
753
 
 
754
 
 
755
class _ParamikoSSHConnection(SSHConnection):
 
756
    """An SSH connection via paramiko."""
 
757
 
 
758
    def __init__(self, channel):
 
759
        self.channel = channel
 
760
 
 
761
    def get_sock_or_pipes(self):
 
762
        return ('socket', self.channel)
 
763
 
 
764
    def close(self):
 
765
        return self.channel.close()
 
766
 
614
767