1
# Copyright (C) 2005 Robey Pointer <robey@lag.net>, Canonical Ltd
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11
# GNU General Public License for more details.
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17
"""Implementation of Transport over SFTP, using paramiko."""
32
from bzrlib.config import config_dir, ensure_config_dir_exists
33
from bzrlib.errors import (ConnectionError,
35
TransportNotPossible, NoSuchFile, PathNotChild,
37
LockError, InvalidURL, ParamikoNotPresent
39
from bzrlib.osutils import pathjoin, fancy_rename
40
from bzrlib.trace import mutter, warning, error
41
from bzrlib.transport import Transport, Server, urlescape, urlunescape
46
except ImportError, e:
47
raise ParamikoNotPresent(e)
49
from paramiko.sftp import (SFTP_FLAG_WRITE, SFTP_FLAG_CREATE,
50
SFTP_FLAG_EXCL, SFTP_FLAG_TRUNC,
52
from paramiko.sftp_attr import SFTPAttributes
53
from paramiko.sftp_file import SFTPFile
54
from paramiko.sftp_client import SFTPClient
56
if 'sftp' not in urlparse.uses_netloc:
57
urlparse.uses_netloc.append('sftp')
59
# don't use prefetch unless paramiko version >= 1.5.2 (there were bugs earlier)
60
_default_do_prefetch = False
61
if getattr(paramiko, '__version_info__', (0, 0, 0)) >= (1, 5, 2):
62
_default_do_prefetch = True
66
if sys.platform == 'win32':
67
# close_fds not supported on win32
72
def _get_ssh_vendor():
73
"""Find out what version of SSH is on the system."""
75
if _ssh_vendor is not None:
80
if 'BZR_SSH' in os.environ:
81
_ssh_vendor = os.environ['BZR_SSH']
82
if _ssh_vendor == 'paramiko':
87
p = subprocess.Popen(['ssh', '-V'],
89
stdin=subprocess.PIPE,
90
stdout=subprocess.PIPE,
91
stderr=subprocess.PIPE)
92
returncode = p.returncode
93
stdout, stderr = p.communicate()
97
if 'OpenSSH' in stderr:
98
mutter('ssh implementation is OpenSSH')
99
_ssh_vendor = 'openssh'
100
elif 'SSH Secure Shell' in stderr:
101
mutter('ssh implementation is SSH Corp.')
104
if _ssh_vendor != 'none':
107
# XXX: 20051123 jamesh
108
# A check for putty's plink or lsh would go here.
110
mutter('falling back to paramiko implementation')
114
class SFTPSubprocess:
115
"""A socket-like object that talks to an ssh subprocess via pipes."""
116
def __init__(self, hostname, vendor, port=None, user=None):
117
assert vendor in ['openssh', 'ssh']
118
if vendor == 'openssh':
120
'-oForwardX11=no', '-oForwardAgent=no',
121
'-oClearAllForwardings=yes', '-oProtocol=2',
122
'-oNoHostAuthenticationForLocalhost=yes']
124
args.extend(['-p', str(port)])
126
args.extend(['-l', user])
127
args.extend(['-s', hostname, 'sftp'])
128
elif vendor == 'ssh':
131
args.extend(['-p', str(port)])
133
args.extend(['-l', user])
134
args.extend(['-s', 'sftp', hostname])
136
self.proc = subprocess.Popen(args, close_fds=_close_fds,
137
stdin=subprocess.PIPE,
138
stdout=subprocess.PIPE)
140
def send(self, data):
141
return os.write(self.proc.stdin.fileno(), data)
143
def recv_ready(self):
144
# TODO: jam 20051215 this function is necessary to support the
145
# pipelined() function. In reality, it probably should use
146
# poll() or select() to actually return if there is data
147
# available, otherwise we probably don't get any benefit
150
def recv(self, count):
151
return os.read(self.proc.stdout.fileno(), count)
154
self.proc.stdin.close()
155
self.proc.stdout.close()
159
class LoopbackSFTP(object):
160
"""Simple wrapper for a socket that pretends to be a paramiko Channel."""
162
def __init__(self, sock):
165
def send(self, data):
166
return self.__socket.send(data)
169
return self.__socket.recv(n)
171
def recv_ready(self):
175
self.__socket.close()
181
# This is a weakref dictionary, so that we can reuse connections
182
# that are still active. Long term, it might be nice to have some
183
# sort of expiration policy, such as disconnect if inactive for
184
# X seconds. But that requires a lot more fanciness.
185
_connected_hosts = weakref.WeakValueDictionary()
188
def load_host_keys():
190
Load system host keys (probably doesn't work on windows) and any
191
"discovered" keys from previous sessions.
193
global SYSTEM_HOSTKEYS, BZR_HOSTKEYS
195
SYSTEM_HOSTKEYS = paramiko.util.load_host_keys(os.path.expanduser('~/.ssh/known_hosts'))
197
mutter('failed to load system host keys: ' + str(e))
198
bzr_hostkey_path = pathjoin(config_dir(), 'ssh_host_keys')
200
BZR_HOSTKEYS = paramiko.util.load_host_keys(bzr_hostkey_path)
202
mutter('failed to load bzr host keys: ' + str(e))
206
def save_host_keys():
208
Save "discovered" host keys in $(config)/ssh_host_keys/.
210
global SYSTEM_HOSTKEYS, BZR_HOSTKEYS
211
bzr_hostkey_path = pathjoin(config_dir(), 'ssh_host_keys')
212
ensure_config_dir_exists()
215
f = open(bzr_hostkey_path, 'w')
216
f.write('# SSH host keys collected by bzr\n')
217
for hostname, keys in BZR_HOSTKEYS.iteritems():
218
for keytype, key in keys.iteritems():
219
f.write('%s %s %s\n' % (hostname, keytype, key.get_base64()))
222
mutter('failed to save bzr host keys: ' + str(e))
225
class SFTPLock(object):
226
"""This fakes a lock in a remote location."""
227
__slots__ = ['path', 'lock_path', 'lock_file', 'transport']
228
def __init__(self, path, transport):
229
assert isinstance(transport, SFTPTransport)
231
self.lock_file = None
233
self.lock_path = path + '.write-lock'
234
self.transport = transport
236
# RBC 20060103 FIXME should we be using private methods here ?
237
abspath = transport._remote_path(self.lock_path)
238
self.lock_file = transport._sftp_open_exclusive(abspath)
240
raise LockError('File %r already locked' % (self.path,))
243
"""Should this warn, or actually try to cleanup?"""
245
warning("SFTPLock %r not explicitly unlocked" % (self.path,))
249
if not self.lock_file:
251
self.lock_file.close()
252
self.lock_file = None
254
self.transport.delete(self.lock_path)
255
except (NoSuchFile,):
256
# What specific errors should we catch here?
259
class SFTPTransport (Transport):
261
Transport implementation for SFTP access.
263
_do_prefetch = _default_do_prefetch
265
def __init__(self, base, clone_from=None):
266
assert base.startswith('sftp://')
267
self._parse_url(base)
268
base = self._unparse_url()
271
super(SFTPTransport, self).__init__(base)
272
if clone_from is None:
275
# use the same ssh connection, etc
276
self._sftp = clone_from._sftp
277
# super saves 'self.base'
279
def should_cache(self):
281
Return True if the data pulled across should be cached locally.
285
def clone(self, offset=None):
287
Return a new SFTPTransport with root at self.base + offset.
288
We share the same SFTP session between such transports, because it's
289
fairly expensive to set them up.
292
return SFTPTransport(self.base, self)
294
return SFTPTransport(self.abspath(offset), self)
296
def abspath(self, relpath):
298
Return the full url to the given relative path.
300
@param relpath: the relative path or path components
301
@type relpath: str or list
303
return self._unparse_url(self._remote_path(relpath))
305
def _remote_path(self, relpath):
306
"""Return the path to be passed along the sftp protocol for relpath.
308
relpath is a urlencoded string.
310
# FIXME: share the common code across transports
311
assert isinstance(relpath, basestring)
312
relpath = urlunescape(relpath).split('/')
313
basepath = self._path.split('/')
314
if len(basepath) > 0 and basepath[-1] == '':
315
basepath = basepath[:-1]
319
if len(basepath) == 0:
320
# In most filesystems, a request for the parent
321
# of root, just returns root.
329
path = '/'.join(basepath)
332
def relpath(self, abspath):
333
username, password, host, port, path = self._split_url(abspath)
335
if (username != self._username):
336
error.append('username mismatch')
337
if (host != self._host):
338
error.append('host mismatch')
339
if (port != self._port):
340
error.append('port mismatch')
341
if (not path.startswith(self._path)):
342
error.append('path mismatch')
344
extra = ': ' + ', '.join(error)
345
raise PathNotChild(abspath, self.base, extra=extra)
347
return path[pl:].strip('/')
349
def has(self, relpath):
351
Does the target location exist?
354
self._sftp.stat(self._remote_path(relpath))
359
def get(self, relpath, decode=False):
361
Get the file at the given relative path.
363
:param relpath: The relative path to the file
366
path = self._remote_path(relpath)
367
f = self._sftp.file(path, mode='rb')
368
if self._do_prefetch and (getattr(f, 'prefetch', None) is not None):
371
except (IOError, paramiko.SSHException), e:
372
self._translate_io_exception(e, path, ': error retrieving')
374
def get_partial(self, relpath, start, length=None):
376
Get just part of a file.
378
:param relpath: Path to the file, relative to base
379
:param start: The starting position to read from
380
:param length: The length to read. A length of None indicates
381
read to the end of the file.
382
:return: A file-like object containing at least the specified bytes.
383
Some implementations may return objects which can be read
384
past this length, but this is not guaranteed.
386
# TODO: implement get_partial_multi to help with knit support
387
f = self.get(relpath)
389
if self._do_prefetch and hasattr(f, 'prefetch'):
393
def put(self, relpath, f, mode=None):
395
Copy the file-like or string object into the location.
397
:param relpath: Location to put the contents, relative to base.
398
:param f: File-like or string object.
399
:param mode: The final mode for the file
401
final_path = self._remote_path(relpath)
402
self._put(final_path, f, mode=mode)
404
def _put(self, abspath, f, mode=None):
405
"""Helper function so both put() and copy_abspaths can reuse the code"""
406
tmp_abspath = '%s.tmp.%.9f.%d.%d' % (abspath, time.time(),
407
os.getpid(), random.randint(0,0x7FFFFFFF))
408
fout = self._sftp_open_exclusive(tmp_abspath, mode=mode)
412
fout.set_pipelined(True)
414
except (IOError, paramiko.SSHException), e:
415
self._translate_io_exception(e, tmp_abspath)
417
self._sftp.chmod(tmp_abspath, mode)
420
self._rename_and_overwrite(tmp_abspath, abspath)
422
# If we fail, try to clean up the temporary file
423
# before we throw the exception
424
# but don't let another exception mess things up
425
# Write out the traceback, because otherwise
426
# the catch and throw destroys it
428
mutter(traceback.format_exc())
432
self._sftp.remove(tmp_abspath)
434
# raise the saved except
436
# raise the original with its traceback if we can.
439
def iter_files_recursive(self):
440
"""Walk the relative paths of all files in this transport."""
441
queue = list(self.list_dir('.'))
443
relpath = urllib.quote(queue.pop(0))
444
st = self.stat(relpath)
445
if stat.S_ISDIR(st.st_mode):
446
for i, basename in enumerate(self.list_dir(relpath)):
447
queue.insert(i, relpath+'/'+basename)
451
def mkdir(self, relpath, mode=None):
452
"""Create a directory at the given path."""
454
path = self._remote_path(relpath)
455
# In the paramiko documentation, it says that passing a mode flag
456
# will filtered against the server umask.
457
# StubSFTPServer does not do this, which would be nice, because it is
458
# what we really want :)
459
# However, real servers do use umask, so we really should do it that way
460
self._sftp.mkdir(path)
462
self._sftp.chmod(path, mode=mode)
463
except (paramiko.SSHException, IOError), e:
464
self._translate_io_exception(e, path, ': unable to mkdir',
465
failure_exc=FileExists)
467
def _translate_io_exception(self, e, path, more_info='', failure_exc=NoSuchFile):
468
"""Translate a paramiko or IOError into a friendlier exception.
470
:param e: The original exception
471
:param path: The path in question when the error is raised
472
:param more_info: Extra information that can be included,
473
such as what was going on
474
:param failure_exc: Paramiko has the super fun ability to raise completely
475
opaque errors that just set "e.args = ('Failure',)" with
477
This sometimes means FileExists, but it also sometimes
480
# paramiko seems to generate detailless errors.
481
self._translate_error(e, path, raise_generic=False)
482
if hasattr(e, 'args'):
483
if (e.args == ('No such file or directory',) or
484
e.args == ('No such file',)):
485
raise NoSuchFile(path, str(e) + more_info)
486
if (e.args == ('mkdir failed',)):
487
raise FileExists(path, str(e) + more_info)
488
# strange but true, for the paramiko server.
489
if (e.args == ('Failure',)):
490
raise failure_exc(path, str(e) + more_info)
491
mutter('Raising exception with args %s', e.args)
492
if hasattr(e, 'errno'):
493
mutter('Raising exception with errno %s', e.errno)
496
def append(self, relpath, f):
498
Append the text in the file-like object into the final
502
path = self._remote_path(relpath)
503
fout = self._sftp.file(path, 'ab')
507
except (IOError, paramiko.SSHException), e:
508
self._translate_io_exception(e, relpath, ': unable to append')
510
def rename(self, rel_from, rel_to):
511
"""Rename without special overwriting"""
513
self._sftp.rename(self._remote_path(rel_from),
514
self._remote_path(rel_to))
515
except (IOError, paramiko.SSHException), e:
516
self._translate_io_exception(e, rel_from,
517
': unable to rename to %r' % (rel_to))
519
def _rename_and_overwrite(self, abs_from, abs_to):
520
"""Do a fancy rename on the remote server.
522
Using the implementation provided by osutils.
525
fancy_rename(abs_from, abs_to,
526
rename_func=self._sftp.rename,
527
unlink_func=self._sftp.remove)
528
except (IOError, paramiko.SSHException), e:
529
self._translate_io_exception(e, abs_from, ': unable to rename to %r' % (abs_to))
531
def move(self, rel_from, rel_to):
532
"""Move the item at rel_from to the location at rel_to"""
533
path_from = self._remote_path(rel_from)
534
path_to = self._remote_path(rel_to)
535
self._rename_and_overwrite(path_from, path_to)
537
def delete(self, relpath):
538
"""Delete the item at relpath"""
539
path = self._remote_path(relpath)
541
self._sftp.remove(path)
542
except (IOError, paramiko.SSHException), e:
543
self._translate_io_exception(e, path, ': unable to delete')
546
"""Return True if this store supports listing."""
549
def list_dir(self, relpath):
551
Return a list of all files at the given location.
553
# does anything actually use this?
554
path = self._remote_path(relpath)
556
return self._sftp.listdir(path)
557
except (IOError, paramiko.SSHException), e:
558
self._translate_io_exception(e, path, ': failed to list_dir')
560
def rmdir(self, relpath):
561
"""See Transport.rmdir."""
562
path = self._remote_path(relpath)
564
return self._sftp.rmdir(path)
565
except (IOError, paramiko.SSHException), e:
566
self._translate_io_exception(e, path, ': failed to rmdir')
568
def stat(self, relpath):
569
"""Return the stat information for a file."""
570
path = self._remote_path(relpath)
572
return self._sftp.stat(path)
573
except (IOError, paramiko.SSHException), e:
574
self._translate_io_exception(e, path, ': unable to stat')
576
def lock_read(self, relpath):
578
Lock the given file for shared (read) access.
579
:return: A lock object, which has an unlock() member function
581
# FIXME: there should be something clever i can do here...
582
class BogusLock(object):
583
def __init__(self, path):
587
return BogusLock(relpath)
589
def lock_write(self, relpath):
591
Lock the given file for exclusive (write) access.
592
WARNING: many transports do not support this, so trying avoid using it
594
:return: A lock object, which has an unlock() member function
596
# This is a little bit bogus, but basically, we create a file
597
# which should not already exist, and if it does, we assume
598
# that there is a lock, and if it doesn't, the we assume
599
# that we have taken the lock.
600
return SFTPLock(relpath, self)
602
def _unparse_url(self, path=None):
605
path = urllib.quote(path)
606
# handle homedir paths
607
if not path.startswith('/'):
609
netloc = urllib.quote(self._host)
610
if self._username is not None:
611
netloc = '%s@%s' % (urllib.quote(self._username), netloc)
612
if self._port is not None:
613
netloc = '%s:%d' % (netloc, self._port)
615
return urlparse.urlunparse(('sftp', netloc, path, '', '', ''))
617
def _split_url(self, url):
618
if isinstance(url, unicode):
619
# TODO: Disallow unicode urls
620
#raise InvalidURL(url, 'urls must not be unicode.')
621
url = url.encode('ascii')
622
(scheme, netloc, path, params,
623
query, fragment) = urlparse.urlparse(url, allow_fragments=False)
624
assert scheme == 'sftp'
625
username = password = host = port = None
627
username, host = netloc.split('@', 1)
629
username, password = username.split(':', 1)
630
password = urllib.unquote(password)
631
username = urllib.unquote(username)
636
host, port = host.rsplit(':', 1)
640
# TODO: Should this be ConnectionError?
641
raise TransportError('%s: invalid port number' % port)
642
host = urllib.unquote(host)
644
path = urlunescape(path)
646
# the initial slash should be removed from the path, and treated
647
# as a homedir relative path (the path begins with a double slash
648
# if it is absolute).
649
# see draft-ietf-secsh-scp-sftp-ssh-uri-03.txt
650
# RBC 20060118 we are not using this as its too user hostile. instead
651
# we are following lftp and using /~/foo to mean '~/foo'.
652
# handle homedir paths
653
if path.startswith('/~/'):
657
return (username, password, host, port, path)
659
def _parse_url(self, url):
660
(self._username, self._password,
661
self._host, self._port, self._path) = self._split_url(url)
663
def _sftp_connect(self):
664
"""Connect to the remote sftp server.
665
After this, self._sftp should have a valid connection (or
666
we raise an TransportError 'could not connect').
668
TODO: Raise a more reasonable ConnectionFailed exception
670
global _connected_hosts
672
idx = (self._host, self._port, self._username)
674
self._sftp = _connected_hosts[idx]
679
vendor = _get_ssh_vendor()
680
if vendor == 'loopback':
681
sock = socket.socket()
682
sock.connect((self._host, self._port))
683
self._sftp = SFTPClient(LoopbackSFTP(sock))
684
elif vendor != 'none':
685
sock = SFTPSubprocess(self._host, vendor, self._port,
687
self._sftp = SFTPClient(sock)
689
self._paramiko_connect()
691
_connected_hosts[idx] = self._sftp
693
def _paramiko_connect(self):
694
global SYSTEM_HOSTKEYS, BZR_HOSTKEYS
699
t = paramiko.Transport((self._host, self._port or 22))
700
t.set_log_channel('bzr.paramiko')
702
except paramiko.SSHException, e:
703
raise ConnectionError('Unable to reach SSH host %s:%d' %
704
(self._host, self._port), e)
706
server_key = t.get_remote_server_key()
707
server_key_hex = paramiko.util.hexify(server_key.get_fingerprint())
708
keytype = server_key.get_name()
709
if SYSTEM_HOSTKEYS.has_key(self._host) and SYSTEM_HOSTKEYS[self._host].has_key(keytype):
710
our_server_key = SYSTEM_HOSTKEYS[self._host][keytype]
711
our_server_key_hex = paramiko.util.hexify(our_server_key.get_fingerprint())
712
elif BZR_HOSTKEYS.has_key(self._host) and BZR_HOSTKEYS[self._host].has_key(keytype):
713
our_server_key = BZR_HOSTKEYS[self._host][keytype]
714
our_server_key_hex = paramiko.util.hexify(our_server_key.get_fingerprint())
716
warning('Adding %s host key for %s: %s' % (keytype, self._host, server_key_hex))
717
if not BZR_HOSTKEYS.has_key(self._host):
718
BZR_HOSTKEYS[self._host] = {}
719
BZR_HOSTKEYS[self._host][keytype] = server_key
720
our_server_key = server_key
721
our_server_key_hex = paramiko.util.hexify(our_server_key.get_fingerprint())
723
if server_key != our_server_key:
724
filename1 = os.path.expanduser('~/.ssh/known_hosts')
725
filename2 = pathjoin(config_dir(), 'ssh_host_keys')
726
raise TransportError('Host keys for %s do not match! %s != %s' % \
727
(self._host, our_server_key_hex, server_key_hex),
728
['Try editing %s or %s' % (filename1, filename2)])
733
self._sftp = t.open_sftp_client()
734
except paramiko.SSHException, e:
735
raise ConnectionError('Unable to start sftp client %s:%d' %
736
(self._host, self._port), e)
738
def _sftp_auth(self, transport):
739
# paramiko requires a username, but it might be none if nothing was supplied
740
# use the local username, just in case.
741
# We don't override self._username, because if we aren't using paramiko,
742
# the username might be specified in ~/.ssh/config and we don't want to
743
# force it to something else
744
# Also, it would mess up the self.relpath() functionality
745
username = self._username or getpass.getuser()
747
# Paramiko tries to open a socket.AF_UNIX in order to connect
748
# to ssh-agent. That attribute doesn't exist on win32 (it does in cygwin)
749
# so we get an AttributeError exception. For now, just don't try to
750
# connect to an agent if we are on win32
751
if sys.platform != 'win32':
752
agent = paramiko.Agent()
753
for key in agent.get_keys():
754
mutter('Trying SSH agent key %s' % paramiko.util.hexify(key.get_fingerprint()))
756
transport.auth_publickey(username, key)
758
except paramiko.SSHException, e:
761
# okay, try finding id_rsa or id_dss? (posix only)
762
if self._try_pkey_auth(transport, paramiko.RSAKey, username, 'id_rsa'):
764
if self._try_pkey_auth(transport, paramiko.DSSKey, username, 'id_dsa'):
769
transport.auth_password(username, self._password)
771
except paramiko.SSHException, e:
774
# FIXME: Don't keep a password held in memory if you can help it
775
#self._password = None
777
# give up and ask for a password
778
password = bzrlib.ui.ui_factory.get_password(
779
prompt='SSH %(user)s@%(host)s password',
780
user=username, host=self._host)
782
transport.auth_password(username, password)
783
except paramiko.SSHException, e:
784
raise ConnectionError('Unable to authenticate to SSH host as %s@%s' %
785
(username, self._host), e)
787
def _try_pkey_auth(self, transport, pkey_class, username, filename):
788
filename = os.path.expanduser('~/.ssh/' + filename)
790
key = pkey_class.from_private_key_file(filename)
791
transport.auth_publickey(username, key)
793
except paramiko.PasswordRequiredException:
794
password = bzrlib.ui.ui_factory.get_password(
795
prompt='SSH %(filename)s password',
798
key = pkey_class.from_private_key_file(filename, password)
799
transport.auth_publickey(username, key)
801
except paramiko.SSHException:
802
mutter('SSH authentication via %s key failed.' % (os.path.basename(filename),))
803
except paramiko.SSHException:
804
mutter('SSH authentication via %s key failed.' % (os.path.basename(filename),))
809
def _sftp_open_exclusive(self, abspath, mode=None):
810
"""Open a remote path exclusively.
812
SFTP supports O_EXCL (SFTP_FLAG_EXCL), which fails if
813
the file already exists. However it does not expose this
814
at the higher level of SFTPClient.open(), so we have to
817
WARNING: This breaks the SFTPClient abstraction, so it
818
could easily break against an updated version of paramiko.
820
:param abspath: The remote absolute path where the file should be opened
821
:param mode: The mode permissions bits for the new file
823
path = self._sftp._adjust_cwd(abspath)
824
attr = SFTPAttributes()
827
omode = (SFTP_FLAG_WRITE | SFTP_FLAG_CREATE
828
| SFTP_FLAG_TRUNC | SFTP_FLAG_EXCL)
830
t, msg = self._sftp._request(CMD_OPEN, path, omode, attr)
832
raise TransportError('Expected an SFTP handle')
833
handle = msg.get_string()
834
return SFTPFile(self._sftp, handle, 'wb', -1)
835
except (paramiko.SSHException, IOError), e:
836
self._translate_io_exception(e, abspath, ': unable to open',
837
failure_exc=FileExists)
840
# ------------- server test implementation --------------
844
from bzrlib.tests.stub_sftp import StubServer, StubSFTPServer
846
STUB_SERVER_KEY = """
847
-----BEGIN RSA PRIVATE KEY-----
848
MIICWgIBAAKBgQDTj1bqB4WmayWNPB+8jVSYpZYk80Ujvj680pOTh2bORBjbIAyz
849
oWGW+GUjzKxTiiPvVmxFgx5wdsFvF03v34lEVVhMpouqPAYQ15N37K/ir5XY+9m/
850
d8ufMCkjeXsQkKqFbAlQcnWMCRnOoPHS3I4vi6hmnDDeeYTSRvfLbW0fhwIBIwKB
851
gBIiOqZYaoqbeD9OS9z2K9KR2atlTxGxOJPXiP4ESqP3NVScWNwyZ3NXHpyrJLa0
852
EbVtzsQhLn6rF+TzXnOlcipFvjsem3iYzCpuChfGQ6SovTcOjHV9z+hnpXvQ/fon
853
soVRZY65wKnF7IAoUwTmJS9opqgrN6kRgCd3DASAMd1bAkEA96SBVWFt/fJBNJ9H
854
tYnBKZGw0VeHOYmVYbvMSstssn8un+pQpUm9vlG/bp7Oxd/m+b9KWEh2xPfv6zqU
855
avNwHwJBANqzGZa/EpzF4J8pGti7oIAPUIDGMtfIcmqNXVMckrmzQ2vTfqtkEZsA
856
4rE1IERRyiJQx6EJsz21wJmGV9WJQ5kCQQDwkS0uXqVdFzgHO6S++tjmjYcxwr3g
857
H0CoFYSgbddOT6miqRskOQF3DZVkJT3kyuBgU2zKygz52ukQZMqxCb1fAkASvuTv
858
qfpH87Qq5kQhNKdbbwbmd2NxlNabazPijWuphGTdW0VfJdWfklyS2Kr+iqrs/5wV
859
HhathJt636Eg7oIjAkA8ht3MQ+XSl9yIJIS8gVpbPxSw5OMfw0PjVE7tBdQruiSc
860
nvuQES5C9BMHjF39LZiGH1iLQy7FgdHyoP+eodI7
861
-----END RSA PRIVATE KEY-----
865
class SingleListener(threading.Thread):
867
def __init__(self, callback):
868
threading.Thread.__init__(self)
869
self._callback = callback
870
self._socket = socket.socket()
871
self._socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
872
self._socket.bind(('localhost', 0))
873
self._socket.listen(1)
874
self.port = self._socket.getsockname()[1]
875
self.stop_event = threading.Event()
878
s, _ = self._socket.accept()
879
# now close the listen socket
882
self._callback(s, self.stop_event)
884
pass #Ignore socket errors
886
# probably a failed test
887
warning('Exception from within unit test server thread: %r' % x)
890
self.stop_event.set()
891
# use a timeout here, because if the test fails, the server thread may
892
# never notice the stop_event.
896
class SFTPServer(Server):
897
"""Common code for SFTP server facilities."""
900
self._original_vendor = None
902
self._server_homedir = None
903
self._listener = None
905
self._vendor = 'none'
909
def _get_sftp_url(self, path):
910
"""Calculate an sftp url to this server for path."""
911
return 'sftp://foo:bar@localhost:%d/%s' % (self._listener.port, path)
913
def log(self, message):
914
"""StubServer uses this to log when a new server is created."""
915
self.logs.append(message)
917
def _run_server(self, s, stop_event):
918
ssh_server = paramiko.Transport(s)
919
key_file = os.path.join(self._homedir, 'test_rsa.key')
920
file(key_file, 'w').write(STUB_SERVER_KEY)
921
host_key = paramiko.RSAKey.from_private_key_file(key_file)
922
ssh_server.add_server_key(host_key)
923
server = StubServer(self)
924
ssh_server.set_subsystem_handler('sftp', paramiko.SFTPServer,
925
StubSFTPServer, root=self._root,
926
home=self._server_homedir)
927
event = threading.Event()
928
ssh_server.start_server(event, server)
930
stop_event.wait(30.0)
934
self._original_vendor = _ssh_vendor
935
_ssh_vendor = self._vendor
936
self._homedir = os.getcwdu()
937
if self._server_homedir is None:
938
self._server_homedir = self._homedir
940
# FIXME WINDOWS: _root should be _server_homedir[0]:/
941
self._listener = SingleListener(self._run_server)
942
self._listener.setDaemon(True)
943
self._listener.start()
946
"""See bzrlib.transport.Server.tearDown."""
948
self._listener.stop()
949
_ssh_vendor = self._original_vendor
952
class SFTPFullAbsoluteServer(SFTPServer):
953
"""A test server for sftp transports, using absolute urls and ssh."""
956
"""See bzrlib.transport.Server.get_url."""
957
return self._get_sftp_url(urlescape(self._homedir[1:]))
960
class SFTPServerWithoutSSH(SFTPServer):
961
"""An SFTP server that uses a simple TCP socket pair rather than SSH."""
964
super(SFTPServerWithoutSSH, self).__init__()
965
self._vendor = 'loopback'
967
def _run_server(self, sock, stop_event):
968
class FakeChannel(object):
969
def get_transport(self):
971
def get_log_channel(self):
975
def get_hexdump(self):
978
server = paramiko.SFTPServer(FakeChannel(), 'sftp', StubServer(self), StubSFTPServer,
979
root=self._root, home=self._server_homedir)
980
server.start_subsystem('sftp', None, sock)
981
server.finish_subsystem()
984
class SFTPAbsoluteServer(SFTPServerWithoutSSH):
985
"""A test server for sftp transports, using absolute urls."""
988
"""See bzrlib.transport.Server.get_url."""
989
return self._get_sftp_url(urlescape(self._homedir[1:]))
992
class SFTPHomeDirServer(SFTPServerWithoutSSH):
993
"""A test server for sftp transports, using homedir relative urls."""
996
"""See bzrlib.transport.Server.get_url."""
997
return self._get_sftp_url("~/")
1000
class SFTPSiblingAbsoluteServer(SFTPAbsoluteServer):
1001
"""A test servere for sftp transports, using absolute urls to non-home."""
1004
self._server_homedir = '/dev/noone/runs/tests/here'
1005
super(SFTPSiblingAbsoluteServer, self).setUp()
1008
def get_test_permutations():
1009
"""Return the permutations to be used in testing."""
1010
return [(SFTPTransport, SFTPAbsoluteServer),
1011
(SFTPTransport, SFTPHomeDirServer),
1012
(SFTPTransport, SFTPSiblingAbsoluteServer),