~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/remote.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2007-12-20 16:16:34 UTC
  • mfrom: (3123.5.18 hardlinks)
  • Revision ID: pqm@pqm.ubuntu.com-20071220161634-2kcjb650o21ydko4
Accelerate build_tree using similar workingtrees (abentley)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2006, 2007, 2008 Canonical Ltd
 
1
# Copyright (C) 2006, 2007 Canonical Ltd
2
2
#
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
17
17
# TODO: At some point, handle upgrades by just passing the whole request
18
18
# across to run on the server.
19
19
 
20
 
import bz2
21
20
from cStringIO import StringIO
22
21
 
23
22
from bzrlib import (
24
23
    branch,
25
 
    debug,
26
24
    errors,
27
 
    graph,
28
25
    lockdir,
29
26
    repository,
30
27
    revision,
31
 
    symbol_versioning,
32
28
)
33
 
from bzrlib.branch import BranchReferenceFormat
 
29
from bzrlib.branch import Branch, BranchReferenceFormat
34
30
from bzrlib.bzrdir import BzrDir, RemoteBzrDirFormat
35
31
from bzrlib.config import BranchConfig, TreeConfig
36
32
from bzrlib.decorators import needs_read_lock, needs_write_lock
37
 
from bzrlib.errors import (
38
 
    NoSuchRevision,
39
 
    SmartProtocolError,
40
 
    )
 
33
from bzrlib.errors import NoSuchRevision
41
34
from bzrlib.lockable_files import LockableFiles
42
 
from bzrlib.pack import ContainerPushParser
 
35
from bzrlib.pack import ContainerReader
43
36
from bzrlib.smart import client, vfs
44
37
from bzrlib.symbol_versioning import (
45
 
    deprecated_in,
46
38
    deprecated_method,
 
39
    zero_ninetyone,
47
40
    )
48
 
from bzrlib.revision import ensure_null, NULL_REVISION
49
 
from bzrlib.trace import mutter, note, warning
50
 
 
 
41
from bzrlib.trace import note
51
42
 
52
43
# Note: RemoteBzrDirFormat is in bzrdir.py
53
44
 
66
57
        self._real_bzrdir = None
67
58
 
68
59
        if _client is None:
69
 
            medium = transport.get_smart_medium()
70
 
            self._client = client._SmartClient(medium)
 
60
            self._shared_medium = transport.get_shared_medium()
 
61
            self._client = client._SmartClient(self._shared_medium)
71
62
        else:
72
63
            self._client = _client
 
64
            self._shared_medium = None
73
65
            return
74
66
 
75
67
        path = self._path_for_remote_call(self._client)
88
80
            self._real_bzrdir = BzrDir.open_from_transport(
89
81
                self.root_transport, _server_formats=False)
90
82
 
91
 
    def cloning_metadir(self):
92
 
        self._ensure_real()
93
 
        return self._real_bzrdir.cloning_metadir()
94
 
 
95
 
    def _translate_error(self, err, **context):
96
 
        _translate_error(err, bzrdir=self, **context)
97
 
        
98
83
    def create_repository(self, shared=False):
99
84
        self._ensure_real()
100
85
        self._real_bzrdir.create_repository(shared=shared)
129
114
    def get_branch_reference(self):
130
115
        """See BzrDir.get_branch_reference()."""
131
116
        path = self._path_for_remote_call(self._client)
132
 
        try:
133
 
            response = self._client.call('BzrDir.open_branch', path)
134
 
        except errors.ErrorFromSmartServer, err:
135
 
            self._translate_error(err)
 
117
        response = self._client.call('BzrDir.open_branch', path)
136
118
        if response[0] == 'ok':
137
119
            if response[1] == '':
138
120
                # branch at this location.
140
122
            else:
141
123
                # a branch reference, use the existing BranchReference logic.
142
124
                return response[1]
 
125
        elif response == ('nobranch',):
 
126
            raise errors.NotBranchError(path=self.root_transport.base)
143
127
        else:
144
128
            raise errors.UnexpectedSmartServerResponse(response)
145
129
 
146
 
    def _get_tree_branch(self):
147
 
        """See BzrDir._get_tree_branch()."""
148
 
        return None, self.open_branch()
149
 
 
150
130
    def open_branch(self, _unsupported=False):
151
 
        if _unsupported:
152
 
            raise NotImplementedError('unsupported flag support not implemented yet.')
 
131
        assert _unsupported == False, 'unsupported flag support not implemented yet.'
153
132
        reference_url = self.get_branch_reference()
154
133
        if reference_url is None:
155
134
            # branch at this location.
161
140
                
162
141
    def open_repository(self):
163
142
        path = self._path_for_remote_call(self._client)
164
 
        verb = 'BzrDir.find_repositoryV2'
165
 
        try:
166
 
            try:
167
 
                response = self._client.call(verb, path)
168
 
            except errors.UnknownSmartMethod:
169
 
                verb = 'BzrDir.find_repository'
170
 
                response = self._client.call(verb, path)
171
 
        except errors.ErrorFromSmartServer, err:
172
 
            self._translate_error(err)
173
 
        if response[0] != 'ok':
174
 
            raise errors.UnexpectedSmartServerResponse(response)
175
 
        if verb == 'BzrDir.find_repository':
176
 
            # servers that don't support the V2 method don't support external
177
 
            # references either.
178
 
            response = response + ('no', )
179
 
        if not (len(response) == 5):
180
 
            raise SmartProtocolError('incorrect response length %s' % (response,))
 
143
        response = self._client.call('BzrDir.find_repository', path)
 
144
        assert response[0] in ('ok', 'norepository'), \
 
145
            'unexpected response code %s' % (response,)
 
146
        if response[0] == 'norepository':
 
147
            raise errors.NoRepositoryPresent(self)
 
148
        assert len(response) == 4, 'incorrect response length %s' % (response,)
181
149
        if response[1] == '':
182
150
            format = RemoteRepositoryFormat()
183
151
            format.rich_root_data = (response[2] == 'yes')
184
152
            format.supports_tree_reference = (response[3] == 'yes')
185
 
            # No wire format to check this yet.
186
 
            format.supports_external_lookups = (response[4] == 'yes')
187
 
            # Used to support creating a real format instance when needed.
188
 
            format._creating_bzrdir = self
189
153
            return RemoteRepository(self, format)
190
154
        else:
191
155
            raise errors.NoRepositoryPresent(self)
221
185
        """Upgrading of remote bzrdirs is not supported yet."""
222
186
        return False
223
187
 
224
 
    def clone(self, url, revision_id=None, force_new_repo=False,
225
 
              preserve_stacking=False):
 
188
    def clone(self, url, revision_id=None, force_new_repo=False):
226
189
        self._ensure_real()
227
190
        return self._real_bzrdir.clone(url, revision_id=revision_id,
228
 
            force_new_repo=force_new_repo, preserve_stacking=preserve_stacking)
229
 
 
230
 
    def get_config(self):
231
 
        self._ensure_real()
232
 
        return self._real_bzrdir.get_config()
 
191
            force_new_repo=force_new_repo)
233
192
 
234
193
 
235
194
class RemoteRepositoryFormat(repository.RepositoryFormat):
245
204
    the class level.
246
205
    """
247
206
 
248
 
    _matchingbzrdir = RemoteBzrDirFormat()
 
207
    _matchingbzrdir = RemoteBzrDirFormat
249
208
 
250
209
    def initialize(self, a_bzrdir, shared=False):
251
 
        if not isinstance(a_bzrdir, RemoteBzrDir):
252
 
            prior_repo = self._creating_bzrdir.open_repository()
253
 
            prior_repo._ensure_real()
254
 
            return prior_repo._real_repository._format.initialize(
255
 
                a_bzrdir, shared=shared)
 
210
        assert isinstance(a_bzrdir, RemoteBzrDir), \
 
211
            '%r is not a RemoteBzrDir' % (a_bzrdir,)
256
212
        return a_bzrdir.create_repository(shared=shared)
257
213
    
258
214
    def open(self, a_bzrdir):
259
 
        if not isinstance(a_bzrdir, RemoteBzrDir):
260
 
            raise AssertionError('%r is not a RemoteBzrDir' % (a_bzrdir,))
 
215
        assert isinstance(a_bzrdir, RemoteBzrDir)
261
216
        return a_bzrdir.open_repository()
262
217
 
263
218
    def get_format_description(self):
300
255
            self._real_repository = None
301
256
        self.bzrdir = remote_bzrdir
302
257
        if _client is None:
303
 
            self._client = remote_bzrdir._client
 
258
            self._client = client._SmartClient(self.bzrdir._shared_medium)
304
259
        else:
305
260
            self._client = _client
306
261
        self._format = format
308
263
        self._lock_token = None
309
264
        self._lock_count = 0
310
265
        self._leave_lock = False
311
 
        # A cache of looked up revision parent data; reset at unlock time.
312
 
        self._parents_map = None
313
 
        if 'hpss' in debug.debug_flags:
314
 
            self._requested_parents = None
315
266
        # For tests:
316
267
        # These depend on the actual remote format, so force them off for
317
268
        # maximum compatibility. XXX: In future these should depend on the
321
272
        self._reconcile_fixes_text_parents = False
322
273
        self._reconcile_backsup_inventory = False
323
274
        self.base = self.bzrdir.transport.base
324
 
        # Additional places to query for data.
325
 
        self._fallback_repositories = []
326
275
 
327
276
    def __str__(self):
328
277
        return "%s(%s)" % (self.__class__.__name__, self.base)
361
310
            #self._real_repository = self.bzrdir._real_bzrdir.open_repository()
362
311
            self._set_real_repository(self.bzrdir._real_bzrdir.open_repository())
363
312
 
364
 
    def _translate_error(self, err, **context):
365
 
        self.bzrdir._translate_error(err, repository=self, **context)
366
 
 
367
313
    def find_text_key_references(self):
368
314
        """Find the text key references within the repository.
369
315
 
389
335
        self._ensure_real()
390
336
        return self._real_repository._generate_text_key_index()
391
337
 
392
 
    @symbol_versioning.deprecated_method(symbol_versioning.one_four)
393
338
    def get_revision_graph(self, revision_id=None):
394
339
        """See Repository.get_revision_graph()."""
395
 
        return self._get_revision_graph(revision_id)
396
 
 
397
 
    def _get_revision_graph(self, revision_id):
398
 
        """Private method for using with old (< 1.2) servers to fallback."""
399
340
        if revision_id is None:
400
341
            revision_id = ''
401
342
        elif revision.is_null(revision_id):
402
343
            return {}
403
344
 
404
345
        path = self.bzrdir._path_for_remote_call(self._client)
405
 
        try:
406
 
            response = self._client.call_expecting_body(
407
 
                'Repository.get_revision_graph', path, revision_id)
408
 
        except errors.ErrorFromSmartServer, err:
409
 
            self._translate_error(err)
410
 
        response_tuple, response_handler = response
411
 
        if response_tuple[0] != 'ok':
412
 
            raise errors.UnexpectedSmartServerResponse(response_tuple)
413
 
        coded = response_handler.read_body_bytes()
414
 
        if coded == '':
415
 
            # no revisions in this repository!
416
 
            return {}
417
 
        lines = coded.split('\n')
418
 
        revision_graph = {}
419
 
        for line in lines:
420
 
            d = tuple(line.split())
421
 
            revision_graph[d[0]] = d[1:]
422
 
            
423
 
        return revision_graph
 
346
        assert type(revision_id) is str
 
347
        response = self._client.call_expecting_body(
 
348
            'Repository.get_revision_graph', path, revision_id)
 
349
        if response[0][0] not in ['ok', 'nosuchrevision']:
 
350
            raise errors.UnexpectedSmartServerResponse(response[0])
 
351
        if response[0][0] == 'ok':
 
352
            coded = response[1].read_body_bytes()
 
353
            if coded == '':
 
354
                # no revisions in this repository!
 
355
                return {}
 
356
            lines = coded.split('\n')
 
357
            revision_graph = {}
 
358
            for line in lines:
 
359
                d = tuple(line.split())
 
360
                revision_graph[d[0]] = d[1:]
 
361
                
 
362
            return revision_graph
 
363
        else:
 
364
            response_body = response[1].read_body_bytes()
 
365
            assert response_body == ''
 
366
            raise NoSuchRevision(self, revision_id)
424
367
 
425
368
    def has_revision(self, revision_id):
426
369
        """See Repository.has_revision()."""
427
 
        if revision_id == NULL_REVISION:
 
370
        if revision_id is None:
428
371
            # The null revision is always present.
429
372
            return True
430
373
        path = self.bzrdir._path_for_remote_call(self._client)
431
 
        response = self._client.call(
432
 
            'Repository.has_revision', path, revision_id)
433
 
        if response[0] not in ('yes', 'no'):
434
 
            raise errors.UnexpectedSmartServerResponse(response)
 
374
        response = self._client.call('Repository.has_revision', path, revision_id)
 
375
        assert response[0] in ('yes', 'no'), 'unexpected response code %s' % (response,)
435
376
        return response[0] == 'yes'
436
377
 
437
 
    def has_revisions(self, revision_ids):
438
 
        """See Repository.has_revisions()."""
439
 
        result = set()
440
 
        for revision_id in revision_ids:
441
 
            if self.has_revision(revision_id):
442
 
                result.add(revision_id)
443
 
        return result
444
 
 
445
378
    def has_same_location(self, other):
446
379
        return (self.__class__ == other.__class__ and
447
380
                self.bzrdir.transport.base == other.bzrdir.transport.base)
448
381
        
449
382
    def get_graph(self, other_repository=None):
450
383
        """Return the graph for this repository format"""
451
 
        parents_provider = self
452
 
        if (other_repository is not None and
453
 
            other_repository.bzrdir.transport.base !=
454
 
            self.bzrdir.transport.base):
455
 
            parents_provider = graph._StackedParentsProvider(
456
 
                [parents_provider, other_repository._make_parents_provider()])
457
 
        return graph.Graph(parents_provider)
 
384
        self._ensure_real()
 
385
        return self._real_repository.get_graph(other_repository)
458
386
 
459
387
    def gather_stats(self, revid=None, committers=None):
460
388
        """See Repository.gather_stats()."""
468
396
            fmt_committers = 'no'
469
397
        else:
470
398
            fmt_committers = 'yes'
471
 
        response_tuple, response_handler = self._client.call_expecting_body(
 
399
        response = self._client.call_expecting_body(
472
400
            'Repository.gather_stats', path, fmt_revid, fmt_committers)
473
 
        if response_tuple[0] != 'ok':
474
 
            raise errors.UnexpectedSmartServerResponse(response_tuple)
 
401
        assert response[0][0] == 'ok', \
 
402
            'unexpected response code %s' % (response[0],)
475
403
 
476
 
        body = response_handler.read_body_bytes()
 
404
        body = response[1].read_body_bytes()
477
405
        result = {}
478
406
        for line in body.split('\n'):
479
407
            if not line:
487
415
 
488
416
        return result
489
417
 
490
 
    def find_branches(self, using=False):
491
 
        """See Repository.find_branches()."""
492
 
        # should be an API call to the server.
493
 
        self._ensure_real()
494
 
        return self._real_repository.find_branches(using=using)
495
 
 
496
418
    def get_physical_lock_status(self):
497
419
        """See Repository.get_physical_lock_status()."""
498
420
        # should be an API call to the server.
514
436
        """See Repository.is_shared()."""
515
437
        path = self.bzrdir._path_for_remote_call(self._client)
516
438
        response = self._client.call('Repository.is_shared', path)
517
 
        if response[0] not in ('yes', 'no'):
518
 
            raise SmartProtocolError('unexpected response code %s' % (response,))
 
439
        assert response[0] in ('yes', 'no'), 'unexpected response code %s' % (response,)
519
440
        return response[0] == 'yes'
520
441
 
521
442
    def is_write_locked(self):
526
447
        if not self._lock_mode:
527
448
            self._lock_mode = 'r'
528
449
            self._lock_count = 1
529
 
            self._parents_map = {}
530
 
            if 'hpss' in debug.debug_flags:
531
 
                self._requested_parents = set()
532
450
            if self._real_repository is not None:
533
451
                self._real_repository.lock_read()
534
452
        else:
538
456
        path = self.bzrdir._path_for_remote_call(self._client)
539
457
        if token is None:
540
458
            token = ''
541
 
        try:
542
 
            response = self._client.call('Repository.lock_write', path, token)
543
 
        except errors.ErrorFromSmartServer, err:
544
 
            self._translate_error(err, token=token)
545
 
 
 
459
        response = self._client.call('Repository.lock_write', path, token)
546
460
        if response[0] == 'ok':
547
461
            ok, token = response
548
462
            return token
 
463
        elif response[0] == 'LockContention':
 
464
            raise errors.LockContention('(remote lock)')
 
465
        elif response[0] == 'UnlockableTransport':
 
466
            raise errors.UnlockableTransport(self.bzrdir.root_transport)
 
467
        elif response[0] == 'LockFailed':
 
468
            raise errors.LockFailed(response[1], response[2])
549
469
        else:
550
470
            raise errors.UnexpectedSmartServerResponse(response)
551
471
 
564
484
                self._leave_lock = False
565
485
            self._lock_mode = 'w'
566
486
            self._lock_count = 1
567
 
            self._parents_map = {}
568
 
            if 'hpss' in debug.debug_flags:
569
 
                self._requested_parents = set()
570
487
        elif self._lock_mode == 'r':
571
488
            raise errors.ReadOnlyError(self)
572
489
        else:
589
506
        :param repository: The repository to fallback to for non-hpss
590
507
            implemented operations.
591
508
        """
592
 
        if isinstance(repository, RemoteRepository):
593
 
            raise AssertionError()
 
509
        assert not isinstance(repository, RemoteRepository)
594
510
        self._real_repository = repository
595
511
        if self._lock_mode == 'w':
596
512
            # if we are already locked, the real repository must be able to
615
531
        if not token:
616
532
            # with no token the remote repository is not persistently locked.
617
533
            return
618
 
        try:
619
 
            response = self._client.call('Repository.unlock', path, token)
620
 
        except errors.ErrorFromSmartServer, err:
621
 
            self._translate_error(err, token=token)
 
534
        response = self._client.call('Repository.unlock', path, token)
622
535
        if response == ('ok',):
623
536
            return
 
537
        elif response[0] == 'TokenMismatch':
 
538
            raise errors.TokenMismatch(token, '(remote token)')
624
539
        else:
625
540
            raise errors.UnexpectedSmartServerResponse(response)
626
541
 
628
543
        self._lock_count -= 1
629
544
        if self._lock_count > 0:
630
545
            return
631
 
        self._parents_map = None
632
 
        if 'hpss' in debug.debug_flags:
633
 
            self._requested_parents = None
634
546
        old_mode = self._lock_mode
635
547
        self._lock_mode = None
636
548
        try:
664
576
        """
665
577
        import tempfile
666
578
        path = self.bzrdir._path_for_remote_call(self._client)
667
 
        try:
668
 
            response, protocol = self._client.call_expecting_body(
669
 
                'Repository.tarball', path, compression)
670
 
        except errors.UnknownSmartMethod:
671
 
            protocol.cancel_read_body()
672
 
            return None
 
579
        response, protocol = self._client.call_expecting_body(
 
580
            'Repository.tarball', path, compression)
673
581
        if response[0] == 'ok':
674
582
            # Extract the tarball and return it
675
583
            t = tempfile.NamedTemporaryFile()
677
585
            t.write(protocol.read_body_bytes())
678
586
            t.seek(0)
679
587
            return t
 
588
        if (response == ('error', "Generic bzr smart protocol error: "
 
589
                "bad request 'Repository.tarball'") or
 
590
              response == ('error', "Generic bzr smart protocol error: "
 
591
                "bad request u'Repository.tarball'")):
 
592
            protocol.cancel_read_body()
 
593
            return None
680
594
        raise errors.UnexpectedSmartServerResponse(response)
681
595
 
682
596
    def sprout(self, to_bzrdir, revision_id=None):
708
622
                committer=committer, revprops=revprops, revision_id=revision_id)
709
623
        return builder
710
624
 
711
 
    def add_fallback_repository(self, repository):
712
 
        """Add a repository to use for looking up data not held locally.
713
 
        
714
 
        :param repository: A repository.
715
 
        """
716
 
        if not self._format.supports_external_lookups:
717
 
            raise errors.UnstackableRepositoryFormat(self._format, self.base)
718
 
        # We need to accumulate additional repositories here, to pass them in
719
 
        # on various RPC's.
720
 
        self._fallback_repositories.append(repository)
721
 
 
 
625
    @needs_write_lock
722
626
    def add_inventory(self, revid, inv, parents):
723
627
        self._ensure_real()
724
628
        return self._real_repository.add_inventory(revid, inv, parents)
725
629
 
 
630
    @needs_write_lock
726
631
    def add_revision(self, rev_id, rev, inv=None, config=None):
727
632
        self._ensure_real()
728
633
        return self._real_repository.add_revision(
733
638
        self._ensure_real()
734
639
        return self._real_repository.get_inventory(revision_id)
735
640
 
736
 
    def iter_inventories(self, revision_ids):
737
 
        self._ensure_real()
738
 
        return self._real_repository.iter_inventories(revision_ids)
739
 
 
740
641
    @needs_read_lock
741
642
    def get_revision(self, revision_id):
742
643
        self._ensure_real()
743
644
        return self._real_repository.get_revision(revision_id)
744
645
 
 
646
    @property
 
647
    def weave_store(self):
 
648
        self._ensure_real()
 
649
        return self._real_repository.weave_store
 
650
 
745
651
    def get_transaction(self):
746
652
        self._ensure_real()
747
653
        return self._real_repository.get_transaction()
752
658
        return self._real_repository.clone(a_bzrdir, revision_id=revision_id)
753
659
 
754
660
    def make_working_trees(self):
755
 
        """See Repository.make_working_trees"""
756
 
        self._ensure_real()
757
 
        return self._real_repository.make_working_trees()
758
 
 
759
 
    def revision_ids_to_search_result(self, result_set):
760
 
        """Convert a set of revision ids to a graph SearchResult."""
761
 
        result_parents = set()
762
 
        for parents in self.get_graph().get_parent_map(
763
 
            result_set).itervalues():
764
 
            result_parents.update(parents)
765
 
        included_keys = result_set.intersection(result_parents)
766
 
        start_keys = result_set.difference(included_keys)
767
 
        exclude_keys = result_parents.difference(result_set)
768
 
        result = graph.SearchResult(start_keys, exclude_keys,
769
 
            len(result_set), result_set)
770
 
        return result
771
 
 
772
 
    @needs_read_lock
773
 
    def search_missing_revision_ids(self, other, revision_id=None, find_ghosts=True):
774
 
        """Return the revision ids that other has that this does not.
775
 
        
776
 
        These are returned in topological order.
777
 
 
778
 
        revision_id: only return revision ids included by revision_id.
779
 
        """
780
 
        return repository.InterRepository.get(
781
 
            other, self).search_missing_revision_ids(revision_id, find_ghosts)
 
661
        """RemoteRepositories never create working trees by default."""
 
662
        return False
782
663
 
783
664
    def fetch(self, source, revision_id=None, pb=None):
784
665
        if self.has_same_location(source):
796
677
        self._ensure_real()
797
678
        self._real_repository.create_bundle(target, base, fileobj, format)
798
679
 
 
680
    @property
 
681
    def control_weaves(self):
 
682
        self._ensure_real()
 
683
        return self._real_repository.control_weaves
 
684
 
799
685
    @needs_read_lock
800
686
    def get_ancestry(self, revision_id, topo_sorted=True):
801
687
        self._ensure_real()
802
688
        return self._real_repository.get_ancestry(revision_id, topo_sorted)
803
689
 
 
690
    @needs_read_lock
 
691
    def get_inventory_weave(self):
 
692
        self._ensure_real()
 
693
        return self._real_repository.get_inventory_weave()
 
694
 
804
695
    def fileids_altered_by_revision_ids(self, revision_ids):
805
696
        self._ensure_real()
806
697
        return self._real_repository.fileids_altered_by_revision_ids(revision_ids)
816
707
        self._ensure_real()
817
708
        return self._real_repository.iter_files_bytes(desired_files)
818
709
 
819
 
    @property
820
 
    def _fetch_order(self):
821
 
        """Decorate the real repository for now.
822
 
 
823
 
        In the long term getting this back from the remote repository as part
824
 
        of open would be more efficient.
825
 
        """
826
 
        self._ensure_real()
827
 
        return self._real_repository._fetch_order
828
 
 
829
 
    @property
830
 
    def _fetch_uses_deltas(self):
831
 
        """Decorate the real repository for now.
832
 
 
833
 
        In the long term getting this back from the remote repository as part
834
 
        of open would be more efficient.
835
 
        """
836
 
        self._ensure_real()
837
 
        return self._real_repository._fetch_uses_deltas
838
 
 
839
 
    @property
840
 
    def _fetch_reconcile(self):
841
 
        """Decorate the real repository for now.
842
 
 
843
 
        In the long term getting this back from the remote repository as part
844
 
        of open would be more efficient.
845
 
        """
846
 
        self._ensure_real()
847
 
        return self._real_repository._fetch_reconcile
848
 
 
849
 
    def get_parent_map(self, keys):
850
 
        """See bzrlib.Graph.get_parent_map()."""
851
 
        # Hack to build up the caching logic.
852
 
        ancestry = self._parents_map
853
 
        if ancestry is None:
854
 
            # Repository is not locked, so there's no cache.
855
 
            missing_revisions = set(keys)
856
 
            ancestry = {}
857
 
        else:
858
 
            missing_revisions = set(key for key in keys if key not in ancestry)
859
 
        if missing_revisions:
860
 
            parent_map = self._get_parent_map(missing_revisions)
861
 
            if 'hpss' in debug.debug_flags:
862
 
                mutter('retransmitted revisions: %d of %d',
863
 
                        len(set(ancestry).intersection(parent_map)),
864
 
                        len(parent_map))
865
 
            ancestry.update(parent_map)
866
 
        present_keys = [k for k in keys if k in ancestry]
867
 
        if 'hpss' in debug.debug_flags:
868
 
            if self._requested_parents is not None and len(ancestry) != 0:
869
 
                self._requested_parents.update(present_keys)
870
 
                mutter('Current RemoteRepository graph hit rate: %d%%',
871
 
                    100.0 * len(self._requested_parents) / len(ancestry))
872
 
        return dict((k, ancestry[k]) for k in present_keys)
873
 
 
874
 
    def _get_parent_map(self, keys):
875
 
        """Helper for get_parent_map that performs the RPC."""
876
 
        medium = self._client._medium
877
 
        if medium._is_remote_before((1, 2)):
878
 
            # We already found out that the server can't understand
879
 
            # Repository.get_parent_map requests, so just fetch the whole
880
 
            # graph.
881
 
            # XXX: Note that this will issue a deprecation warning. This is ok
882
 
            # :- its because we're working with a deprecated server anyway, and
883
 
            # the user will almost certainly have seen a warning about the
884
 
            # server version already.
885
 
            rg = self.get_revision_graph()
886
 
            # There is an api discrepency between get_parent_map and
887
 
            # get_revision_graph. Specifically, a "key:()" pair in
888
 
            # get_revision_graph just means a node has no parents. For
889
 
            # "get_parent_map" it means the node is a ghost. So fix up the
890
 
            # graph to correct this.
891
 
            #   https://bugs.launchpad.net/bzr/+bug/214894
892
 
            # There is one other "bug" which is that ghosts in
893
 
            # get_revision_graph() are not returned at all. But we won't worry
894
 
            # about that for now.
895
 
            for node_id, parent_ids in rg.iteritems():
896
 
                if parent_ids == ():
897
 
                    rg[node_id] = (NULL_REVISION,)
898
 
            rg[NULL_REVISION] = ()
899
 
            return rg
900
 
 
901
 
        keys = set(keys)
902
 
        if None in keys:
903
 
            raise ValueError('get_parent_map(None) is not valid')
904
 
        if NULL_REVISION in keys:
905
 
            keys.discard(NULL_REVISION)
906
 
            found_parents = {NULL_REVISION:()}
907
 
            if not keys:
908
 
                return found_parents
909
 
        else:
910
 
            found_parents = {}
911
 
        # TODO(Needs analysis): We could assume that the keys being requested
912
 
        # from get_parent_map are in a breadth first search, so typically they
913
 
        # will all be depth N from some common parent, and we don't have to
914
 
        # have the server iterate from the root parent, but rather from the
915
 
        # keys we're searching; and just tell the server the keyspace we
916
 
        # already have; but this may be more traffic again.
917
 
 
918
 
        # Transform self._parents_map into a search request recipe.
919
 
        # TODO: Manage this incrementally to avoid covering the same path
920
 
        # repeatedly. (The server will have to on each request, but the less
921
 
        # work done the better).
922
 
        parents_map = self._parents_map
923
 
        if parents_map is None:
924
 
            # Repository is not locked, so there's no cache.
925
 
            parents_map = {}
926
 
        start_set = set(parents_map)
927
 
        result_parents = set()
928
 
        for parents in parents_map.itervalues():
929
 
            result_parents.update(parents)
930
 
        stop_keys = result_parents.difference(start_set)
931
 
        included_keys = start_set.intersection(result_parents)
932
 
        start_set.difference_update(included_keys)
933
 
        recipe = (start_set, stop_keys, len(parents_map))
934
 
        body = self._serialise_search_recipe(recipe)
935
 
        path = self.bzrdir._path_for_remote_call(self._client)
936
 
        for key in keys:
937
 
            if type(key) is not str:
938
 
                raise ValueError(
939
 
                    "key %r not a plain string" % (key,))
940
 
        verb = 'Repository.get_parent_map'
941
 
        args = (path,) + tuple(keys)
942
 
        try:
943
 
            response = self._client.call_with_body_bytes_expecting_body(
944
 
                verb, args, self._serialise_search_recipe(recipe))
945
 
        except errors.UnknownSmartMethod:
946
 
            # Server does not support this method, so get the whole graph.
947
 
            # Worse, we have to force a disconnection, because the server now
948
 
            # doesn't realise it has a body on the wire to consume, so the
949
 
            # only way to recover is to abandon the connection.
950
 
            warning(
951
 
                'Server is too old for fast get_parent_map, reconnecting.  '
952
 
                '(Upgrade the server to Bazaar 1.2 to avoid this)')
953
 
            medium.disconnect()
954
 
            # To avoid having to disconnect repeatedly, we keep track of the
955
 
            # fact the server doesn't understand remote methods added in 1.2.
956
 
            medium._remember_remote_is_before((1, 2))
957
 
            return self.get_revision_graph(None)
958
 
        response_tuple, response_handler = response
959
 
        if response_tuple[0] not in ['ok']:
960
 
            response_handler.cancel_read_body()
961
 
            raise errors.UnexpectedSmartServerResponse(response_tuple)
962
 
        if response_tuple[0] == 'ok':
963
 
            coded = bz2.decompress(response_handler.read_body_bytes())
964
 
            if coded == '':
965
 
                # no revisions found
966
 
                return {}
967
 
            lines = coded.split('\n')
968
 
            revision_graph = {}
969
 
            for line in lines:
970
 
                d = tuple(line.split())
971
 
                if len(d) > 1:
972
 
                    revision_graph[d[0]] = d[1:]
973
 
                else:
974
 
                    # No parents - so give the Graph result (NULL_REVISION,).
975
 
                    revision_graph[d[0]] = (NULL_REVISION,)
976
 
            return revision_graph
977
 
 
978
710
    @needs_read_lock
979
711
    def get_signature_text(self, revision_id):
980
712
        self._ensure_real()
981
713
        return self._real_repository.get_signature_text(revision_id)
982
714
 
983
715
    @needs_read_lock
984
 
    @symbol_versioning.deprecated_method(symbol_versioning.one_three)
985
716
    def get_revision_graph_with_ghosts(self, revision_ids=None):
986
717
        self._ensure_real()
987
718
        return self._real_repository.get_revision_graph_with_ghosts(
1040
771
        from bzrlib import osutils
1041
772
        import tarfile
1042
773
        import tempfile
 
774
        from StringIO import StringIO
1043
775
        # TODO: Maybe a progress bar while streaming the tarball?
1044
776
        note("Copying repository content as tarball...")
1045
777
        tar_file = self._get_tarball('bz2')
1063
795
        # TODO: Suggestion from john: using external tar is much faster than
1064
796
        # python's tarfile library, but it may not work on windows.
1065
797
 
1066
 
    @property
1067
 
    def inventories(self):
1068
 
        """Decorate the real repository for now.
1069
 
 
1070
 
        In the long term a full blown network facility is needed to
1071
 
        avoid creating a real repository object locally.
1072
 
        """
1073
 
        self._ensure_real()
1074
 
        return self._real_repository.inventories
1075
 
 
1076
798
    @needs_write_lock
1077
799
    def pack(self):
1078
800
        """Compress the data within the repository.
1082
804
        self._ensure_real()
1083
805
        return self._real_repository.pack()
1084
806
 
1085
 
    @property
1086
 
    def revisions(self):
1087
 
        """Decorate the real repository for now.
1088
 
 
1089
 
        In the short term this should become a real object to intercept graph
1090
 
        lookups.
1091
 
 
1092
 
        In the long term a full blown network facility is needed.
1093
 
        """
1094
 
        self._ensure_real()
1095
 
        return self._real_repository.revisions
1096
 
 
1097
807
    def set_make_working_trees(self, new_value):
1098
 
        self._ensure_real()
1099
 
        self._real_repository.set_make_working_trees(new_value)
1100
 
 
1101
 
    @property
1102
 
    def signatures(self):
1103
 
        """Decorate the real repository for now.
1104
 
 
1105
 
        In the long term a full blown network facility is needed to avoid
1106
 
        creating a real repository object locally.
1107
 
        """
1108
 
        self._ensure_real()
1109
 
        return self._real_repository.signatures
 
808
        raise NotImplementedError(self.set_make_working_trees)
1110
809
 
1111
810
    @needs_write_lock
1112
811
    def sign_revision(self, revision_id, gpg_strategy):
1113
812
        self._ensure_real()
1114
813
        return self._real_repository.sign_revision(revision_id, gpg_strategy)
1115
814
 
1116
 
    @property
1117
 
    def texts(self):
1118
 
        """Decorate the real repository for now.
1119
 
 
1120
 
        In the long term a full blown network facility is needed to avoid
1121
 
        creating a real repository object locally.
1122
 
        """
1123
 
        self._ensure_real()
1124
 
        return self._real_repository.texts
1125
 
 
1126
815
    @needs_read_lock
1127
816
    def get_revisions(self, revision_ids):
1128
817
        self._ensure_real()
1154
843
        self._ensure_real()
1155
844
        return self._real_repository.has_signature_for_revision_id(revision_id)
1156
845
 
 
846
    def get_data_stream(self, revision_ids):
 
847
        path = self.bzrdir._path_for_remote_call(self._client)
 
848
        response, protocol = self._client.call_expecting_body(
 
849
            'Repository.stream_knit_data_for_revisions', path, *revision_ids)
 
850
        if response == ('ok',):
 
851
            return self._deserialise_stream(protocol)
 
852
        elif (response == ('error', "Generic bzr smart protocol error: "
 
853
                "bad request 'Repository.stream_knit_data_for_revisions'") or
 
854
              response == ('error', "Generic bzr smart protocol error: "
 
855
                "bad request u'Repository.stream_knit_data_for_revisions'")):
 
856
            protocol.cancel_read_body()
 
857
            self._ensure_real()
 
858
            return self._real_repository.get_data_stream(revision_ids)
 
859
        else:
 
860
            raise errors.UnexpectedSmartServerResponse(response)
 
861
 
 
862
    def _deserialise_stream(self, protocol):
 
863
        buffer = StringIO(protocol.read_body_bytes())
 
864
        reader = ContainerReader(buffer)
 
865
        for record_names, read_bytes in reader.iter_records():
 
866
            try:
 
867
                # These records should have only one name, and that name
 
868
                # should be a one-element tuple.
 
869
                [name_tuple] = record_names
 
870
            except ValueError:
 
871
                raise errors.SmartProtocolError(
 
872
                    'Repository data stream had invalid record name %r'
 
873
                    % (record_names,))
 
874
            yield name_tuple, read_bytes(None)
 
875
 
 
876
    def insert_data_stream(self, stream):
 
877
        self._ensure_real()
 
878
        self._real_repository.insert_data_stream(stream)
 
879
 
1157
880
    def item_keys_introduced_by(self, revision_ids, _files_pb=None):
1158
881
        self._ensure_real()
1159
882
        return self._real_repository.item_keys_introduced_by(revision_ids,
1173
896
        return self._real_repository._check_for_inconsistent_revision_parents()
1174
897
 
1175
898
    def _make_parents_provider(self):
1176
 
        return self
1177
 
 
1178
 
    def _serialise_search_recipe(self, recipe):
1179
 
        """Serialise a graph search recipe.
1180
 
 
1181
 
        :param recipe: A search recipe (start, stop, count).
1182
 
        :return: Serialised bytes.
1183
 
        """
1184
 
        start_keys = ' '.join(recipe[0])
1185
 
        stop_keys = ' '.join(recipe[1])
1186
 
        count = str(recipe[2])
1187
 
        return '\n'.join((start_keys, stop_keys, count))
 
899
        self._ensure_real()
 
900
        return self._real_repository._make_parents_provider()
1188
901
 
1189
902
 
1190
903
class RemoteBranchLockableFiles(LockableFiles):
1206
919
        self._dir_mode = None
1207
920
        self._file_mode = None
1208
921
 
 
922
    def get(self, path):
 
923
        """'get' a remote path as per the LockableFiles interface.
 
924
 
 
925
        :param path: the file to 'get'. If this is 'branch.conf', we do not
 
926
             just retrieve a file, instead we ask the smart server to generate
 
927
             a configuration for us - which is retrieved as an INI file.
 
928
        """
 
929
        if path == 'branch.conf':
 
930
            path = self.bzrdir._path_for_remote_call(self._client)
 
931
            response = self._client.call_expecting_body(
 
932
                'Branch.get_config_file', path)
 
933
            assert response[0][0] == 'ok', \
 
934
                'unexpected response code %s' % (response[0],)
 
935
            return StringIO(response[1].read_body_bytes())
 
936
        else:
 
937
            # VFS fallback.
 
938
            return LockableFiles.get(self, path)
 
939
 
1209
940
 
1210
941
class RemoteBranchFormat(branch.BranchFormat):
1211
942
 
1220
951
        return 'Remote BZR Branch'
1221
952
 
1222
953
    def open(self, a_bzrdir):
 
954
        assert isinstance(a_bzrdir, RemoteBzrDir)
1223
955
        return a_bzrdir.open_branch()
1224
956
 
1225
957
    def initialize(self, a_bzrdir):
 
958
        assert isinstance(a_bzrdir, RemoteBzrDir)
1226
959
        return a_bzrdir.create_branch()
1227
960
 
1228
961
    def supports_tags(self):
1250
983
        # And the parent's __init__ doesn't do much anyway.
1251
984
        self._revision_id_to_revno_cache = None
1252
985
        self._revision_history_cache = None
1253
 
        self._last_revision_info_cache = None
1254
986
        self.bzrdir = remote_bzrdir
1255
987
        if _client is not None:
1256
988
            self._client = _client
1257
989
        else:
1258
 
            self._client = remote_bzrdir._client
 
990
            self._client = client._SmartClient(self.bzrdir._shared_medium)
1259
991
        self.repository = remote_repository
1260
992
        if real_branch is not None:
1261
993
            self._real_branch = real_branch
1275
1007
        self._control_files = None
1276
1008
        self._lock_mode = None
1277
1009
        self._lock_token = None
1278
 
        self._repo_lock_token = None
1279
1010
        self._lock_count = 0
1280
1011
        self._leave_lock = False
1281
1012
 
1282
 
    def _get_real_transport(self):
1283
 
        # if we try vfs access, return the real branch's vfs transport
1284
 
        self._ensure_real()
1285
 
        return self._real_branch._transport
1286
 
 
1287
 
    _transport = property(_get_real_transport)
1288
 
 
1289
1013
    def __str__(self):
1290
1014
        return "%s(%s)" % (self.__class__.__name__, self.base)
1291
1015
 
1296
1020
 
1297
1021
        Used before calls to self._real_branch.
1298
1022
        """
1299
 
        if self._real_branch is None:
1300
 
            if not vfs.vfs_enabled():
1301
 
                raise AssertionError('smart server vfs must be enabled '
1302
 
                    'to use vfs implementation')
 
1023
        if not self._real_branch:
 
1024
            assert vfs.vfs_enabled()
1303
1025
            self.bzrdir._ensure_real()
1304
1026
            self._real_branch = self.bzrdir._real_bzrdir.open_branch()
1305
1027
            # Give the remote repository the matching real repo.
1314
1036
            if self._lock_mode == 'r':
1315
1037
                self._real_branch.lock_read()
1316
1038
 
1317
 
    def _translate_error(self, err, **context):
1318
 
        self.repository._translate_error(err, branch=self, **context)
1319
 
 
1320
 
    def _clear_cached_state(self):
1321
 
        super(RemoteBranch, self)._clear_cached_state()
1322
 
        if self._real_branch is not None:
1323
 
            self._real_branch._clear_cached_state()
1324
 
 
1325
 
    def _clear_cached_state_of_remote_branch_only(self):
1326
 
        """Like _clear_cached_state, but doesn't clear the cache of
1327
 
        self._real_branch.
1328
 
 
1329
 
        This is useful when falling back to calling a method of
1330
 
        self._real_branch that changes state.  In that case the underlying
1331
 
        branch changes, so we need to invalidate this RemoteBranch's cache of
1332
 
        it.  However, there's no need to invalidate the _real_branch's cache
1333
 
        too, in fact doing so might harm performance.
1334
 
        """
1335
 
        super(RemoteBranch, self)._clear_cached_state()
1336
 
        
1337
1039
    @property
1338
1040
    def control_files(self):
1339
1041
        # Defer actually creating RemoteBranchLockableFiles until its needed,
1353
1055
        self._ensure_real()
1354
1056
        return self._real_branch.get_physical_lock_status()
1355
1057
 
1356
 
    def get_stacked_on_url(self):
1357
 
        """Get the URL this branch is stacked against.
1358
 
 
1359
 
        :raises NotStacked: If the branch is not stacked.
1360
 
        :raises UnstackableBranchFormat: If the branch does not support
1361
 
            stacking.
1362
 
        :raises UnstackableRepositoryFormat: If the repository does not support
1363
 
            stacking.
1364
 
        """
1365
 
        self._ensure_real()
1366
 
        return self._real_branch.get_stacked_on_url()
1367
 
 
1368
1058
    def lock_read(self):
1369
1059
        if not self._lock_mode:
1370
1060
            self._lock_mode = 'r'
1382
1072
            repo_token = self.repository.lock_write()
1383
1073
            self.repository.unlock()
1384
1074
        path = self.bzrdir._path_for_remote_call(self._client)
1385
 
        try:
1386
 
            response = self._client.call(
1387
 
                'Branch.lock_write', path, branch_token, repo_token or '')
1388
 
        except errors.ErrorFromSmartServer, err:
1389
 
            self._translate_error(err, token=token)
1390
 
        if response[0] != 'ok':
 
1075
        response = self._client.call('Branch.lock_write', path, branch_token,
 
1076
                                     repo_token or '')
 
1077
        if response[0] == 'ok':
 
1078
            ok, branch_token, repo_token = response
 
1079
            return branch_token, repo_token
 
1080
        elif response[0] == 'LockContention':
 
1081
            raise errors.LockContention('(remote lock)')
 
1082
        elif response[0] == 'TokenMismatch':
 
1083
            raise errors.TokenMismatch(token, '(remote token)')
 
1084
        elif response[0] == 'UnlockableTransport':
 
1085
            raise errors.UnlockableTransport(self.bzrdir.root_transport)
 
1086
        elif response[0] == 'ReadOnlyError':
 
1087
            raise errors.ReadOnlyError(self)
 
1088
        elif response[0] == 'LockFailed':
 
1089
            raise errors.LockFailed(response[1], response[2])
 
1090
        else:
1391
1091
            raise errors.UnexpectedSmartServerResponse(response)
1392
 
        ok, branch_token, repo_token = response
1393
 
        return branch_token, repo_token
1394
1092
            
1395
1093
    def lock_write(self, token=None):
1396
1094
        if not self._lock_mode:
1397
1095
            remote_tokens = self._remote_lock_write(token)
1398
1096
            self._lock_token, self._repo_lock_token = remote_tokens
1399
 
            if not self._lock_token:
1400
 
                raise SmartProtocolError('Remote server did not return a token!')
 
1097
            assert self._lock_token, 'Remote server did not return a token!'
1401
1098
            # TODO: We really, really, really don't want to call _ensure_real
1402
1099
            # here, but it's the easiest way to ensure coherency between the
1403
1100
            # state of the RemoteBranch and RemoteRepository objects and the
1433
1130
 
1434
1131
    def _unlock(self, branch_token, repo_token):
1435
1132
        path = self.bzrdir._path_for_remote_call(self._client)
1436
 
        try:
1437
 
            response = self._client.call('Branch.unlock', path, branch_token,
1438
 
                                         repo_token or '')
1439
 
        except errors.ErrorFromSmartServer, err:
1440
 
            self._translate_error(err, token=str((branch_token, repo_token)))
 
1133
        response = self._client.call('Branch.unlock', path, branch_token,
 
1134
                                     repo_token or '')
1441
1135
        if response == ('ok',):
1442
1136
            return
1443
 
        raise errors.UnexpectedSmartServerResponse(response)
 
1137
        elif response[0] == 'TokenMismatch':
 
1138
            raise errors.TokenMismatch(
 
1139
                str((branch_token, repo_token)), '(remote tokens)')
 
1140
        else:
 
1141
            raise errors.UnexpectedSmartServerResponse(response)
1444
1142
 
1445
1143
    def unlock(self):
1446
1144
        self._lock_count -= 1
1461
1159
                # Only write-locked branched need to make a remote method call
1462
1160
                # to perfom the unlock.
1463
1161
                return
1464
 
            if not self._lock_token:
1465
 
                raise AssertionError('Locked, but no token!')
 
1162
            assert self._lock_token, 'Locked, but no token!'
1466
1163
            branch_token = self._lock_token
1467
1164
            repo_token = self._repo_lock_token
1468
1165
            self._lock_token = None
1484
1181
            raise NotImplementedError(self.dont_leave_lock_in_place)
1485
1182
        self._leave_lock = False
1486
1183
 
1487
 
    def _last_revision_info(self):
 
1184
    def last_revision_info(self):
 
1185
        """See Branch.last_revision_info()."""
1488
1186
        path = self.bzrdir._path_for_remote_call(self._client)
1489
1187
        response = self._client.call('Branch.last_revision_info', path)
1490
 
        if response[0] != 'ok':
1491
 
            raise SmartProtocolError('unexpected response code %s' % (response,))
 
1188
        assert response[0] == 'ok', 'unexpected response code %s' % (response,)
1492
1189
        revno = int(response[1])
1493
1190
        last_revision = response[2]
1494
1191
        return (revno, last_revision)
1496
1193
    def _gen_revision_history(self):
1497
1194
        """See Branch._gen_revision_history()."""
1498
1195
        path = self.bzrdir._path_for_remote_call(self._client)
1499
 
        response_tuple, response_handler = self._client.call_expecting_body(
 
1196
        response = self._client.call_expecting_body(
1500
1197
            'Branch.revision_history', path)
1501
 
        if response_tuple[0] != 'ok':
1502
 
            raise errors.UnexpectedSmartServerResponse(response_tuple)
1503
 
        result = response_handler.read_body_bytes().split('\x00')
 
1198
        assert response[0][0] == 'ok', ('unexpected response code %s'
 
1199
                                        % (response[0],))
 
1200
        result = response[1].read_body_bytes().split('\x00')
1504
1201
        if result == ['']:
1505
1202
            return []
1506
1203
        return result
1507
1204
 
1508
 
    def _set_last_revision_descendant(self, revision_id, other_branch,
1509
 
            allow_diverged=False, allow_overwrite_descendant=False):
1510
 
        path = self.bzrdir._path_for_remote_call(self._client)
1511
 
        try:
1512
 
            response = self._client.call('Branch.set_last_revision_ex',
1513
 
                path, self._lock_token, self._repo_lock_token, revision_id,
1514
 
                int(allow_diverged), int(allow_overwrite_descendant))
1515
 
        except errors.ErrorFromSmartServer, err:
1516
 
            self._translate_error(err, other_branch=other_branch)
1517
 
        self._clear_cached_state()
1518
 
        if len(response) != 3 and response[0] != 'ok':
1519
 
            raise errors.UnexpectedSmartServerResponse(response)
1520
 
        new_revno, new_revision_id = response[1:]
1521
 
        self._last_revision_info_cache = new_revno, new_revision_id
1522
 
        self._real_branch._last_revision_info_cache = new_revno, new_revision_id
1523
 
 
1524
 
    def _set_last_revision(self, revision_id):
1525
 
        path = self.bzrdir._path_for_remote_call(self._client)
1526
 
        self._clear_cached_state()
1527
 
        try:
1528
 
            response = self._client.call('Branch.set_last_revision',
1529
 
                path, self._lock_token, self._repo_lock_token, revision_id)
1530
 
        except errors.ErrorFromSmartServer, err:
1531
 
            self._translate_error(err)
1532
 
        if response != ('ok',):
1533
 
            raise errors.UnexpectedSmartServerResponse(response)
1534
 
 
1535
1205
    @needs_write_lock
1536
1206
    def set_revision_history(self, rev_history):
1537
1207
        # Send just the tip revision of the history; the server will generate
1538
1208
        # the full history from that.  If the revision doesn't exist in this
1539
1209
        # branch, NoSuchRevision will be raised.
 
1210
        path = self.bzrdir._path_for_remote_call(self._client)
1540
1211
        if rev_history == []:
1541
1212
            rev_id = 'null:'
1542
1213
        else:
1543
1214
            rev_id = rev_history[-1]
1544
 
        self._set_last_revision(rev_id)
 
1215
        self._clear_cached_state()
 
1216
        response = self._client.call('Branch.set_last_revision',
 
1217
            path, self._lock_token, self._repo_lock_token, rev_id)
 
1218
        if response[0] == 'NoSuchRevision':
 
1219
            raise NoSuchRevision(self, rev_id)
 
1220
        else:
 
1221
            assert response == ('ok',), (
 
1222
                'unexpected response code %r' % (response,))
1545
1223
        self._cache_revision_history(rev_history)
1546
1224
 
1547
1225
    def get_parent(self):
1552
1230
        self._ensure_real()
1553
1231
        return self._real_branch.set_parent(url)
1554
1232
        
1555
 
    def set_stacked_on_url(self, stacked_location):
1556
 
        """Set the URL this branch is stacked against.
1557
 
 
1558
 
        :raises UnstackableBranchFormat: If the branch does not support
1559
 
            stacking.
1560
 
        :raises UnstackableRepositoryFormat: If the repository does not support
1561
 
            stacking.
1562
 
        """
1563
 
        self._ensure_real()
1564
 
        return self._real_branch.set_stacked_on_url(stacked_location)
 
1233
    def get_config(self):
 
1234
        return RemoteBranchConfig(self)
1565
1235
 
1566
1236
    def sprout(self, to_bzrdir, revision_id=None):
1567
1237
        # Like Branch.sprout, except that it sprouts a branch in the default
1577
1247
    @needs_write_lock
1578
1248
    def pull(self, source, overwrite=False, stop_revision=None,
1579
1249
             **kwargs):
1580
 
        self._clear_cached_state_of_remote_branch_only()
 
1250
        # FIXME: This asks the real branch to run the hooks, which means
 
1251
        # they're called with the wrong target branch parameter. 
 
1252
        # The test suite specifically allows this at present but it should be
 
1253
        # fixed.  It should get a _override_hook_target branch,
 
1254
        # as push does.  -- mbp 20070405
1581
1255
        self._ensure_real()
1582
 
        return self._real_branch.pull(
 
1256
        self._real_branch.pull(
1583
1257
            source, overwrite=overwrite, stop_revision=stop_revision,
1584
 
            _override_hook_target=self, **kwargs)
 
1258
            **kwargs)
1585
1259
 
1586
1260
    @needs_read_lock
1587
1261
    def push(self, target, overwrite=False, stop_revision=None):
1593
1267
    def is_locked(self):
1594
1268
        return self._lock_count >= 1
1595
1269
 
1596
 
    @needs_read_lock
1597
 
    def revision_id_to_revno(self, revision_id):
1598
 
        self._ensure_real()
1599
 
        return self._real_branch.revision_id_to_revno(revision_id)
1600
 
 
1601
 
    @needs_write_lock
1602
1270
    def set_last_revision_info(self, revno, revision_id):
1603
 
        revision_id = ensure_null(revision_id)
1604
 
        path = self.bzrdir._path_for_remote_call(self._client)
1605
 
        try:
1606
 
            response = self._client.call('Branch.set_last_revision_info',
1607
 
                path, self._lock_token, self._repo_lock_token, str(revno), revision_id)
1608
 
        except errors.UnknownSmartMethod:
1609
 
            self._ensure_real()
1610
 
            self._clear_cached_state_of_remote_branch_only()
1611
 
            self._real_branch.set_last_revision_info(revno, revision_id)
1612
 
            self._last_revision_info_cache = revno, revision_id
1613
 
            return
1614
 
        except errors.ErrorFromSmartServer, err:
1615
 
            self._translate_error(err)
1616
 
        if response == ('ok',):
1617
 
            self._clear_cached_state()
1618
 
            self._last_revision_info_cache = revno, revision_id
1619
 
            # Update the _real_branch's cache too.
1620
 
            if self._real_branch is not None:
1621
 
                cache = self._last_revision_info_cache
1622
 
                self._real_branch._last_revision_info_cache = cache
1623
 
        else:
1624
 
            raise errors.UnexpectedSmartServerResponse(response)
 
1271
        self._ensure_real()
 
1272
        self._clear_cached_state()
 
1273
        return self._real_branch.set_last_revision_info(revno, revision_id)
1625
1274
 
1626
 
    @needs_write_lock
1627
1275
    def generate_revision_history(self, revision_id, last_rev=None,
1628
1276
                                  other_branch=None):
1629
 
        medium = self._client._medium
1630
 
        if not medium._is_remote_before((1, 6)):
1631
 
            try:
1632
 
                self._set_last_revision_descendant(revision_id, other_branch,
1633
 
                    allow_diverged=True, allow_overwrite_descendant=True)
1634
 
                return
1635
 
            except errors.UnknownSmartMethod:
1636
 
                medium._remember_remote_is_before((1, 6))
1637
 
        self._clear_cached_state_of_remote_branch_only()
1638
1277
        self._ensure_real()
1639
 
        self._real_branch.generate_revision_history(
 
1278
        return self._real_branch.generate_revision_history(
1640
1279
            revision_id, last_rev=last_rev, other_branch=other_branch)
1641
1280
 
1642
1281
    @property
1648
1287
        self._ensure_real()
1649
1288
        return self._real_branch.set_push_location(location)
1650
1289
 
1651
 
    @needs_write_lock
1652
 
    def update_revisions(self, other, stop_revision=None, overwrite=False,
1653
 
                         graph=None):
1654
 
        """See Branch.update_revisions."""
1655
 
        other.lock_read()
1656
 
        try:
1657
 
            if stop_revision is None:
1658
 
                stop_revision = other.last_revision()
1659
 
                if revision.is_null(stop_revision):
1660
 
                    # if there are no commits, we're done.
1661
 
                    return
1662
 
            self.fetch(other, stop_revision)
1663
 
 
1664
 
            if overwrite:
1665
 
                # Just unconditionally set the new revision.  We don't care if
1666
 
                # the branches have diverged.
1667
 
                self._set_last_revision(stop_revision)
1668
 
            else:
1669
 
                medium = self._client._medium
1670
 
                if not medium._is_remote_before((1, 6)):
1671
 
                    try:
1672
 
                        self._set_last_revision_descendant(stop_revision, other)
1673
 
                        return
1674
 
                    except errors.UnknownSmartMethod:
1675
 
                        medium._remember_remote_is_before((1, 6))
1676
 
                # Fallback for pre-1.6 servers: check for divergence
1677
 
                # client-side, then do _set_last_revision.
1678
 
                last_rev = revision.ensure_null(self.last_revision())
1679
 
                if graph is None:
1680
 
                    graph = self.repository.get_graph()
1681
 
                if self._check_if_descendant_or_diverged(
1682
 
                        stop_revision, last_rev, graph, other):
1683
 
                    # stop_revision is a descendant of last_rev, but we aren't
1684
 
                    # overwriting, so we're done.
1685
 
                    return
1686
 
                self._set_last_revision(stop_revision)
1687
 
        finally:
1688
 
            other.unlock()
 
1290
    def update_revisions(self, other, stop_revision=None, overwrite=False):
 
1291
        self._ensure_real()
 
1292
        return self._real_branch.update_revisions(
 
1293
            other, stop_revision=stop_revision, overwrite=overwrite)
 
1294
 
 
1295
 
 
1296
class RemoteBranchConfig(BranchConfig):
 
1297
 
 
1298
    def username(self):
 
1299
        self.branch._ensure_real()
 
1300
        return self.branch._real_branch.get_config().username()
 
1301
 
 
1302
    def _get_branch_data_config(self):
 
1303
        self.branch._ensure_real()
 
1304
        if self._branch_data_config is None:
 
1305
            self._branch_data_config = TreeConfig(self.branch._real_branch)
 
1306
        return self._branch_data_config
1689
1307
 
1690
1308
 
1691
1309
def _extract_tar(tar, to_dir):
1695
1313
    """
1696
1314
    for tarinfo in tar:
1697
1315
        tar.extract(tarinfo, to_dir)
1698
 
 
1699
 
 
1700
 
def _translate_error(err, **context):
1701
 
    """Translate an ErrorFromSmartServer into a more useful error.
1702
 
 
1703
 
    Possible context keys:
1704
 
      - branch
1705
 
      - repository
1706
 
      - bzrdir
1707
 
      - token
1708
 
      - other_branch
1709
 
    """
1710
 
    def find(name):
1711
 
        try:
1712
 
            return context[name]
1713
 
        except KeyError, keyErr:
1714
 
            mutter('Missing key %r in context %r', keyErr.args[0], context)
1715
 
            raise err
1716
 
    if err.error_verb == 'NoSuchRevision':
1717
 
        raise NoSuchRevision(find('branch'), err.error_args[0])
1718
 
    elif err.error_verb == 'nosuchrevision':
1719
 
        raise NoSuchRevision(find('repository'), err.error_args[0])
1720
 
    elif err.error_tuple == ('nobranch',):
1721
 
        raise errors.NotBranchError(path=find('bzrdir').root_transport.base)
1722
 
    elif err.error_verb == 'norepository':
1723
 
        raise errors.NoRepositoryPresent(find('bzrdir'))
1724
 
    elif err.error_verb == 'LockContention':
1725
 
        raise errors.LockContention('(remote lock)')
1726
 
    elif err.error_verb == 'UnlockableTransport':
1727
 
        raise errors.UnlockableTransport(find('bzrdir').root_transport)
1728
 
    elif err.error_verb == 'LockFailed':
1729
 
        raise errors.LockFailed(err.error_args[0], err.error_args[1])
1730
 
    elif err.error_verb == 'TokenMismatch':
1731
 
        raise errors.TokenMismatch(find('token'), '(remote token)')
1732
 
    elif err.error_verb == 'Diverged':
1733
 
        raise errors.DivergedBranches(find('branch'), find('other_branch'))
1734
 
    elif err.error_verb == 'TipChangeRejected':
1735
 
        raise errors.TipChangeRejected(err.error_args[0].decode('utf8'))
1736
 
    raise
1737