42
44
from bzrlib.errors import (FileExists,
45
NoSuchFile, PathNotChild,
47
49
ParamikoNotPresent,
49
from bzrlib.osutils import fancy_rename
51
from bzrlib.osutils import pathjoin, fancy_rename, getcwd
52
from bzrlib.symbol_versioning import (
50
55
from bzrlib.trace import mutter, warning
51
56
from bzrlib.transport import (
55
62
ConnectedTransport,
76
83
from paramiko.sftp import (SFTP_FLAG_WRITE, SFTP_FLAG_CREATE,
77
84
SFTP_FLAG_EXCL, SFTP_FLAG_TRUNC,
78
SFTP_OK, CMD_HANDLE, CMD_OPEN)
79
86
from paramiko.sftp_attr import SFTPAttributes
80
87
from paramiko.sftp_file import SFTPFile
107
114
except FileExists:
108
115
raise LockError('File %r already locked' % (self.path,))
118
"""Should this warn, or actually try to cleanup?"""
120
warning("SFTPLock %r not explicitly unlocked" % (self.path,))
110
123
def unlock(self):
111
124
if not self.lock_file:
268
281
buffered = buffered[buffered_offset:]
269
282
buffered_data = [buffered]
270
283
buffered_len = len(buffered)
271
# now that the data stream is done, close the handle
274
285
buffered = ''.join(buffered_data)
275
286
del buffered_data[:]
330
341
# up the request itself, rather than us having to worry about it
331
342
_max_request_size = 32768
344
def __init__(self, base, _from_transport=None):
345
super(SFTPTransport, self).__init__(base,
346
_from_transport=_from_transport)
333
348
def _remote_path(self, relpath):
334
349
"""Return the path to be passed along the sftp protocol for relpath.
336
351
:param relpath: is a urlencoded string.
338
remote_path = self._parsed_url.clone(relpath).path
353
relative = urlutils.unescape(relpath).encode('utf-8')
354
remote_path = self._combine_paths(self._path, relative)
339
355
# the initial slash should be removed from the path, and treated as a
340
356
# homedir relative path (the path begins with a double slash if it is
341
357
# absolute). see draft-ietf-secsh-scp-sftp-ssh-uri-03.txt
360
376
in base url at transport creation time.
362
378
if credentials is None:
363
password = self._parsed_url.password
379
password = self._password
365
381
password = credentials
367
383
vendor = ssh._get_ssh_vendor()
368
user = self._parsed_url.user
370
386
auth = config.AuthenticationConfig()
371
user = auth.get_user('ssh', self._parsed_url.host,
372
self._parsed_url.port)
373
connection = vendor.connect_sftp(self._parsed_url.user, password,
374
self._parsed_url.host, self._parsed_url.port)
387
user = auth.get_user('ssh', self._host, self._port)
388
connection = vendor.connect_sftp(self._user, password,
389
self._host, self._port)
375
390
return connection, (user, password)
377
def disconnect(self):
378
connection = self._get_connection()
379
if connection is not None:
382
392
def _get_sftp(self):
383
393
"""Ensures that a connection is established"""
384
394
connection = self._get_connection()
406
416
:param relpath: The relative path to the file
419
# FIXME: by returning the file directly, we don't pass this
420
# through to report_activity. We could try wrapping the object
421
# before it's returned. For readv and get_bytes it's handled in
422
# the higher-level function.
409
424
path = self._remote_path(relpath)
410
425
f = self._get_sftp().file(path, mode='rb')
411
426
if self._do_prefetch and (getattr(f, 'prefetch', None) is not None):
700
715
if (e.args[0].startswith('Directory not empty: ')
701
716
or getattr(e, 'errno', None) == errno.ENOTEMPTY):
702
717
raise errors.DirectoryNotEmpty(path, str(e))
703
if e.args == ('Operation unsupported',):
704
raise errors.TransportNotPossible()
705
718
mutter('Raising exception with args %s', e.args)
706
719
if getattr(e, 'errno', None) is not None:
707
720
mutter('Raising exception with errno %s', e.errno)
797
810
"""Return the stat information for a file."""
798
811
path = self._remote_path(relpath)
800
return self._get_sftp().lstat(path)
813
return self._get_sftp().stat(path)
801
814
except (IOError, paramiko.SSHException), e:
802
815
self._translate_io_exception(e, path, ': unable to stat')
804
def readlink(self, relpath):
805
"""See Transport.readlink."""
806
path = self._remote_path(relpath)
808
return self._get_sftp().readlink(path)
809
except (IOError, paramiko.SSHException), e:
810
self._translate_io_exception(e, path, ': unable to readlink')
812
def symlink(self, source, link_name):
813
"""See Transport.symlink."""
815
conn = self._get_sftp()
816
sftp_retval = conn.symlink(source, link_name)
817
if SFTP_OK != sftp_retval:
818
raise TransportError(
819
'%r: unable to create symlink to %r' % (link_name, source),
822
except (IOError, paramiko.SSHException), e:
823
self._translate_io_exception(e, link_name,
824
': unable to create symlink to %r' % (source))
826
817
def lock_read(self, relpath):
828
819
Lock the given file for shared (read) access.