~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/transport/ssh.py

  • Committer: Robert Collins
  • Date: 2009-09-07 03:08:30 UTC
  • mto: This revision was merged to the branch mainline in revision 4690.
  • Revision ID: robertc@robertcollins.net-20090907030830-rf59kt28d550eauj
Milestones language tightning, internal consistency.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
# Copyright (C) 2005 Robey Pointer <robey@lag.net>
 
2
# Copyright (C) 2005, 2006, 2007 Canonical Ltd
 
3
#
 
4
# This program is free software; you can redistribute it and/or modify
 
5
# it under the terms of the GNU General Public License as published by
 
6
# the Free Software Foundation; either version 2 of the License, or
 
7
# (at your option) any later version.
 
8
#
 
9
# This program is distributed in the hope that it will be useful,
 
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
12
# GNU General Public License for more details.
 
13
#
 
14
# You should have received a copy of the GNU General Public License
 
15
# along with this program; if not, write to the Free Software
 
16
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
17
 
 
18
"""Foundation SSH support for SFTP and smart server."""
 
19
 
 
20
import errno
 
21
import getpass
 
22
import logging
 
23
import os
 
24
import socket
 
25
import subprocess
 
26
import sys
 
27
 
 
28
from bzrlib import (
 
29
    config,
 
30
    errors,
 
31
    osutils,
 
32
    trace,
 
33
    ui,
 
34
    )
 
35
 
 
36
try:
 
37
    import paramiko
 
38
except ImportError, e:
 
39
    # If we have an ssh subprocess, we don't strictly need paramiko for all ssh
 
40
    # access
 
41
    paramiko = None
 
42
else:
 
43
    from paramiko.sftp_client import SFTPClient
 
44
 
 
45
 
 
46
SYSTEM_HOSTKEYS = {}
 
47
BZR_HOSTKEYS = {}
 
48
 
 
49
 
 
50
_paramiko_version = getattr(paramiko, '__version_info__', (0, 0, 0))
 
51
 
 
52
# Paramiko 1.5 tries to open a socket.AF_UNIX in order to connect
 
53
# to ssh-agent. That attribute doesn't exist on win32 (it does in cygwin)
 
54
# so we get an AttributeError exception. So we will not try to
 
55
# connect to an agent if we are on win32 and using Paramiko older than 1.6
 
56
_use_ssh_agent = (sys.platform != 'win32' or _paramiko_version >= (1, 6, 0))
 
57
 
 
58
 
 
59
class SSHVendorManager(object):
 
60
    """Manager for manage SSH vendors."""
 
61
 
 
62
    # Note, although at first sign the class interface seems similar to
 
63
    # bzrlib.registry.Registry it is not possible/convenient to directly use
 
64
    # the Registry because the class just has "get()" interface instead of the
 
65
    # Registry's "get(key)".
 
66
 
 
67
    def __init__(self):
 
68
        self._ssh_vendors = {}
 
69
        self._cached_ssh_vendor = None
 
70
        self._default_ssh_vendor = None
 
71
 
 
72
    def register_default_vendor(self, vendor):
 
73
        """Register default SSH vendor."""
 
74
        self._default_ssh_vendor = vendor
 
75
 
 
76
    def register_vendor(self, name, vendor):
 
77
        """Register new SSH vendor by name."""
 
78
        self._ssh_vendors[name] = vendor
 
79
 
 
80
    def clear_cache(self):
 
81
        """Clear previously cached lookup result."""
 
82
        self._cached_ssh_vendor = None
 
83
 
 
84
    def _get_vendor_by_environment(self, environment=None):
 
85
        """Return the vendor or None based on BZR_SSH environment variable.
 
86
 
 
87
        :raises UnknownSSH: if the BZR_SSH environment variable contains
 
88
                            unknown vendor name
 
89
        """
 
90
        if environment is None:
 
91
            environment = os.environ
 
92
        if 'BZR_SSH' in environment:
 
93
            vendor_name = environment['BZR_SSH']
 
94
            try:
 
95
                vendor = self._ssh_vendors[vendor_name]
 
96
            except KeyError:
 
97
                raise errors.UnknownSSH(vendor_name)
 
98
            return vendor
 
99
        return None
 
100
 
 
101
    def _get_ssh_version_string(self, args):
 
102
        """Return SSH version string from the subprocess."""
 
103
        try:
 
104
            p = subprocess.Popen(args,
 
105
                                 stdout=subprocess.PIPE,
 
106
                                 stderr=subprocess.PIPE,
 
107
                                 **os_specific_subprocess_params())
 
108
            stdout, stderr = p.communicate()
 
109
        except OSError:
 
110
            stdout = stderr = ''
 
111
        return stdout + stderr
 
112
 
 
113
    def _get_vendor_by_version_string(self, version, args):
 
114
        """Return the vendor or None based on output from the subprocess.
 
115
 
 
116
        :param version: The output of 'ssh -V' like command.
 
117
        :param args: Command line that was run.
 
118
        """
 
119
        vendor = None
 
120
        if 'OpenSSH' in version:
 
121
            trace.mutter('ssh implementation is OpenSSH')
 
122
            vendor = OpenSSHSubprocessVendor()
 
123
        elif 'SSH Secure Shell' in version:
 
124
            trace.mutter('ssh implementation is SSH Corp.')
 
125
            vendor = SSHCorpSubprocessVendor()
 
126
        # Auto-detect of plink vendor disabled, on Windows recommended
 
127
        # default ssh-client is paramiko
 
128
        # see https://bugs.launchpad.net/bugs/414743
 
129
        #~elif 'plink' in version and args[0] == 'plink':
 
130
        #~    # Checking if "plink" was the executed argument as Windows
 
131
        #~    # sometimes reports 'ssh -V' incorrectly with 'plink' in it's
 
132
        #~    # version.  See https://bugs.launchpad.net/bzr/+bug/107155
 
133
        #~    trace.mutter("ssh implementation is Putty's plink.")
 
134
        #~    vendor = PLinkSubprocessVendor()
 
135
        return vendor
 
136
 
 
137
    def _get_vendor_by_inspection(self):
 
138
        """Return the vendor or None by checking for known SSH implementations."""
 
139
        for args in (['ssh', '-V'], ['plink', '-V']):
 
140
            version = self._get_ssh_version_string(args)
 
141
            vendor = self._get_vendor_by_version_string(version, args)
 
142
            if vendor is not None:
 
143
                return vendor
 
144
        return None
 
145
 
 
146
    def get_vendor(self, environment=None):
 
147
        """Find out what version of SSH is on the system.
 
148
 
 
149
        :raises SSHVendorNotFound: if no any SSH vendor is found
 
150
        :raises UnknownSSH: if the BZR_SSH environment variable contains
 
151
                            unknown vendor name
 
152
        """
 
153
        if self._cached_ssh_vendor is None:
 
154
            vendor = self._get_vendor_by_environment(environment)
 
155
            if vendor is None:
 
156
                vendor = self._get_vendor_by_inspection()
 
157
                if vendor is None:
 
158
                    trace.mutter('falling back to default implementation')
 
159
                    vendor = self._default_ssh_vendor
 
160
                    if vendor is None:
 
161
                        raise errors.SSHVendorNotFound()
 
162
            self._cached_ssh_vendor = vendor
 
163
        return self._cached_ssh_vendor
 
164
 
 
165
_ssh_vendor_manager = SSHVendorManager()
 
166
_get_ssh_vendor = _ssh_vendor_manager.get_vendor
 
167
register_default_ssh_vendor = _ssh_vendor_manager.register_default_vendor
 
168
register_ssh_vendor = _ssh_vendor_manager.register_vendor
 
169
 
 
170
 
 
171
def _ignore_sigint():
 
172
    # TODO: This should possibly ignore SIGHUP as well, but bzr currently
 
173
    # doesn't handle it itself.
 
174
    # <https://launchpad.net/products/bzr/+bug/41433/+index>
 
175
    import signal
 
176
    signal.signal(signal.SIGINT, signal.SIG_IGN)
 
177
 
 
178
 
 
179
class SocketAsChannelAdapter(object):
 
180
    """Simple wrapper for a socket that pretends to be a paramiko Channel."""
 
181
 
 
182
    def __init__(self, sock):
 
183
        self.__socket = sock
 
184
 
 
185
    def get_name(self):
 
186
        return "bzr SocketAsChannelAdapter"
 
187
 
 
188
    def send(self, data):
 
189
        return self.__socket.send(data)
 
190
 
 
191
    def recv(self, n):
 
192
        try:
 
193
            return self.__socket.recv(n)
 
194
        except socket.error, e:
 
195
            if e.args[0] in (errno.EPIPE, errno.ECONNRESET, errno.ECONNABORTED,
 
196
                             errno.EBADF):
 
197
                # Connection has closed.  Paramiko expects an empty string in
 
198
                # this case, not an exception.
 
199
                return ''
 
200
            raise
 
201
 
 
202
    def recv_ready(self):
 
203
        # TODO: jam 20051215 this function is necessary to support the
 
204
        # pipelined() function. In reality, it probably should use
 
205
        # poll() or select() to actually return if there is data
 
206
        # available, otherwise we probably don't get any benefit
 
207
        return True
 
208
 
 
209
    def close(self):
 
210
        self.__socket.close()
 
211
 
 
212
 
 
213
class SSHVendor(object):
 
214
    """Abstract base class for SSH vendor implementations."""
 
215
 
 
216
    def connect_sftp(self, username, password, host, port):
 
217
        """Make an SSH connection, and return an SFTPClient.
 
218
 
 
219
        :param username: an ascii string
 
220
        :param password: an ascii string
 
221
        :param host: a host name as an ascii string
 
222
        :param port: a port number
 
223
        :type port: int
 
224
 
 
225
        :raises: ConnectionError if it cannot connect.
 
226
 
 
227
        :rtype: paramiko.sftp_client.SFTPClient
 
228
        """
 
229
        raise NotImplementedError(self.connect_sftp)
 
230
 
 
231
    def connect_ssh(self, username, password, host, port, command):
 
232
        """Make an SSH connection.
 
233
 
 
234
        :returns: something with a `close` method, and a `get_filelike_channels`
 
235
            method that returns a pair of (read, write) filelike objects.
 
236
        """
 
237
        raise NotImplementedError(self.connect_ssh)
 
238
 
 
239
    def _raise_connection_error(self, host, port=None, orig_error=None,
 
240
                                msg='Unable to connect to SSH host'):
 
241
        """Raise a SocketConnectionError with properly formatted host.
 
242
 
 
243
        This just unifies all the locations that try to raise ConnectionError,
 
244
        so that they format things properly.
 
245
        """
 
246
        raise errors.SocketConnectionError(host=host, port=port, msg=msg,
 
247
                                           orig_error=orig_error)
 
248
 
 
249
 
 
250
class LoopbackVendor(SSHVendor):
 
251
    """SSH "vendor" that connects over a plain TCP socket, not SSH."""
 
252
 
 
253
    def connect_sftp(self, username, password, host, port):
 
254
        sock = socket.socket()
 
255
        try:
 
256
            sock.connect((host, port))
 
257
        except socket.error, e:
 
258
            self._raise_connection_error(host, port=port, orig_error=e)
 
259
        return SFTPClient(SocketAsChannelAdapter(sock))
 
260
 
 
261
register_ssh_vendor('loopback', LoopbackVendor())
 
262
 
 
263
 
 
264
class _ParamikoSSHConnection(object):
 
265
    def __init__(self, channel):
 
266
        self.channel = channel
 
267
 
 
268
    def get_filelike_channels(self):
 
269
        return self.channel.makefile('rb'), self.channel.makefile('wb')
 
270
 
 
271
    def close(self):
 
272
        return self.channel.close()
 
273
 
 
274
 
 
275
class ParamikoVendor(SSHVendor):
 
276
    """Vendor that uses paramiko."""
 
277
 
 
278
    def _connect(self, username, password, host, port):
 
279
        global SYSTEM_HOSTKEYS, BZR_HOSTKEYS
 
280
 
 
281
        load_host_keys()
 
282
 
 
283
        try:
 
284
            t = paramiko.Transport((host, port or 22))
 
285
            t.set_log_channel('bzr.paramiko')
 
286
            t.start_client()
 
287
        except (paramiko.SSHException, socket.error), e:
 
288
            self._raise_connection_error(host, port=port, orig_error=e)
 
289
 
 
290
        server_key = t.get_remote_server_key()
 
291
        server_key_hex = paramiko.util.hexify(server_key.get_fingerprint())
 
292
        keytype = server_key.get_name()
 
293
        if host in SYSTEM_HOSTKEYS and keytype in SYSTEM_HOSTKEYS[host]:
 
294
            our_server_key = SYSTEM_HOSTKEYS[host][keytype]
 
295
            our_server_key_hex = paramiko.util.hexify(
 
296
                our_server_key.get_fingerprint())
 
297
        elif host in BZR_HOSTKEYS and keytype in BZR_HOSTKEYS[host]:
 
298
            our_server_key = BZR_HOSTKEYS[host][keytype]
 
299
            our_server_key_hex = paramiko.util.hexify(
 
300
                our_server_key.get_fingerprint())
 
301
        else:
 
302
            trace.warning('Adding %s host key for %s: %s'
 
303
                          % (keytype, host, server_key_hex))
 
304
            add = getattr(BZR_HOSTKEYS, 'add', None)
 
305
            if add is not None: # paramiko >= 1.X.X
 
306
                BZR_HOSTKEYS.add(host, keytype, server_key)
 
307
            else:
 
308
                BZR_HOSTKEYS.setdefault(host, {})[keytype] = server_key
 
309
            our_server_key = server_key
 
310
            our_server_key_hex = paramiko.util.hexify(
 
311
                our_server_key.get_fingerprint())
 
312
            save_host_keys()
 
313
        if server_key != our_server_key:
 
314
            filename1 = os.path.expanduser('~/.ssh/known_hosts')
 
315
            filename2 = osutils.pathjoin(config.config_dir(), 'ssh_host_keys')
 
316
            raise errors.TransportError(
 
317
                'Host keys for %s do not match!  %s != %s' %
 
318
                (host, our_server_key_hex, server_key_hex),
 
319
                ['Try editing %s or %s' % (filename1, filename2)])
 
320
 
 
321
        _paramiko_auth(username, password, host, port, t)
 
322
        return t
 
323
 
 
324
    def connect_sftp(self, username, password, host, port):
 
325
        t = self._connect(username, password, host, port)
 
326
        try:
 
327
            return t.open_sftp_client()
 
328
        except paramiko.SSHException, e:
 
329
            self._raise_connection_error(host, port=port, orig_error=e,
 
330
                                         msg='Unable to start sftp client')
 
331
 
 
332
    def connect_ssh(self, username, password, host, port, command):
 
333
        t = self._connect(username, password, host, port)
 
334
        try:
 
335
            channel = t.open_session()
 
336
            cmdline = ' '.join(command)
 
337
            channel.exec_command(cmdline)
 
338
            return _ParamikoSSHConnection(channel)
 
339
        except paramiko.SSHException, e:
 
340
            self._raise_connection_error(host, port=port, orig_error=e,
 
341
                                         msg='Unable to invoke remote bzr')
 
342
 
 
343
if paramiko is not None:
 
344
    vendor = ParamikoVendor()
 
345
    register_ssh_vendor('paramiko', vendor)
 
346
    register_ssh_vendor('none', vendor)
 
347
    register_default_ssh_vendor(vendor)
 
348
    _sftp_connection_errors = (EOFError, paramiko.SSHException)
 
349
    del vendor
 
350
else:
 
351
    _sftp_connection_errors = (EOFError,)
 
352
 
 
353
 
 
354
class SubprocessVendor(SSHVendor):
 
355
    """Abstract base class for vendors that use pipes to a subprocess."""
 
356
 
 
357
    def _connect(self, argv):
 
358
        proc = subprocess.Popen(argv,
 
359
                                stdin=subprocess.PIPE,
 
360
                                stdout=subprocess.PIPE,
 
361
                                **os_specific_subprocess_params())
 
362
        return SSHSubprocess(proc)
 
363
 
 
364
    def connect_sftp(self, username, password, host, port):
 
365
        try:
 
366
            argv = self._get_vendor_specific_argv(username, host, port,
 
367
                                                  subsystem='sftp')
 
368
            sock = self._connect(argv)
 
369
            return SFTPClient(SocketAsChannelAdapter(sock))
 
370
        except _sftp_connection_errors, e:
 
371
            self._raise_connection_error(host, port=port, orig_error=e)
 
372
        except (OSError, IOError), e:
 
373
            # If the machine is fast enough, ssh can actually exit
 
374
            # before we try and send it the sftp request, which
 
375
            # raises a Broken Pipe
 
376
            if e.errno not in (errno.EPIPE,):
 
377
                raise
 
378
            self._raise_connection_error(host, port=port, orig_error=e)
 
379
 
 
380
    def connect_ssh(self, username, password, host, port, command):
 
381
        try:
 
382
            argv = self._get_vendor_specific_argv(username, host, port,
 
383
                                                  command=command)
 
384
            return self._connect(argv)
 
385
        except (EOFError), e:
 
386
            self._raise_connection_error(host, port=port, orig_error=e)
 
387
        except (OSError, IOError), e:
 
388
            # If the machine is fast enough, ssh can actually exit
 
389
            # before we try and send it the sftp request, which
 
390
            # raises a Broken Pipe
 
391
            if e.errno not in (errno.EPIPE,):
 
392
                raise
 
393
            self._raise_connection_error(host, port=port, orig_error=e)
 
394
 
 
395
    def _get_vendor_specific_argv(self, username, host, port, subsystem=None,
 
396
                                  command=None):
 
397
        """Returns the argument list to run the subprocess with.
 
398
 
 
399
        Exactly one of 'subsystem' and 'command' must be specified.
 
400
        """
 
401
        raise NotImplementedError(self._get_vendor_specific_argv)
 
402
 
 
403
 
 
404
class OpenSSHSubprocessVendor(SubprocessVendor):
 
405
    """SSH vendor that uses the 'ssh' executable from OpenSSH."""
 
406
 
 
407
    def _get_vendor_specific_argv(self, username, host, port, subsystem=None,
 
408
                                  command=None):
 
409
        args = ['ssh',
 
410
                '-oForwardX11=no', '-oForwardAgent=no',
 
411
                '-oClearAllForwardings=yes', '-oProtocol=2',
 
412
                '-oNoHostAuthenticationForLocalhost=yes']
 
413
        if port is not None:
 
414
            args.extend(['-p', str(port)])
 
415
        if username is not None:
 
416
            args.extend(['-l', username])
 
417
        if subsystem is not None:
 
418
            args.extend(['-s', host, subsystem])
 
419
        else:
 
420
            args.extend([host] + command)
 
421
        return args
 
422
 
 
423
register_ssh_vendor('openssh', OpenSSHSubprocessVendor())
 
424
 
 
425
 
 
426
class SSHCorpSubprocessVendor(SubprocessVendor):
 
427
    """SSH vendor that uses the 'ssh' executable from SSH Corporation."""
 
428
 
 
429
    def _get_vendor_specific_argv(self, username, host, port, subsystem=None,
 
430
                                  command=None):
 
431
        args = ['ssh', '-x']
 
432
        if port is not None:
 
433
            args.extend(['-p', str(port)])
 
434
        if username is not None:
 
435
            args.extend(['-l', username])
 
436
        if subsystem is not None:
 
437
            args.extend(['-s', subsystem, host])
 
438
        else:
 
439
            args.extend([host] + command)
 
440
        return args
 
441
 
 
442
register_ssh_vendor('ssh', SSHCorpSubprocessVendor())
 
443
 
 
444
 
 
445
class PLinkSubprocessVendor(SubprocessVendor):
 
446
    """SSH vendor that uses the 'plink' executable from Putty."""
 
447
 
 
448
    def _get_vendor_specific_argv(self, username, host, port, subsystem=None,
 
449
                                  command=None):
 
450
        args = ['plink', '-x', '-a', '-ssh', '-2', '-batch']
 
451
        if port is not None:
 
452
            args.extend(['-P', str(port)])
 
453
        if username is not None:
 
454
            args.extend(['-l', username])
 
455
        if subsystem is not None:
 
456
            args.extend(['-s', host, subsystem])
 
457
        else:
 
458
            args.extend([host] + command)
 
459
        return args
 
460
 
 
461
register_ssh_vendor('plink', PLinkSubprocessVendor())
 
462
 
 
463
 
 
464
def _paramiko_auth(username, password, host, port, paramiko_transport):
 
465
    auth = config.AuthenticationConfig()
 
466
    # paramiko requires a username, but it might be none if nothing was
 
467
    # supplied.  If so, use the local username.
 
468
    if username is None:
 
469
        username = auth.get_user('ssh', host, port=port,
 
470
                                 default=getpass.getuser())
 
471
    if _use_ssh_agent:
 
472
        agent = paramiko.Agent()
 
473
        for key in agent.get_keys():
 
474
            trace.mutter('Trying SSH agent key %s'
 
475
                         % paramiko.util.hexify(key.get_fingerprint()))
 
476
            try:
 
477
                paramiko_transport.auth_publickey(username, key)
 
478
                return
 
479
            except paramiko.SSHException, e:
 
480
                pass
 
481
 
 
482
    # okay, try finding id_rsa or id_dss?  (posix only)
 
483
    if _try_pkey_auth(paramiko_transport, paramiko.RSAKey, username, 'id_rsa'):
 
484
        return
 
485
    if _try_pkey_auth(paramiko_transport, paramiko.DSSKey, username, 'id_dsa'):
 
486
        return
 
487
 
 
488
    # If we have gotten this far, we are about to try for passwords, do an
 
489
    # auth_none check to see if it is even supported.
 
490
    supported_auth_types = []
 
491
    try:
 
492
        # Note that with paramiko <1.7.5 this logs an INFO message:
 
493
        #    Authentication type (none) not permitted.
 
494
        # So we explicitly disable the logging level for this action
 
495
        old_level = paramiko_transport.logger.level
 
496
        paramiko_transport.logger.setLevel(logging.WARNING)
 
497
        try:
 
498
            paramiko_transport.auth_none(username)
 
499
        finally:
 
500
            paramiko_transport.logger.setLevel(old_level)
 
501
    except paramiko.BadAuthenticationType, e:
 
502
        # Supported methods are in the exception
 
503
        supported_auth_types = e.allowed_types
 
504
    except paramiko.SSHException, e:
 
505
        # Don't know what happened, but just ignore it
 
506
        pass
 
507
    if 'password' not in supported_auth_types:
 
508
        raise errors.ConnectionError('Unable to authenticate to SSH host as'
 
509
            '\n  %s@%s\nsupported auth types: %s'
 
510
            % (username, host, supported_auth_types))
 
511
 
 
512
    if password:
 
513
        try:
 
514
            paramiko_transport.auth_password(username, password)
 
515
            return
 
516
        except paramiko.SSHException, e:
 
517
            pass
 
518
 
 
519
    # give up and ask for a password
 
520
    password = auth.get_password('ssh', host, username, port=port)
 
521
    # get_password can still return None, which means we should not prompt
 
522
    if password is not None:
 
523
        try:
 
524
            paramiko_transport.auth_password(username, password)
 
525
        except paramiko.SSHException, e:
 
526
            raise errors.ConnectionError(
 
527
                'Unable to authenticate to SSH host as'
 
528
                '\n  %s@%s\n' % (username, host), e)
 
529
    else:
 
530
        raise errors.ConnectionError('Unable to authenticate to SSH host as'
 
531
                                     '  %s@%s' % (username, host))
 
532
 
 
533
 
 
534
def _try_pkey_auth(paramiko_transport, pkey_class, username, filename):
 
535
    filename = os.path.expanduser('~/.ssh/' + filename)
 
536
    try:
 
537
        key = pkey_class.from_private_key_file(filename)
 
538
        paramiko_transport.auth_publickey(username, key)
 
539
        return True
 
540
    except paramiko.PasswordRequiredException:
 
541
        password = ui.ui_factory.get_password(
 
542
            prompt='SSH %(filename)s password', filename=filename)
 
543
        try:
 
544
            key = pkey_class.from_private_key_file(filename, password)
 
545
            paramiko_transport.auth_publickey(username, key)
 
546
            return True
 
547
        except paramiko.SSHException:
 
548
            trace.mutter('SSH authentication via %s key failed.'
 
549
                         % (os.path.basename(filename),))
 
550
    except paramiko.SSHException:
 
551
        trace.mutter('SSH authentication via %s key failed.'
 
552
                     % (os.path.basename(filename),))
 
553
    except IOError:
 
554
        pass
 
555
    return False
 
556
 
 
557
 
 
558
def load_host_keys():
 
559
    """
 
560
    Load system host keys (probably doesn't work on windows) and any
 
561
    "discovered" keys from previous sessions.
 
562
    """
 
563
    global SYSTEM_HOSTKEYS, BZR_HOSTKEYS
 
564
    try:
 
565
        SYSTEM_HOSTKEYS = paramiko.util.load_host_keys(
 
566
            os.path.expanduser('~/.ssh/known_hosts'))
 
567
    except IOError, e:
 
568
        trace.mutter('failed to load system host keys: ' + str(e))
 
569
    bzr_hostkey_path = osutils.pathjoin(config.config_dir(), 'ssh_host_keys')
 
570
    try:
 
571
        BZR_HOSTKEYS = paramiko.util.load_host_keys(bzr_hostkey_path)
 
572
    except IOError, e:
 
573
        trace.mutter('failed to load bzr host keys: ' + str(e))
 
574
        save_host_keys()
 
575
 
 
576
 
 
577
def save_host_keys():
 
578
    """
 
579
    Save "discovered" host keys in $(config)/ssh_host_keys/.
 
580
    """
 
581
    global SYSTEM_HOSTKEYS, BZR_HOSTKEYS
 
582
    bzr_hostkey_path = osutils.pathjoin(config.config_dir(), 'ssh_host_keys')
 
583
    config.ensure_config_dir_exists()
 
584
 
 
585
    try:
 
586
        f = open(bzr_hostkey_path, 'w')
 
587
        f.write('# SSH host keys collected by bzr\n')
 
588
        for hostname, keys in BZR_HOSTKEYS.iteritems():
 
589
            for keytype, key in keys.iteritems():
 
590
                f.write('%s %s %s\n' % (hostname, keytype, key.get_base64()))
 
591
        f.close()
 
592
    except IOError, e:
 
593
        trace.mutter('failed to save bzr host keys: ' + str(e))
 
594
 
 
595
 
 
596
def os_specific_subprocess_params():
 
597
    """Get O/S specific subprocess parameters."""
 
598
    if sys.platform == 'win32':
 
599
        # setting the process group and closing fds is not supported on
 
600
        # win32
 
601
        return {}
 
602
    else:
 
603
        # We close fds other than the pipes as the child process does not need
 
604
        # them to be open.
 
605
        #
 
606
        # We also set the child process to ignore SIGINT.  Normally the signal
 
607
        # would be sent to every process in the foreground process group, but
 
608
        # this causes it to be seen only by bzr and not by ssh.  Python will
 
609
        # generate a KeyboardInterrupt in bzr, and we will then have a chance
 
610
        # to release locks or do other cleanup over ssh before the connection
 
611
        # goes away.
 
612
        # <https://launchpad.net/products/bzr/+bug/5987>
 
613
        #
 
614
        # Running it in a separate process group is not good because then it
 
615
        # can't get non-echoed input of a password or passphrase.
 
616
        # <https://launchpad.net/products/bzr/+bug/40508>
 
617
        return {'preexec_fn': _ignore_sigint,
 
618
                'close_fds': True,
 
619
                }
 
620
 
 
621
 
 
622
class SSHSubprocess(object):
 
623
    """A socket-like object that talks to an ssh subprocess via pipes."""
 
624
 
 
625
    def __init__(self, proc):
 
626
        self.proc = proc
 
627
 
 
628
    def send(self, data):
 
629
        return os.write(self.proc.stdin.fileno(), data)
 
630
 
 
631
    def recv(self, count):
 
632
        return os.read(self.proc.stdout.fileno(), count)
 
633
 
 
634
    def close(self):
 
635
        self.proc.stdin.close()
 
636
        self.proc.stdout.close()
 
637
        self.proc.wait()
 
638
 
 
639
    def get_filelike_channels(self):
 
640
        return (self.proc.stdout, self.proc.stdin)
 
641