13
13
# You should have received a copy of the GNU General Public License
14
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
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
17
"""RemoteTransport client for the smart-server.
60
63
RemoteTCPTransport, etc.
66
# When making a readv request, cap it at requesting 5MB of data
67
_max_readv_bytes = 5*1024*1024
63
69
# IMPORTANT FOR IMPLEMENTORS: RemoteTransport MUST NOT be given encoding
64
70
# responsibilities: Put those on SmartClient or similar. This is vital for
65
71
# the ability to support multiple versions of the smart protocol over time:
66
# RemoteTransport is an adapter from the Transport object model to the
72
# RemoteTransport is an adapter from the Transport object model to the
67
73
# SmartClient model, not an encoder.
69
75
# FIXME: the medium parameter should be private, only the tests requires
86
92
should only be used for testing purposes; normally this is
87
93
determined from the medium.
89
super(RemoteTransport, self).__init__(url,
90
_from_transport=_from_transport)
95
super(RemoteTransport, self).__init__(
96
url, _from_transport=_from_transport)
92
98
# The medium is the connection, except when we need to share it with
93
99
# other objects (RemoteBzrDir, RemoteRepository etc). In these cases
94
100
# what we want to share is really the shared connection.
96
if _from_transport is None:
102
if (_from_transport is not None
103
and isinstance(_from_transport, RemoteTransport)):
104
_client = _from_transport._client
105
elif _from_transport is None:
97
106
# If no _from_transport is specified, we need to intialize the
99
108
credentials = None
152
168
def get_smart_medium(self):
153
169
return self._get_connection()
155
@deprecated_method(one_four)
156
def get_shared_medium(self):
157
return self._get_shared_connection()
159
171
def _remote_path(self, relpath):
160
172
"""Returns the Unicode version of the absolute path for relpath."""
161
173
return self._combine_paths(self._path, relpath)
163
175
def _call(self, method, *args):
165
resp = self._call2(method, *args)
166
except errors.ErrorFromSmartServer, err:
167
self._translate_error(err.error_tuple)
168
self._translate_error(resp)
176
resp = self._call2(method, *args)
177
self._ensure_ok(resp)
170
179
def _call2(self, method, *args):
171
180
"""Call a method on the remote server."""
173
182
return self._client.call(method, *args)
174
183
except errors.ErrorFromSmartServer, err:
175
self._translate_error(err.error_tuple)
184
# The first argument, if present, is always a path.
186
context = {'relpath': args[0]}
189
self._translate_error(err, **context)
177
191
def _call_with_body_bytes(self, method, args, body):
178
192
"""Call a method on the remote server with body bytes."""
180
194
return self._client.call_with_body_bytes(method, args, body)
181
195
except errors.ErrorFromSmartServer, err:
182
self._translate_error(err.error_tuple)
196
# The first argument, if present, is always a path.
198
context = {'relpath': args[0]}
201
self._translate_error(err, **context)
184
203
def has(self, relpath):
185
204
"""Indicate whether a remote file of the given name exists or not.
314
332
sorted_offsets = sorted(offsets)
315
333
coalesced = list(self._coalesce_offsets(sorted_offsets,
316
334
limit=self._max_readv_combine,
317
fudge_factor=self._bytes_to_read_before_seek))
320
result = self._client.call_with_body_readv_array(
321
('readv', self._remote_path(relpath),),
322
[(c.start, c.length) for c in coalesced])
323
resp, response_handler = result
324
except errors.ErrorFromSmartServer, err:
325
self._translate_error(err.error_tuple)
327
if resp[0] != 'readv':
328
# This should raise an exception
329
response_handler.cancel_read_body()
330
raise errors.UnexpectedSmartServerResponse(resp)
332
return self._handle_response(offsets, coalesced, response_handler)
334
def _handle_response(self, offsets, coalesced, response_handler):
335
# turn the list of offsets into a stack
335
fudge_factor=self._bytes_to_read_before_seek,
336
max_size=self._max_readv_bytes))
338
# now that we've coallesced things, avoid making enormous requests
343
if c.length + cur_len > self._max_readv_bytes:
344
requests.append(cur_request)
348
cur_request.append(c)
351
requests.append(cur_request)
352
if 'hpss' in debug.debug_flags:
353
trace.mutter('%s.readv %s offsets => %s coalesced'
354
' => %s requests (%s)',
355
self.__class__.__name__, len(offsets), len(coalesced),
356
len(requests), sum(map(len, requests)))
357
# Cache the results, but only until they have been fulfilled
359
# turn the list of offsets into a single stack to iterate
336
360
offset_stack = iter(offsets)
337
cur_offset_and_size = offset_stack.next()
361
# using a list so it can be modified when passing down and coming back
362
next_offset = [offset_stack.next()]
363
for cur_request in requests:
365
result = self._client.call_with_body_readv_array(
366
('readv', self._remote_path(relpath),),
367
[(c.start, c.length) for c in cur_request])
368
resp, response_handler = result
369
except errors.ErrorFromSmartServer, err:
370
self._translate_error(err, relpath)
372
if resp[0] != 'readv':
373
# This should raise an exception
374
response_handler.cancel_read_body()
375
raise errors.UnexpectedSmartServerResponse(resp)
377
for res in self._handle_response(offset_stack, cur_request,
383
def _handle_response(self, offset_stack, coalesced, response_handler,
384
data_map, next_offset):
385
cur_offset_and_size = next_offset[0]
338
386
# FIXME: this should know how many bytes are needed, for clarity.
339
387
data = response_handler.read_body_bytes()
340
# Cache the results, but only until they have been fulfilled
343
389
for c_offset in coalesced:
344
390
if len(data) < c_offset.length:
381
427
def rmdir(self, relpath):
382
428
resp = self._call('rmdir', self._remote_path(relpath))
384
def _translate_error(self, resp, orig_path=None):
385
"""Raise an exception from a response"""
392
elif what == 'NoSuchFile':
393
if orig_path is not None:
394
error_path = orig_path
397
raise errors.NoSuchFile(error_path)
398
elif what == 'error':
399
raise errors.SmartProtocolError(unicode(resp[1]))
400
elif what == 'FileExists':
401
raise errors.FileExists(resp[1])
402
elif what == 'DirectoryNotEmpty':
403
raise errors.DirectoryNotEmpty(resp[1])
404
elif what == 'ShortReadvError':
405
raise errors.ShortReadvError(resp[1], int(resp[2]),
406
int(resp[3]), int(resp[4]))
407
elif what in ('UnicodeEncodeError', 'UnicodeDecodeError'):
408
encoding = str(resp[1]) # encoding must always be a string
412
reason = str(resp[5]) # reason must always be a string
413
if val.startswith('u:'):
414
val = val[2:].decode('utf-8')
415
elif val.startswith('s:'):
416
val = val[2:].decode('base64')
417
if what == 'UnicodeDecodeError':
418
raise UnicodeDecodeError(encoding, val, start, end, reason)
419
elif what == 'UnicodeEncodeError':
420
raise UnicodeEncodeError(encoding, val, start, end, reason)
421
elif what == "ReadOnlyError":
422
raise errors.TransportNotPossible('readonly transport')
423
elif what == "ReadError":
424
if orig_path is not None:
425
error_path = orig_path
428
raise errors.ReadError(error_path)
429
elif what == "PermissionDenied":
430
if orig_path is not None:
431
error_path = orig_path
434
raise errors.PermissionDenied(error_path)
436
raise errors.SmartProtocolError('unexpected smart server error: %r' % (resp,))
430
def _ensure_ok(self, resp):
432
raise errors.UnexpectedSmartServerResponse(resp)
434
def _translate_error(self, err, relpath=None):
435
remote._translate_error(err, path=relpath)
438
437
def disconnect(self):
439
438
self.get_smart_medium().disconnect()
465
463
resp = self._call2('list_dir', self._remote_path(relpath))
466
464
if resp[0] == 'names':
467
465
return [name.encode('ascii') for name in resp[1:]]
469
self._translate_error(resp)
466
raise errors.UnexpectedSmartServerResponse(resp)
471
468
def iter_files_recursive(self):
472
469
resp = self._call2('iter_files_recursive', self._remote_path(''))
473
470
if resp[0] == 'names':
476
self._translate_error(resp)
472
raise errors.UnexpectedSmartServerResponse(resp)
479
475
class RemoteTCPTransport(RemoteTransport):
480
476
"""Connection to smart server over plain tcp.
482
478
This is essentially just a factory to get 'RemoteTransport(url,
483
479
SmartTCPClientMedium).
514
510
def _build_medium(self):
515
# ssh will prompt the user for a password if needed and if none is
516
# provided but it will not give it back, so no credentials can be
518
511
location_config = config.LocationConfig(self.base)
519
512
bzr_remote_path = location_config.get_bzr_remote_path()
515
auth = config.AuthenticationConfig()
516
user = auth.get_user('ssh', self._host, self._port)
520
517
client_medium = medium.SmartSSHClientMedium(self._host, self._port,
521
self._user, self._password, self.base,
518
user, self._password, self.base,
522
519
bzr_remote_path=bzr_remote_path)
523
return client_medium, None
520
return client_medium, (user, self._password)
526
523
class RemoteHTTPTransport(RemoteTransport):
527
524
"""Just a way to connect between a bzr+http:// url and http://.
529
526
This connection operates slightly differently than the RemoteSSHTransport.
530
527
It uses a plain http:// transport underneath, which defines what remote
531
528
.bzr/smart URL we are connected to. From there, all paths that are sent are
580
577
_from_transport=self,
581
578
http_transport=self._http_transport)
580
def _redirected_to(self, source, target):
581
"""See transport._redirected_to"""
582
redirected = self._http_transport._redirected_to(source, target)
583
if (redirected is not None
584
and isinstance(redirected, type(self._http_transport))):
585
return RemoteHTTPTransport('bzr+' + redirected.external_url(),
586
http_transport=redirected)
588
# Either None or a transport for a different protocol
592
class HintingSSHTransport(transport.Transport):
593
"""Simple transport that handles ssh:// and points out bzr+ssh://."""
595
def __init__(self, url):
596
raise errors.UnsupportedProtocol(url,
597
'bzr supports bzr+ssh to operate over ssh, use "bzr+%s".' % url)
584
600
def get_test_permutations():
585
601
"""Return (transport, server) permutations for testing."""