1
# Copyright (C) 2005 Robey Pointer <robey@lag.net>
2
# Copyright (C) 2005, 2006, 2007 Canonical Ltd
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.
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.
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18
"""Implementation of Transport over SFTP, using paramiko."""
20
# TODO: Remove the transport-based lock_read and lock_write methods. They'll
21
# then raise TransportNotPossible, which will break remote access to any
22
# formats which rely on OS-level locks. That should be fine as those formats
23
# are pretty old, but these combinations may have to be removed from the test
24
# suite. Those formats all date back to 0.7; so we should be able to remove
25
# these methods when we officially drop support for those formats.
43
from bzrlib.errors import (FileExists,
44
NoSuchFile, PathNotChild,
50
from bzrlib.osutils import pathjoin, fancy_rename, getcwd
51
from bzrlib.trace import mutter, warning
52
from bzrlib.transport import (
53
register_urlparse_netloc_protocol,
59
from bzrlib.transport.local import LocalURLServer
63
except ImportError, e:
64
raise ParamikoNotPresent(e)
66
from paramiko.sftp import (SFTP_FLAG_WRITE, SFTP_FLAG_CREATE,
67
SFTP_FLAG_EXCL, SFTP_FLAG_TRUNC,
69
from paramiko.sftp_attr import SFTPAttributes
70
from paramiko.sftp_file import SFTPFile
73
register_urlparse_netloc_protocol('sftp')
76
# This is a weakref dictionary, so that we can reuse connections
77
# that are still active. Long term, it might be nice to have some
78
# sort of expiration policy, such as disconnect if inactive for
79
# X seconds. But that requires a lot more fanciness.
80
_connected_hosts = weakref.WeakValueDictionary()
83
_paramiko_version = getattr(paramiko, '__version_info__', (0, 0, 0))
84
# don't use prefetch unless paramiko version >= 1.5.5 (there were bugs earlier)
85
_default_do_prefetch = (_paramiko_version >= (1, 5, 5))
88
def clear_connection_cache():
89
"""Remove all hosts from the SFTP connection cache.
91
Primarily useful for test cases wanting to force garbage collection.
93
_connected_hosts.clear()
96
class SFTPLock(object):
97
"""This fakes a lock in a remote location.
99
A present lock is indicated just by the existence of a file. This
100
doesn't work well on all transports and they are only used in
101
deprecated storage formats.
104
__slots__ = ['path', 'lock_path', 'lock_file', 'transport']
106
def __init__(self, path, transport):
107
assert isinstance(transport, SFTPTransport)
109
self.lock_file = None
111
self.lock_path = path + '.write-lock'
112
self.transport = transport
114
# RBC 20060103 FIXME should we be using private methods here ?
115
abspath = transport._remote_path(self.lock_path)
116
self.lock_file = transport._sftp_open_exclusive(abspath)
118
raise LockError('File %r already locked' % (self.path,))
121
"""Should this warn, or actually try to cleanup?"""
123
warning("SFTPLock %r not explicitly unlocked" % (self.path,))
127
if not self.lock_file:
129
self.lock_file.close()
130
self.lock_file = None
132
self.transport.delete(self.lock_path)
133
except (NoSuchFile,):
134
# What specific errors should we catch here?
138
class SFTPUrlHandling(Transport):
139
"""Mix-in that does common handling of SSH/SFTP URLs."""
141
def __init__(self, base):
142
self._parse_url(base)
143
base = self._unparse_url(self._path)
146
super(SFTPUrlHandling, self).__init__(base)
148
def _parse_url(self, url):
150
self._username, self._password,
151
self._host, self._port, self._path) = self._split_url(url)
153
def _unparse_url(self, path):
154
"""Return a URL for a path relative to this transport.
156
path = urllib.quote(path)
157
# handle homedir paths
158
if not path.startswith('/'):
160
netloc = urllib.quote(self._host)
161
if self._username is not None:
162
netloc = '%s@%s' % (urllib.quote(self._username), netloc)
163
if self._port is not None:
164
netloc = '%s:%d' % (netloc, self._port)
165
return urlparse.urlunparse((self._scheme, netloc, path, '', '', ''))
167
def _split_url(self, url):
168
(scheme, username, password, host, port, path) = split_url(url)
169
## assert scheme == 'sftp'
171
# the initial slash should be removed from the path, and treated
172
# as a homedir relative path (the path begins with a double slash
173
# if it is absolute).
174
# see draft-ietf-secsh-scp-sftp-ssh-uri-03.txt
175
# RBC 20060118 we are not using this as its too user hostile. instead
176
# we are following lftp and using /~/foo to mean '~/foo'.
177
# handle homedir paths
178
if path.startswith('/~/'):
182
return (scheme, username, password, host, port, path)
184
def abspath(self, relpath):
185
"""Return the full url to the given relative path.
187
@param relpath: the relative path or path components
188
@type relpath: str or list
190
return self._unparse_url(self._remote_path(relpath))
192
def _remote_path(self, relpath):
193
"""Return the path to be passed along the sftp protocol for relpath.
195
:param relpath: is a urlencoded string.
197
return self._combine_paths(self._path, relpath)
200
class SFTPTransport(SFTPUrlHandling):
201
"""Transport implementation for SFTP access."""
203
_do_prefetch = _default_do_prefetch
204
# TODO: jam 20060717 Conceivably these could be configurable, either
205
# by auto-tuning at run-time, or by a configuration (per host??)
206
# but the performance curve is pretty flat, so just going with
207
# reasonable defaults.
208
_max_readv_combine = 200
209
# Having to round trip to the server means waiting for a response,
210
# so it is better to download extra bytes.
211
# 8KiB had good performance for both local and remote network operations
212
_bytes_to_read_before_seek = 8192
214
# The sftp spec says that implementations SHOULD allow reads
215
# to be at least 32K. paramiko.readv() does an async request
216
# for the chunks. So we need to keep it within a single request
217
# size for paramiko <= 1.6.1. paramiko 1.6.2 will probably chop
218
# up the request itself, rather than us having to worry about it
219
_max_request_size = 32768
221
def __init__(self, base, clone_from=None):
222
super(SFTPTransport, self).__init__(base)
223
if clone_from is None:
226
# use the same ssh connection, etc
227
self._sftp = clone_from._sftp
228
# super saves 'self.base'
230
def should_cache(self):
232
Return True if the data pulled across should be cached locally.
236
def clone(self, offset=None):
238
Return a new SFTPTransport with root at self.base + offset.
239
We share the same SFTP session between such transports, because it's
240
fairly expensive to set them up.
243
return SFTPTransport(self.base, self)
245
return SFTPTransport(self.abspath(offset), self)
247
def _remote_path(self, relpath):
248
"""Return the path to be passed along the sftp protocol for relpath.
250
relpath is a urlencoded string.
252
:return: a path prefixed with / for regular abspath-based urls, or a
253
path that does not begin with / for urls which begin with /~/.
255
# how does this work?
256
# it processes relpath with respect to
258
# firstly we create a path to evaluate:
259
# if relpath is an abspath or homedir path, its the entire thing
260
# otherwise we join our base with relpath
261
# then we eliminate all empty segments (double //'s) outside the first
262
# two elements of the list. This avoids problems with trailing
263
# slashes, or other abnormalities.
264
# finally we evaluate the entire path in a single pass
266
# '..' result in popping the left most already
267
# processed path (which can never be empty because of the check for
268
# abspath and homedir meaning that its not, or that we've used our
269
# path. If the pop would pop the root, we ignore it.
271
# Specific case examinations:
272
# remove the special casefor ~: if the current root is ~/ popping of it
273
# = / thus our seed for a ~ based path is ['', '~']
274
# and if we end up with [''] then we had basically ('', '..') (which is
275
# '/..' so we append '' if the length is one, and assert that the first
276
# element is still ''. Lastly, if we end with ['', '~'] as a prefix for
277
# the output, we've got a homedir path, so we strip that prefix before
278
# '/' joining the resulting list.
280
# case one: '/' -> ['', ''] cannot shrink
281
# case two: '/' + '../foo' -> ['', 'foo'] (take '', '', '..', 'foo')
282
# and pop the second '' for the '..', append 'foo'
283
# case three: '/~/' -> ['', '~', '']
284
# case four: '/~/' + '../foo' -> ['', '~', '', '..', 'foo'],
285
# and we want to get '/foo' - the empty path in the middle
286
# needs to be stripped, then normal path manipulation will
288
# case five: '/..' ['', '..'], we want ['', '']
289
# stripping '' outside the first two is ok
290
# ignore .. if its too high up
292
# lastly this code is possibly reusable by FTP, but not reusable by
293
# local paths: ~ is resolvable correctly, nor by HTTP or the smart
294
# server: ~ is resolved remotely.
296
# however, a version of this that acts on self.base is possible to be
297
# written which manipulates the URL in canonical form, and would be
298
# reusable for all transports, if a flag for allowing ~/ at all was
300
assert isinstance(relpath, basestring)
301
relpath = urlutils.unescape(relpath)
304
if relpath.startswith('/'):
305
# abspath - normal split is fine.
306
current_path = relpath.split('/')
307
elif relpath.startswith('~/'):
308
# root is homedir based: normal split and prefix '' to remote the
310
current_path = [''].extend(relpath.split('/'))
312
# root is from the current directory:
313
if self._path.startswith('/'):
314
# abspath, take the regular split
317
# homedir based, add the '', '~' not present in self._path
318
current_path = ['', '~']
319
# add our current dir
320
current_path.extend(self._path.split('/'))
321
# add the users relpath
322
current_path.extend(relpath.split('/'))
323
# strip '' segments that are not in the first one - the leading /.
324
to_process = current_path[:1]
325
for segment in current_path[1:]:
327
to_process.append(segment)
329
# process '.' and '..' segments into output_path.
331
for segment in to_process:
333
# directory pop. Remove a directory
334
# as long as we are not at the root
335
if len(output_path) > 1:
338
# cannot pop beyond the root, so do nothing
340
continue # strip the '.' from the output.
342
# this will append '' to output_path for the root elements,
343
# which is appropriate: its why we strip '' in the first pass.
344
output_path.append(segment)
346
# check output special cases:
347
if output_path == ['']:
349
output_path = ['', '']
350
elif output_path[:2] == ['', '~']:
351
# ['', '~', ...] -> ...
352
output_path = output_path[2:]
353
path = '/'.join(output_path)
356
def relpath(self, abspath):
357
scheme, username, password, host, port, path = self._split_url(abspath)
359
if (username != self._username):
360
error.append('username mismatch')
361
if (host != self._host):
362
error.append('host mismatch')
363
if (port != self._port):
364
error.append('port mismatch')
365
if (not path.startswith(self._path)):
366
error.append('path mismatch')
368
extra = ': ' + ', '.join(error)
369
raise PathNotChild(abspath, self.base, extra=extra)
371
return path[pl:].strip('/')
373
def has(self, relpath):
375
Does the target location exist?
378
self._sftp.stat(self._remote_path(relpath))
383
def get(self, relpath):
385
Get the file at the given relative path.
387
:param relpath: The relative path to the file
390
path = self._remote_path(relpath)
391
f = self._sftp.file(path, mode='rb')
392
if self._do_prefetch and (getattr(f, 'prefetch', None) is not None):
395
except (IOError, paramiko.SSHException), e:
396
self._translate_io_exception(e, path, ': error retrieving')
398
def readv(self, relpath, offsets):
399
"""See Transport.readv()"""
400
# We overload the default readv() because we want to use a file
401
# that does not have prefetch enabled.
402
# Also, if we have a new paramiko, it implements an async readv()
407
path = self._remote_path(relpath)
408
fp = self._sftp.file(path, mode='rb')
409
readv = getattr(fp, 'readv', None)
411
return self._sftp_readv(fp, offsets, relpath)
412
mutter('seek and read %s offsets', len(offsets))
413
return self._seek_and_read(fp, offsets, relpath)
414
except (IOError, paramiko.SSHException), e:
415
self._translate_io_exception(e, path, ': error retrieving')
417
def _sftp_readv(self, fp, offsets, relpath='<unknown>'):
418
"""Use the readv() member of fp to do async readv.
420
And then read them using paramiko.readv(). paramiko.readv()
421
does not support ranges > 64K, so it caps the request size, and
422
just reads until it gets all the stuff it wants
424
offsets = list(offsets)
425
sorted_offsets = sorted(offsets)
427
# The algorithm works as follows:
428
# 1) Coalesce nearby reads into a single chunk
429
# This generates a list of combined regions, the total size
430
# and the size of the sub regions. This coalescing step is limited
431
# in the number of nearby chunks to combine, and is allowed to
432
# skip small breaks in the requests. Limiting it makes sure that
433
# we can start yielding some data earlier, and skipping means we
434
# make fewer requests. (Beneficial even when using async)
435
# 2) Break up this combined regions into chunks that are smaller
436
# than 64KiB. Technically the limit is 65536, but we are a
437
# little bit conservative. This is because sftp has a maximum
438
# return chunk size of 64KiB (max size of an unsigned short)
439
# 3) Issue a readv() to paramiko to create an async request for
441
# 4) Read in the data as it comes back, until we've read one
442
# continuous section as determined in step 1
443
# 5) Break up the full sections into hunks for the original requested
444
# offsets. And put them in a cache
445
# 6) Check if the next request is in the cache, and if it is, remove
446
# it from the cache, and yield its data. Continue until no more
447
# entries are in the cache.
448
# 7) loop back to step 4 until all data has been read
450
# TODO: jam 20060725 This could be optimized one step further, by
451
# attempting to yield whatever data we have read, even before
452
# the first coallesced section has been fully processed.
454
# When coalescing for use with readv(), we don't really need to
455
# use any fudge factor, because the requests are made asynchronously
456
coalesced = list(self._coalesce_offsets(sorted_offsets,
457
limit=self._max_readv_combine,
461
for c_offset in coalesced:
462
start = c_offset.start
463
size = c_offset.length
465
# We need to break this up into multiple requests
467
next_size = min(size, self._max_request_size)
468
requests.append((start, next_size))
472
mutter('SFTP.readv() %s offsets => %s coalesced => %s requests',
473
len(offsets), len(coalesced), len(requests))
475
# Queue the current read until we have read the full coalesced section
478
cur_coalesced_stack = iter(coalesced)
479
cur_coalesced = cur_coalesced_stack.next()
481
# Cache the results, but only until they have been fulfilled
483
# turn the list of offsets into a stack
484
offset_stack = iter(offsets)
485
cur_offset_and_size = offset_stack.next()
487
for data in fp.readv(requests):
489
cur_data_len += len(data)
491
if cur_data_len < cur_coalesced.length:
493
assert cur_data_len == cur_coalesced.length, \
494
"Somehow we read too much: %s != %s" % (cur_data_len,
495
cur_coalesced.length)
496
all_data = ''.join(cur_data)
500
for suboffset, subsize in cur_coalesced.ranges:
501
key = (cur_coalesced.start+suboffset, subsize)
502
data_map[key] = all_data[suboffset:suboffset+subsize]
504
# Now that we've read some data, see if we can yield anything back
505
while cur_offset_and_size in data_map:
506
this_data = data_map.pop(cur_offset_and_size)
507
yield cur_offset_and_size[0], this_data
508
cur_offset_and_size = offset_stack.next()
510
# We read a coalesced entry, so mark it as done
512
# Now that we've read all of the data for this coalesced section
514
cur_coalesced = cur_coalesced_stack.next()
516
if cur_coalesced is not None:
517
raise errors.ShortReadvError(relpath, cur_coalesced.start,
518
cur_coalesced.length, len(data))
520
def put_file(self, relpath, f, mode=None):
522
Copy the file-like object into the location.
524
:param relpath: Location to put the contents, relative to base.
525
:param f: File-like object.
526
:param mode: The final mode for the file
528
final_path = self._remote_path(relpath)
529
self._put(final_path, f, mode=mode)
531
def _put(self, abspath, f, mode=None):
532
"""Helper function so both put() and copy_abspaths can reuse the code"""
533
tmp_abspath = '%s.tmp.%.9f.%d.%d' % (abspath, time.time(),
534
os.getpid(), random.randint(0,0x7FFFFFFF))
535
fout = self._sftp_open_exclusive(tmp_abspath, mode=mode)
539
fout.set_pipelined(True)
541
except (IOError, paramiko.SSHException), e:
542
self._translate_io_exception(e, tmp_abspath)
543
# XXX: This doesn't truly help like we would like it to.
544
# The problem is that openssh strips sticky bits. So while we
545
# can properly set group write permission, we lose the group
546
# sticky bit. So it is probably best to stop chmodding, and
547
# just tell users that they need to set the umask correctly.
548
# The attr.st_mode = mode, in _sftp_open_exclusive
549
# will handle when the user wants the final mode to be more
550
# restrictive. And then we avoid a round trip. Unless
551
# paramiko decides to expose an async chmod()
553
# This is designed to chmod() right before we close.
554
# Because we set_pipelined() earlier, theoretically we might
555
# avoid the round trip for fout.close()
557
self._sftp.chmod(tmp_abspath, mode)
560
self._rename_and_overwrite(tmp_abspath, abspath)
562
# If we fail, try to clean up the temporary file
563
# before we throw the exception
564
# but don't let another exception mess things up
565
# Write out the traceback, because otherwise
566
# the catch and throw destroys it
568
mutter(traceback.format_exc())
572
self._sftp.remove(tmp_abspath)
574
# raise the saved except
576
# raise the original with its traceback if we can.
579
def _put_non_atomic_helper(self, relpath, writer, mode=None,
580
create_parent_dir=False,
582
abspath = self._remote_path(relpath)
584
# TODO: jam 20060816 paramiko doesn't publicly expose a way to
585
# set the file mode at create time. If it does, use it.
586
# But for now, we just chmod later anyway.
588
def _open_and_write_file():
589
"""Try to open the target file, raise error on failure"""
593
fout = self._sftp.file(abspath, mode='wb')
594
fout.set_pipelined(True)
596
except (paramiko.SSHException, IOError), e:
597
self._translate_io_exception(e, abspath,
600
# This is designed to chmod() right before we close.
601
# Because we set_pipelined() earlier, theoretically we might
602
# avoid the round trip for fout.close()
604
self._sftp.chmod(abspath, mode)
609
if not create_parent_dir:
610
_open_and_write_file()
613
# Try error handling to create the parent directory if we need to
615
_open_and_write_file()
617
# Try to create the parent directory, and then go back to
619
parent_dir = os.path.dirname(abspath)
620
self._mkdir(parent_dir, dir_mode)
621
_open_and_write_file()
623
def put_file_non_atomic(self, relpath, f, mode=None,
624
create_parent_dir=False,
626
"""Copy the file-like object into the target location.
628
This function is not strictly safe to use. It is only meant to
629
be used when you already know that the target does not exist.
630
It is not safe, because it will open and truncate the remote
631
file. So there may be a time when the file has invalid contents.
633
:param relpath: The remote location to put the contents.
634
:param f: File-like object.
635
:param mode: Possible access permissions for new file.
636
None means do not set remote permissions.
637
:param create_parent_dir: If we cannot create the target file because
638
the parent directory does not exist, go ahead and
639
create it, and then try again.
643
self._put_non_atomic_helper(relpath, writer, mode=mode,
644
create_parent_dir=create_parent_dir,
647
def put_bytes_non_atomic(self, relpath, bytes, mode=None,
648
create_parent_dir=False,
652
self._put_non_atomic_helper(relpath, writer, mode=mode,
653
create_parent_dir=create_parent_dir,
656
def iter_files_recursive(self):
657
"""Walk the relative paths of all files in this transport."""
658
queue = list(self.list_dir('.'))
660
relpath = queue.pop(0)
661
st = self.stat(relpath)
662
if stat.S_ISDIR(st.st_mode):
663
for i, basename in enumerate(self.list_dir(relpath)):
664
queue.insert(i, relpath+'/'+basename)
668
def _mkdir(self, abspath, mode=None):
674
self._sftp.mkdir(abspath, local_mode)
676
self._sftp.chmod(abspath, mode=mode)
677
except (paramiko.SSHException, IOError), e:
678
self._translate_io_exception(e, abspath, ': unable to mkdir',
679
failure_exc=FileExists)
681
def mkdir(self, relpath, mode=None):
682
"""Create a directory at the given path."""
683
self._mkdir(self._remote_path(relpath), mode=mode)
685
def _translate_io_exception(self, e, path, more_info='',
686
failure_exc=PathError):
687
"""Translate a paramiko or IOError into a friendlier exception.
689
:param e: The original exception
690
:param path: The path in question when the error is raised
691
:param more_info: Extra information that can be included,
692
such as what was going on
693
:param failure_exc: Paramiko has the super fun ability to raise completely
694
opaque errors that just set "e.args = ('Failure',)" with
696
If this parameter is set, it defines the exception
697
to raise in these cases.
699
# paramiko seems to generate detailless errors.
700
self._translate_error(e, path, raise_generic=False)
701
if getattr(e, 'args', None) is not None:
702
if (e.args == ('No such file or directory',) or
703
e.args == ('No such file',)):
704
raise NoSuchFile(path, str(e) + more_info)
705
if (e.args == ('mkdir failed',)):
706
raise FileExists(path, str(e) + more_info)
707
# strange but true, for the paramiko server.
708
if (e.args == ('Failure',)):
709
raise failure_exc(path, str(e) + more_info)
710
mutter('Raising exception with args %s', e.args)
711
if getattr(e, 'errno', None) is not None:
712
mutter('Raising exception with errno %s', e.errno)
715
def append_file(self, relpath, f, mode=None):
717
Append the text in the file-like object into the final
721
path = self._remote_path(relpath)
722
fout = self._sftp.file(path, 'ab')
724
self._sftp.chmod(path, mode)
728
except (IOError, paramiko.SSHException), e:
729
self._translate_io_exception(e, relpath, ': unable to append')
731
def rename(self, rel_from, rel_to):
732
"""Rename without special overwriting"""
734
self._sftp.rename(self._remote_path(rel_from),
735
self._remote_path(rel_to))
736
except (IOError, paramiko.SSHException), e:
737
self._translate_io_exception(e, rel_from,
738
': unable to rename to %r' % (rel_to))
740
def _rename_and_overwrite(self, abs_from, abs_to):
741
"""Do a fancy rename on the remote server.
743
Using the implementation provided by osutils.
746
fancy_rename(abs_from, abs_to,
747
rename_func=self._sftp.rename,
748
unlink_func=self._sftp.remove)
749
except (IOError, paramiko.SSHException), e:
750
self._translate_io_exception(e, abs_from, ': unable to rename to %r' % (abs_to))
752
def move(self, rel_from, rel_to):
753
"""Move the item at rel_from to the location at rel_to"""
754
path_from = self._remote_path(rel_from)
755
path_to = self._remote_path(rel_to)
756
self._rename_and_overwrite(path_from, path_to)
758
def delete(self, relpath):
759
"""Delete the item at relpath"""
760
path = self._remote_path(relpath)
762
self._sftp.remove(path)
763
except (IOError, paramiko.SSHException), e:
764
self._translate_io_exception(e, path, ': unable to delete')
767
"""Return True if this store supports listing."""
770
def list_dir(self, relpath):
772
Return a list of all files at the given location.
774
# does anything actually use this?
776
# This is at least used by copy_tree for remote upgrades.
777
# -- David Allouche 2006-08-11
778
path = self._remote_path(relpath)
780
entries = self._sftp.listdir(path)
781
except (IOError, paramiko.SSHException), e:
782
self._translate_io_exception(e, path, ': failed to list_dir')
783
return [urlutils.escape(entry) for entry in entries]
785
def rmdir(self, relpath):
786
"""See Transport.rmdir."""
787
path = self._remote_path(relpath)
789
return self._sftp.rmdir(path)
790
except (IOError, paramiko.SSHException), e:
791
self._translate_io_exception(e, path, ': failed to rmdir')
793
def stat(self, relpath):
794
"""Return the stat information for a file."""
795
path = self._remote_path(relpath)
797
return self._sftp.stat(path)
798
except (IOError, paramiko.SSHException), e:
799
self._translate_io_exception(e, path, ': unable to stat')
801
def lock_read(self, relpath):
803
Lock the given file for shared (read) access.
804
:return: A lock object, which has an unlock() member function
806
# FIXME: there should be something clever i can do here...
807
class BogusLock(object):
808
def __init__(self, path):
812
return BogusLock(relpath)
814
def lock_write(self, relpath):
816
Lock the given file for exclusive (write) access.
817
WARNING: many transports do not support this, so trying avoid using it
819
:return: A lock object, which has an unlock() member function
821
# This is a little bit bogus, but basically, we create a file
822
# which should not already exist, and if it does, we assume
823
# that there is a lock, and if it doesn't, the we assume
824
# that we have taken the lock.
825
return SFTPLock(relpath, self)
827
def _sftp_connect(self):
828
"""Connect to the remote sftp server.
829
After this, self._sftp should have a valid connection (or
830
we raise an TransportError 'could not connect').
832
TODO: Raise a more reasonable ConnectionFailed exception
834
self._sftp = _sftp_connect(self._host, self._port, self._username,
837
def _sftp_open_exclusive(self, abspath, mode=None):
838
"""Open a remote path exclusively.
840
SFTP supports O_EXCL (SFTP_FLAG_EXCL), which fails if
841
the file already exists. However it does not expose this
842
at the higher level of SFTPClient.open(), so we have to
845
WARNING: This breaks the SFTPClient abstraction, so it
846
could easily break against an updated version of paramiko.
848
:param abspath: The remote absolute path where the file should be opened
849
:param mode: The mode permissions bits for the new file
851
# TODO: jam 20060816 Paramiko >= 1.6.2 (probably earlier) supports
852
# using the 'x' flag to indicate SFTP_FLAG_EXCL.
853
# However, there is no way to set the permission mode at open
854
# time using the sftp_client.file() functionality.
855
path = self._sftp._adjust_cwd(abspath)
856
# mutter('sftp abspath %s => %s', abspath, path)
857
attr = SFTPAttributes()
860
omode = (SFTP_FLAG_WRITE | SFTP_FLAG_CREATE
861
| SFTP_FLAG_TRUNC | SFTP_FLAG_EXCL)
863
t, msg = self._sftp._request(CMD_OPEN, path, omode, attr)
865
raise TransportError('Expected an SFTP handle')
866
handle = msg.get_string()
867
return SFTPFile(self._sftp, handle, 'wb', -1)
868
except (paramiko.SSHException, IOError), e:
869
self._translate_io_exception(e, abspath, ': unable to open',
870
failure_exc=FileExists)
872
def _can_roundtrip_unix_modebits(self):
873
if sys.platform == 'win32':
879
# ------------- server test implementation --------------
882
from bzrlib.tests.stub_sftp import StubServer, StubSFTPServer
884
STUB_SERVER_KEY = """
885
-----BEGIN RSA PRIVATE KEY-----
886
MIICWgIBAAKBgQDTj1bqB4WmayWNPB+8jVSYpZYk80Ujvj680pOTh2bORBjbIAyz
887
oWGW+GUjzKxTiiPvVmxFgx5wdsFvF03v34lEVVhMpouqPAYQ15N37K/ir5XY+9m/
888
d8ufMCkjeXsQkKqFbAlQcnWMCRnOoPHS3I4vi6hmnDDeeYTSRvfLbW0fhwIBIwKB
889
gBIiOqZYaoqbeD9OS9z2K9KR2atlTxGxOJPXiP4ESqP3NVScWNwyZ3NXHpyrJLa0
890
EbVtzsQhLn6rF+TzXnOlcipFvjsem3iYzCpuChfGQ6SovTcOjHV9z+hnpXvQ/fon
891
soVRZY65wKnF7IAoUwTmJS9opqgrN6kRgCd3DASAMd1bAkEA96SBVWFt/fJBNJ9H
892
tYnBKZGw0VeHOYmVYbvMSstssn8un+pQpUm9vlG/bp7Oxd/m+b9KWEh2xPfv6zqU
893
avNwHwJBANqzGZa/EpzF4J8pGti7oIAPUIDGMtfIcmqNXVMckrmzQ2vTfqtkEZsA
894
4rE1IERRyiJQx6EJsz21wJmGV9WJQ5kCQQDwkS0uXqVdFzgHO6S++tjmjYcxwr3g
895
H0CoFYSgbddOT6miqRskOQF3DZVkJT3kyuBgU2zKygz52ukQZMqxCb1fAkASvuTv
896
qfpH87Qq5kQhNKdbbwbmd2NxlNabazPijWuphGTdW0VfJdWfklyS2Kr+iqrs/5wV
897
HhathJt636Eg7oIjAkA8ht3MQ+XSl9yIJIS8gVpbPxSw5OMfw0PjVE7tBdQruiSc
898
nvuQES5C9BMHjF39LZiGH1iLQy7FgdHyoP+eodI7
899
-----END RSA PRIVATE KEY-----
903
class SocketListener(threading.Thread):
905
def __init__(self, callback):
906
threading.Thread.__init__(self)
907
self._callback = callback
908
self._socket = socket.socket()
909
self._socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
910
self._socket.bind(('localhost', 0))
911
self._socket.listen(1)
912
self.port = self._socket.getsockname()[1]
913
self._stop_event = threading.Event()
916
# called from outside this thread
917
self._stop_event.set()
918
# use a timeout here, because if the test fails, the server thread may
919
# never notice the stop_event.
925
readable, writable_unused, exception_unused = \
926
select.select([self._socket], [], [], 0.1)
927
if self._stop_event.isSet():
929
if len(readable) == 0:
932
s, addr_unused = self._socket.accept()
933
# because the loopback socket is inline, and transports are
934
# never explicitly closed, best to launch a new thread.
935
threading.Thread(target=self._callback, args=(s,)).start()
936
except socket.error, x:
937
sys.excepthook(*sys.exc_info())
938
warning('Socket error during accept() within unit test server'
941
# probably a failed test; unit test thread will log the
943
sys.excepthook(*sys.exc_info())
944
warning('Exception from within unit test server thread: %r' %
948
class SocketDelay(object):
949
"""A socket decorator to make TCP appear slower.
951
This changes recv, send, and sendall to add a fixed latency to each python
952
call if a new roundtrip is detected. That is, when a recv is called and the
953
flag new_roundtrip is set, latency is charged. Every send and send_all
956
In addition every send, sendall and recv sleeps a bit per character send to
959
Not all methods are implemented, this is deliberate as this class is not a
960
replacement for the builtin sockets layer. fileno is not implemented to
961
prevent the proxy being bypassed.
965
_proxied_arguments = dict.fromkeys([
966
"close", "getpeername", "getsockname", "getsockopt", "gettimeout",
967
"setblocking", "setsockopt", "settimeout", "shutdown"])
969
def __init__(self, sock, latency, bandwidth=1.0,
972
:param bandwith: simulated bandwith (MegaBit)
973
:param really_sleep: If set to false, the SocketDelay will just
974
increase a counter, instead of calling time.sleep. This is useful for
975
unittesting the SocketDelay.
978
self.latency = latency
979
self.really_sleep = really_sleep
980
self.time_per_byte = 1 / (bandwidth / 8.0 * 1024 * 1024)
981
self.new_roundtrip = False
984
if self.really_sleep:
987
SocketDelay.simulated_time += s
989
def __getattr__(self, attr):
990
if attr in SocketDelay._proxied_arguments:
991
return getattr(self.sock, attr)
992
raise AttributeError("'SocketDelay' object has no attribute %r" %
996
return SocketDelay(self.sock.dup(), self.latency, self.time_per_byte,
999
def recv(self, *args):
1000
data = self.sock.recv(*args)
1001
if data and self.new_roundtrip:
1002
self.new_roundtrip = False
1003
self.sleep(self.latency)
1004
self.sleep(len(data) * self.time_per_byte)
1007
def sendall(self, data, flags=0):
1008
if not self.new_roundtrip:
1009
self.new_roundtrip = True
1010
self.sleep(self.latency)
1011
self.sleep(len(data) * self.time_per_byte)
1012
return self.sock.sendall(data, flags)
1014
def send(self, data, flags=0):
1015
if not self.new_roundtrip:
1016
self.new_roundtrip = True
1017
self.sleep(self.latency)
1018
bytes_sent = self.sock.send(data, flags)
1019
self.sleep(bytes_sent * self.time_per_byte)
1023
class SFTPServer(Server):
1024
"""Common code for SFTP server facilities."""
1026
def __init__(self, server_interface=StubServer):
1027
self._original_vendor = None
1028
self._homedir = None
1029
self._server_homedir = None
1030
self._listener = None
1032
self._vendor = ssh.ParamikoVendor()
1033
self._server_interface = server_interface
1036
self.add_latency = 0
1038
def _get_sftp_url(self, path):
1039
"""Calculate an sftp url to this server for path."""
1040
return 'sftp://foo:bar@localhost:%d/%s' % (self._listener.port, path)
1042
def log(self, message):
1043
"""StubServer uses this to log when a new server is created."""
1044
self.logs.append(message)
1046
def _run_server_entry(self, sock):
1047
"""Entry point for all implementations of _run_server.
1049
If self.add_latency is > 0.000001 then sock is given a latency adding
1052
if self.add_latency > 0.000001:
1053
sock = SocketDelay(sock, self.add_latency)
1054
return self._run_server(sock)
1056
def _run_server(self, s):
1057
ssh_server = paramiko.Transport(s)
1058
key_file = pathjoin(self._homedir, 'test_rsa.key')
1059
f = open(key_file, 'w')
1060
f.write(STUB_SERVER_KEY)
1062
host_key = paramiko.RSAKey.from_private_key_file(key_file)
1063
ssh_server.add_server_key(host_key)
1064
server = self._server_interface(self)
1065
ssh_server.set_subsystem_handler('sftp', paramiko.SFTPServer,
1066
StubSFTPServer, root=self._root,
1067
home=self._server_homedir)
1068
event = threading.Event()
1069
ssh_server.start_server(event, server)
1072
def setUp(self, vfs_server=None):
1073
# XXX: TODO: make sftpserver back onto vfs_server rather than local disk.
1074
assert vfs_server is None or isinstance(vfs_server, LocalURLServer), \
1075
"SFTPServer currently assumes local transport, got %s" % vfs_server
1076
self._original_vendor = ssh._ssh_vendor_manager._cached_ssh_vendor
1077
ssh._ssh_vendor_manager._cached_ssh_vendor = self._vendor
1078
if sys.platform == 'win32':
1079
# Win32 needs to use the UNICODE api
1080
self._homedir = getcwd()
1082
# But Linux SFTP servers should just deal in bytestreams
1083
self._homedir = os.getcwd()
1084
if self._server_homedir is None:
1085
self._server_homedir = self._homedir
1087
if sys.platform == 'win32':
1089
self._listener = SocketListener(self._run_server_entry)
1090
self._listener.setDaemon(True)
1091
self._listener.start()
1094
"""See bzrlib.transport.Server.tearDown."""
1095
self._listener.stop()
1096
ssh._ssh_vendor_manager._cached_ssh_vendor = self._original_vendor
1098
def get_bogus_url(self):
1099
"""See bzrlib.transport.Server.get_bogus_url."""
1100
# this is chosen to try to prevent trouble with proxies, wierd dns, etc
1101
# we bind a random socket, so that we get a guaranteed unused port
1102
# we just never listen on that port
1104
s.bind(('localhost', 0))
1105
return 'sftp://%s:%s/' % s.getsockname()
1108
class SFTPFullAbsoluteServer(SFTPServer):
1109
"""A test server for sftp transports, using absolute urls and ssh."""
1112
"""See bzrlib.transport.Server.get_url."""
1113
homedir = self._homedir
1114
if sys.platform != 'win32':
1115
# Remove the initial '/' on all platforms but win32
1116
homedir = homedir[1:]
1117
return self._get_sftp_url(urlutils.escape(homedir))
1120
class SFTPServerWithoutSSH(SFTPServer):
1121
"""An SFTP server that uses a simple TCP socket pair rather than SSH."""
1124
super(SFTPServerWithoutSSH, self).__init__()
1125
self._vendor = ssh.LoopbackVendor()
1127
def _run_server(self, sock):
1128
# Re-import these as locals, so that they're still accessible during
1129
# interpreter shutdown (when all module globals get set to None, leading
1130
# to confusing errors like "'NoneType' object has no attribute 'error'".
1131
class FakeChannel(object):
1132
def get_transport(self):
1134
def get_log_channel(self):
1138
def get_hexdump(self):
1143
server = paramiko.SFTPServer(FakeChannel(), 'sftp', StubServer(self), StubSFTPServer,
1144
root=self._root, home=self._server_homedir)
1146
server.start_subsystem('sftp', None, sock)
1147
except socket.error, e:
1148
if (len(e.args) > 0) and (e.args[0] == errno.EPIPE):
1149
# it's okay for the client to disconnect abruptly
1150
# (bug in paramiko 1.6: it should absorb this exception)
1154
except Exception, e:
1155
import sys; sys.stderr.write('\nEXCEPTION %r\n\n' % e.__class__)
1156
server.finish_subsystem()
1159
class SFTPAbsoluteServer(SFTPServerWithoutSSH):
1160
"""A test server for sftp transports, using absolute urls."""
1163
"""See bzrlib.transport.Server.get_url."""
1164
homedir = self._homedir
1165
if sys.platform != 'win32':
1166
# Remove the initial '/' on all platforms but win32
1167
homedir = homedir[1:]
1168
return self._get_sftp_url(urlutils.escape(homedir))
1171
class SFTPHomeDirServer(SFTPServerWithoutSSH):
1172
"""A test server for sftp transports, using homedir relative urls."""
1175
"""See bzrlib.transport.Server.get_url."""
1176
return self._get_sftp_url("~/")
1179
class SFTPSiblingAbsoluteServer(SFTPAbsoluteServer):
1180
"""A test servere for sftp transports, using absolute urls to non-home."""
1183
self._server_homedir = '/dev/noone/runs/tests/here'
1184
super(SFTPSiblingAbsoluteServer, self).setUp()
1187
def _sftp_connect(host, port, username, password):
1188
"""Connect to the remote sftp server.
1190
:raises: a TransportError 'could not connect'.
1192
:returns: an paramiko.sftp_client.SFTPClient
1194
TODO: Raise a more reasonable ConnectionFailed exception
1196
idx = (host, port, username)
1198
return _connected_hosts[idx]
1202
sftp = _sftp_connect_uncached(host, port, username, password)
1203
_connected_hosts[idx] = sftp
1206
def _sftp_connect_uncached(host, port, username, password):
1207
vendor = ssh._get_ssh_vendor()
1208
sftp = vendor.connect_sftp(username, password, host, port)
1212
def get_test_permutations():
1213
"""Return the permutations to be used in testing."""
1214
return [(SFTPTransport, SFTPAbsoluteServer),
1215
(SFTPTransport, SFTPHomeDirServer),
1216
(SFTPTransport, SFTPSiblingAbsoluteServer),