# HG changeset patch # User Matt Mackall # Date 2012-05-17 20:34:59 # Node ID 27b2e1823e830e7d117cedcf9a9f88d0fbd728b9 # Parent 1c9f58a6c8f11605825b766f79f32db90470dc69 branchcache: backout 0311a6abd38a diff --git a/mercurial/localrepo.py b/mercurial/localrepo.py --- a/mercurial/localrepo.py +++ b/mercurial/localrepo.py @@ -550,9 +550,6 @@ class localrepository(repo.repository): continue node, label = l.split(" ", 1) label = encoding.tolocal(label.strip()) - if not node in self: - raise ValueError('invalidating branch cache because node '+ - '%s does not exist' % node) partial.setdefault(label, []).append(bin(node)) except KeyboardInterrupt: raise @@ -574,10 +571,6 @@ class localrepository(repo.repository): pass def _updatebranchcache(self, partial, ctxgen): - """Given a branchhead cache, partial, that may have extra nodes or be - missing heads, and a generator of nodes that are at least a superset of - heads missing, this function updates partial to be correct. - """ # collect new branch entries newbranches = {} for c in ctxgen: @@ -588,42 +581,21 @@ class localrepository(repo.repository): for branch, newnodes in newbranches.iteritems(): bheads = partial.setdefault(branch, []) bheads.extend(newnodes) - # Remove duplicates - nodes that are in newnodes and are already in - # bheads. This can happen if you strip a node and its parent was - # already a head (because they're on different branches). - bheads = set(bheads) - - # Remove candidate heads that no longer are in the repo (e.g., as - # the result of a strip that just happened). - # avoid using 'bhead in self' here because that dives down into - # branchcache code somewhat recrusively. - bheads = [bhead for bhead in bheads \ - if self.changelog.hasnode(bhead)] - if len(bheads) > 1: - bheads = sorted(bheads, key=lambda x: self[x].rev()) - # starting from tip means fewer passes over reachable - while newnodes: - latest = newnodes.pop() - if latest not in bheads: - continue - minbhnode = self[bheads[0]].node() - reachable = self.changelog.reachable(latest, minbhnode) - reachable.remove(latest) - if reachable: - bheads = [b for b in bheads if b not in reachable] + if len(bheads) <= 1: + continue + bheads = sorted(bheads, key=lambda x: self[x].rev()) + # starting from tip means fewer passes over reachable + while newnodes: + latest = newnodes.pop() + if latest not in bheads: + continue + minbhnode = self[bheads[0]].node() + reachable = self.changelog.reachable(latest, minbhnode) + reachable.remove(latest) + if reachable: + bheads = [b for b in bheads if b not in reachable] partial[branch] = bheads - # There may be branches that cease to exist when the last commit in the - # branch was stripped. This code filters them out. Note that the - # branch that ceased to exist may not be in newbranches because - # newbranches is the set of candidate heads, which when you strip the - # last commit in a branch will be the parent branch. - for branch in partial.keys(): - nodes = [head for head in partial[branch] \ - if self.changelog.hasnode(head)] - if len(nodes) < 1: - del partial[branch] - def lookup(self, key): return self[key].node() @@ -886,9 +858,6 @@ class localrepository(repo.repository): else: ui.status(_('working directory now based on ' 'revision %d\n') % parents) - # TODO: if we know which new heads may result from this rollback, pass - # them to destroy(), which will prevent the branchhead cache from being - # invalidated. self.destroyed() return 0 @@ -1332,27 +1301,12 @@ class localrepository(repo.repository): tr.release() lock.release() - def destroyed(self, newheadrevs=None): + def destroyed(self): '''Inform the repository that nodes have been destroyed. Intended for use by strip and rollback, so there's a common - place for anything that has to be done after destroying history. - - If you know the branchheadcache was uptodate before nodes were removed - and you also know the set of candidate set of new heads that may have - resulted from the destruction, you can set newheadrevs. This will - enable the code to update the branchheads cache, rather than having - future code decide it's invalid and regenrating it. - ''' - if newheadrevs: - tiprev = len(self) - 1 - ctxgen = (self[rev] for rev in newheadrevs) - self._updatebranchcache(self._branchcache, ctxgen) - self._writebranchcache(self._branchcache, self.changelog.tip(), - tiprev) - else: - # No info to update the cache. If nodes were destroyed, the cache - # is stale and this will be caught the next time it is read. - pass + place for anything that has to be done after destroying history.''' + # XXX it might be nice if we could take the list of destroyed + # nodes, but I don't see an easy way for rollback() to do that # Ensure the persistent tag cache is updated. Doing it now # means that the tag cache only has to worry about destroyed diff --git a/mercurial/repair.py b/mercurial/repair.py --- a/mercurial/repair.py +++ b/mercurial/repair.py @@ -56,11 +56,6 @@ def _collectbrokencsets(repo, files, str return s def strip(ui, repo, nodelist, backup="all", topic='backup'): - # It simplifies the logic around updating the branchheads cache if we only - # have to consider the effect of the stripped revisions and not revisions - # missing because the cache is out-of-date. - repo.updatebranchcache() - cl = repo.changelog # TODO handle undo of merge sets if isinstance(nodelist, str): @@ -68,14 +63,6 @@ def strip(ui, repo, nodelist, backup="al striplist = [cl.rev(node) for node in nodelist] striprev = min(striplist) - # Set of potential new heads resulting from the strip. The parents of any - # node removed could be a new head because the node to be removed could have - # been the only child of the parent. - # Do a list->set->list conversion to remove duplicates. - stringstriplist = [str(rev) for rev in striplist] - newheadrevs = set(repo.revs("parents(%lr::) - %lr::", stringstriplist, - stringstriplist)) - keeppartialbundle = backup == 'strip' # Some revisions with rev > striprev may not be descendants of striprev. @@ -182,4 +169,4 @@ def strip(ui, repo, nodelist, backup="al % chgrpfile) raise - repo.destroyed(newheadrevs) + repo.destroyed() diff --git a/tests/test-commit-amend.t b/tests/test-commit-amend.t --- a/tests/test-commit-amend.t +++ b/tests/test-commit-amend.t @@ -247,7 +247,6 @@ Same thing, different code path: $ echo b >> b $ hg ci -m 'reopen branch' - created new head reopening closed branch head 4 $ echo b >> b $ hg ci --amend --close-branch diff --git a/tests/test-mq-caches.t b/tests/test-mq-caches.t --- a/tests/test-mq-caches.t +++ b/tests/test-mq-caches.t @@ -36,8 +36,7 @@ mq patch on an empty repo $ hg qrefresh -m 'patch 1' $ show_branch_cache tip: 0 - d986d5caac23a7d44a46efc0ddaf5eb9665844cf 0 - d986d5caac23a7d44a46efc0ddaf5eb9665844cf default + No branch cache some regular revisions @@ -76,8 +75,8 @@ add some mq patches $ hg qrefresh -m 'patch 2' $ show_branch_cache 1 tip: 3 - 982611f6955f9c48d3365decea203217c945ef0d 2 - 982611f6955f9c48d3365decea203217c945ef0d bar + c229711f16da3d7591f89b1b8d963b79bda22714 1 + c229711f16da3d7591f89b1b8d963b79bda22714 bar dc25e3827021582e979f600811852e36cbe57341 foo branch foo: 3 branch bar: 2