discovery.py
659 lines
| 23.5 KiB
| text/x-python
|
PythonLexer
/ mercurial / discovery.py
Dirkjan Ochtman
|
r11313 | # discovery.py - protocol changeset discovery functions | ||
Dirkjan Ochtman
|
r11301 | # | ||
Raphaël Gomès
|
r47575 | # Copyright 2010 Olivia Mackall <olivia@selenic.com> | ||
Dirkjan Ochtman
|
r11301 | # | ||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Gregory Szorc
|
r25944 | |||
Pierre-Yves David
|
r32009 | import functools | ||
Gregory Szorc
|
r25944 | from .i18n import _ | ||
from .node import ( | ||||
Pierre-Yves David
|
r32009 | hex, | ||
Gregory Szorc
|
r25944 | short, | ||
) | ||||
from . import ( | ||||
bookmarks, | ||||
branchmap, | ||||
Pierre-Yves David
|
r26587 | error, | ||
r52487 | node as nodemod, | |||
r49537 | obsolete, | |||
Gregory Szorc
|
r25944 | phases, | ||
Gregory Szorc
|
r43376 | pycompat, | ||
Boris Feld
|
r35185 | scmutil, | ||
Gregory Szorc
|
r25944 | setdiscovery, | ||
treediscovery, | ||||
util, | ||||
) | ||||
Dirkjan Ochtman
|
r11301 | |||
Augie Fackler
|
r43346 | |||
Boris Feld
|
r35306 | def findcommonincoming(repo, remote, heads=None, force=False, ancestorsof=None): | ||
Peter Arrenbrecht
|
r14073 | """Return a tuple (common, anyincoming, heads) used to identify the common | ||
subset of nodes between repo and remote. | ||||
Dirkjan Ochtman
|
r11301 | |||
Peter Arrenbrecht
|
r14073 | "common" is a list of (at least) the heads of the common subset. | ||
"anyincoming" is testable as a boolean indicating if any nodes are missing | ||||
locally. If remote does not support getbundle, this actually is a list of | ||||
roots of the nodes that would be incoming, to be supplied to | ||||
changegroupsubset. No code except for pull should be relying on this fact | ||||
any longer. | ||||
"heads" is either the supplied heads, or else the remote's heads. | ||||
Boris Feld
|
r35306 | "ancestorsof" if not None, restrict the discovery to a subset defined by | ||
Manuel Jacob
|
r45702 | these nodes. Changeset outside of this set won't be considered (but may | ||
still appear in "common"). | ||||
Peter Arrenbrecht
|
r14164 | |||
Mads Kiilerich
|
r17424 | If you pass heads and they are all known locally, the response lists just | ||
Peter Arrenbrecht
|
r14164 | these heads in "common" and in "heads". | ||
Peter Arrenbrecht
|
r14213 | |||
Please use findcommonoutgoing to compute the set of outgoing nodes to give | ||||
extensions a good hook into outgoing. | ||||
Dirkjan Ochtman
|
r11301 | """ | ||
Peter Arrenbrecht
|
r14073 | |||
Augie Fackler
|
r43347 | if not remote.capable(b'getbundle'): | ||
Peter Arrenbrecht
|
r14164 | return treediscovery.findcommonincoming(repo, remote, heads, force) | ||
Dirkjan Ochtman
|
r11301 | |||
Peter Arrenbrecht
|
r14164 | if heads: | ||
Augie Fackler
|
r43346 | knownnode = repo.changelog.hasnode # no nodemap until it is filtered | ||
Martin von Zweigbergk
|
r35897 | if all(knownnode(h) for h in heads): | ||
Peter Arrenbrecht
|
r14164 | return (heads, False, heads) | ||
Dirkjan Ochtman
|
r11301 | |||
Augie Fackler
|
r43346 | res = setdiscovery.findcommonheads( | ||
repo.ui, | ||||
repo, | ||||
remote, | ||||
abortwhenunrelated=not force, | ||||
ancestorsof=ancestorsof, | ||||
) | ||||
Peter Arrenbrecht
|
r14164 | common, anyinc, srvheads = res | ||
r45166 | if heads and not anyinc: | |||
# server could be lying on the advertised heads | ||||
has_node = repo.changelog.hasnode | ||||
anyinc = any(not has_node(n) for n in heads) | ||||
Peter Arrenbrecht
|
r14164 | return (list(common), anyinc, heads or list(srvheads)) | ||
Dirkjan Ochtman
|
r11301 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class outgoing: | ||
Augie Fackler
|
r46554 | """Represents the result of a findcommonoutgoing() call. | ||
Pierre-Yves David
|
r15837 | |||
Members: | ||||
Manuel Jacob
|
r45703 | ancestorsof is a list of the nodes whose ancestors are included in the | ||
outgoing operation. | ||||
missing is a list of those ancestors of ancestorsof that are present in | ||||
local but not in remote. | ||||
common is a set containing revs common between the local and the remote | ||||
repository (at least all of those that are ancestors of ancestorsof). | ||||
Pierre-Yves David
|
r15837 | commonheads is the list of heads of common. | ||
Manuel Jacob
|
r45703 | excluded is the list of missing changeset that shouldn't be sent | ||
remotely. | ||||
Some members are computed on demand from the heads, unless provided upfront | ||||
Augie Fackler
|
r46554 | by discovery.""" | ||
Pierre-Yves David
|
r15837 | |||
Augie Fackler
|
r43346 | def __init__( | ||
Manuel Jacob
|
r45704 | self, repo, commonheads=None, ancestorsof=None, missingroots=None | ||
Augie Fackler
|
r43346 | ): | ||
r52487 | # at most one of them must not be set | |||
if commonheads is not None and missingroots is not None: | ||||
m = 'commonheads and missingroots arguments are mutually exclusive' | ||||
raise error.ProgrammingError(m) | ||||
Pierre-Yves David
|
r29805 | cl = repo.changelog | ||
r52488 | unfi = repo.unfiltered() | |||
ucl = unfi.changelog | ||||
to_node = ucl.node | ||||
r52487 | missing = None | |||
common = None | ||||
r52488 | arg_anc = ancestorsof | |||
Manuel Jacob
|
r45704 | if ancestorsof is None: | ||
ancestorsof = cl.heads() | ||||
r52488 | ||||
# XXX-perf: do we need all this to be node-list? They would be simpler | ||||
# as rev-num sets (and smartset) | ||||
if missingroots == [nodemod.nullrev] or missingroots == []: | ||||
commonheads = [repo.nullid] | ||||
common = set() | ||||
if arg_anc is None: | ||||
missing = [to_node(r) for r in cl] | ||||
else: | ||||
missing_rev = repo.revs('::%ln', missingroots, ancestorsof) | ||||
missing = [to_node(r) for r in missing_rev] | ||||
elif missingroots is not None: | ||||
Pierre-Yves David
|
r29806 | # TODO remove call to nodesbetween. | ||
r52487 | missing_rev = repo.revs('%ln::%ln', missingroots, ancestorsof) | |||
ancestorsof = [to_node(r) for r in ucl.headrevs(missing_rev)] | ||||
parent_revs = ucl.parentrevs | ||||
common_legs = set() | ||||
for r in missing_rev: | ||||
p1, p2 = parent_revs(r) | ||||
if p1 not in missing_rev: | ||||
common_legs.add(p1) | ||||
if p2 not in missing_rev: | ||||
common_legs.add(p2) | ||||
common_legs.discard(nodemod.nullrev) | ||||
if not common_legs: | ||||
commonheads = [repo.nullid] | ||||
common = set() | ||||
else: | ||||
commonheads_revs = unfi.revs( | ||||
'heads(%ld::%ld)', | ||||
common_legs, | ||||
common_legs, | ||||
) | ||||
commonheads = [to_node(r) for r in commonheads_revs] | ||||
common = ucl.ancestors(commonheads_revs, inclusive=True) | ||||
missing = [to_node(r) for r in missing_rev] | ||||
Pierre-Yves David
|
r29806 | elif not commonheads: | ||
Joerg Sonnenberger
|
r47771 | commonheads = [repo.nullid] | ||
Pierre-Yves David
|
r15837 | self.commonheads = commonheads | ||
Manuel Jacob
|
r45704 | self.ancestorsof = ancestorsof | ||
Pierre-Yves David
|
r29805 | self._revlog = cl | ||
r52487 | self._common = common | |||
self._missing = missing | ||||
Pierre-Yves David
|
r15838 | self.excluded = [] | ||
Pierre-Yves David
|
r15837 | |||
def _computecommonmissing(self): | ||||
Augie Fackler
|
r43346 | sets = self._revlog.findcommonmissing( | ||
Manuel Jacob
|
r45704 | self.commonheads, self.ancestorsof | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r15837 | self._common, self._missing = sets | ||
@util.propertycache | ||||
def common(self): | ||||
if self._common is None: | ||||
self._computecommonmissing() | ||||
return self._common | ||||
@util.propertycache | ||||
def missing(self): | ||||
if self._missing is None: | ||||
self._computecommonmissing() | ||||
return self._missing | ||||
Augie Fackler
|
r43346 | |||
def findcommonoutgoing( | ||||
repo, other, onlyheads=None, force=False, commoninc=None, portable=False | ||||
): | ||||
Augie Fackler
|
r46554 | """Return an outgoing instance to identify the nodes present in repo but | ||
Pierre-Yves David
|
r15837 | not in other. | ||
Peter Arrenbrecht
|
r14213 | |||
Brodie Rao
|
r16683 | If onlyheads is given, only nodes ancestral to nodes in onlyheads | ||
(inclusive) are included. If you already know the local repo's heads, | ||||
passing them in onlyheads is faster than letting them be recomputed here. | ||||
Peter Arrenbrecht
|
r14213 | |||
Mads Kiilerich
|
r17251 | If commoninc is given, it must be the result of a prior call to | ||
Sune Foldager
|
r16736 | findcommonincoming(repo, other, force) to avoid recomputing it here. | ||
Manuel Jacob
|
r45704 | If portable is given, compute more conservative common and ancestorsof, | ||
Augie Fackler
|
r46554 | to make bundles created from the instance more portable.""" | ||
Pierre-Yves David
|
r15838 | # declare an empty outgoing object to be filled later | ||
Pierre-Yves David
|
r29804 | og = outgoing(repo, None, None) | ||
Pierre-Yves David
|
r15838 | |||
# get common set if not provided | ||||
if commoninc is None: | ||||
Augie Fackler
|
r43346 | commoninc = findcommonincoming( | ||
repo, other, force=force, ancestorsof=onlyheads | ||||
) | ||||
Pierre-Yves David
|
r15838 | og.commonheads, _any, _hds = commoninc | ||
# compute outgoing | ||||
r52295 | mayexclude = phases.hassecret(repo) or repo.obsstore | |||
Pierre-Yves David
|
r17206 | if not mayexclude: | ||
Manuel Jacob
|
r45704 | og.ancestorsof = onlyheads or repo.heads() | ||
Pierre-Yves David
|
r15838 | elif onlyheads is None: | ||
# use visible heads as it should be cached | ||||
Manuel Jacob
|
r45704 | og.ancestorsof = repo.filtered(b"served").heads() | ||
Augie Fackler
|
r43347 | og.excluded = [ctx.node() for ctx in repo.set(b'secret() or extinct()')] | ||
Pierre-Yves David
|
r15838 | else: | ||
# compute common, missing and exclude secret stuff | ||||
sets = repo.changelog.findcommonmissing(og.commonheads, onlyheads) | ||||
og._common, allmissing = sets | ||||
og._missing = missing = [] | ||||
Pierre-Yves David
|
r15951 | og.excluded = excluded = [] | ||
Pierre-Yves David
|
r15838 | for node in allmissing: | ||
Pierre-Yves David
|
r17206 | ctx = repo[node] | ||
Patrick Mezard
|
r17248 | if ctx.phase() >= phases.secret or ctx.extinct(): | ||
excluded.append(node) | ||||
else: | ||||
missing.append(node) | ||||
Pierre-Yves David
|
r17206 | if len(missing) == len(allmissing): | ||
Manuel Jacob
|
r45704 | ancestorsof = onlyheads | ||
Augie Fackler
|
r43346 | else: # update missing heads | ||
r52473 | to_rev = repo.changelog.index.rev | |||
to_node = repo.changelog.node | ||||
excluded_revs = [to_rev(r) for r in excluded] | ||||
onlyheads_revs = [to_rev(r) for r in onlyheads] | ||||
new_heads = phases.new_heads(repo, onlyheads_revs, excluded_revs) | ||||
ancestorsof = [to_node(r) for r in new_heads] | ||||
Manuel Jacob
|
r45704 | og.ancestorsof = ancestorsof | ||
Sune Foldager
|
r16736 | if portable: | ||
Manuel Jacob
|
r45704 | # recompute common and ancestorsof as if -r<rev> had been given for | ||
Sune Foldager
|
r16736 | # each head of missing, and --base <rev> for each head of the proper | ||
# ancestors of missing | ||||
og._computecommonmissing() | ||||
cl = repo.changelog | ||||
Augie Fackler
|
r44937 | missingrevs = {cl.rev(n) for n in og._missing} | ||
Bryan O'Sullivan
|
r16866 | og._common = set(cl.ancestors(missingrevs)) - missingrevs | ||
Sune Foldager
|
r16736 | commonheads = set(og.commonheads) | ||
Manuel Jacob
|
r45704 | og.ancestorsof = [h for h in og.ancestorsof if h not in commonheads] | ||
Sune Foldager
|
r16736 | |||
Pierre-Yves David
|
r15838 | return og | ||
Peter Arrenbrecht
|
r14213 | |||
Augie Fackler
|
r43346 | |||
r32706 | def _headssummary(pushop): | |||
Pierre-Yves David
|
r17209 | """compute a summary of branch and heads status before and after push | ||
r32708 | return {'branch': ([remoteheads], [newheads], | |||
[unsyncedheads], [discardedheads])} mapping | ||||
Pierre-Yves David
|
r17211 | |||
r32708 | - branch: the branch name, | |||
Pierre-Yves David
|
r17211 | - remoteheads: the list of remote heads known locally | ||
r32708 | None if the branch is new, | |||
- newheads: the new remote heads (known locally) with outgoing pushed, | ||||
- unsyncedheads: the list of remote heads unknown locally, | ||||
- discardedheads: the list of heads made obsolete by the push. | ||||
Pierre-Yves David
|
r17209 | """ | ||
r32706 | repo = pushop.repo.unfiltered() | |||
remote = pushop.remote | ||||
outgoing = pushop.outgoing | ||||
Pierre-Yves David
|
r17209 | cl = repo.changelog | ||
Pierre-Yves David
|
r17211 | headssum = {} | ||
Pulkit Goyal
|
r42193 | missingctx = set() | ||
Pierre-Yves David
|
r17209 | # A. Create set of branches involved in the push. | ||
Pulkit Goyal
|
r42193 | branches = set() | ||
for n in outgoing.missing: | ||||
ctx = repo[n] | ||||
missingctx.add(ctx) | ||||
branches.add(ctx.branch()) | ||||
Gregory Szorc
|
r37655 | |||
with remote.commandexecutor() as e: | ||||
Augie Fackler
|
r43347 | remotemap = e.callcommand(b'branchmap', {}).result() | ||
Gregory Szorc
|
r37655 | |||
Augie Fackler
|
r43346 | knownnode = cl.hasnode # do not use nodemap until it is filtered | ||
Pulkit Goyal
|
r42195 | # A. register remote heads of branches which are in outgoing set | ||
Gregory Szorc
|
r49768 | for branch, heads in remotemap.items(): | ||
Pulkit Goyal
|
r42195 | # don't add head info about branches which we don't have locally | ||
if branch not in branches: | ||||
continue | ||||
Pierre-Yves David
|
r17211 | known = [] | ||
unsynced = [] | ||||
for h in heads: | ||||
Pierre-Yves David
|
r20225 | if knownnode(h): | ||
Pierre-Yves David
|
r17211 | known.append(h) | ||
else: | ||||
unsynced.append(h) | ||||
headssum[branch] = (known, list(known), unsynced) | ||||
Pulkit Goyal
|
r42195 | |||
Pierre-Yves David
|
r17211 | # B. add new branch data | ||
Pulkit Goyal
|
r42194 | for branch in branches: | ||
Pierre-Yves David
|
r17211 | if branch not in headssum: | ||
headssum[branch] = (None, [], []) | ||||
Pierre-Yves David
|
r17209 | |||
Pulkit Goyal
|
r42195 | # C. Update newmap with outgoing changes. | ||
Pierre-Yves David
|
r17209 | # This will possibly add new heads and remove existing ones. | ||
Augie Fackler
|
r43346 | newmap = branchmap.remotebranchcache( | ||
Joerg Sonnenberger
|
r47538 | repo, | ||
( | ||||
(branch, heads[1]) | ||||
Gregory Szorc
|
r49768 | for branch, heads in headssum.items() | ||
Joerg Sonnenberger
|
r47538 | if heads[0] is not None | ||
), | ||||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r18305 | newmap.update(repo, (ctx.rev() for ctx in missingctx)) | ||
Gregory Szorc
|
r49768 | for branch, newheads in newmap.items(): | ||
Pierre-Yves David
|
r17211 | headssum[branch][1][:] = newheads | ||
Gregory Szorc
|
r49768 | for branch, items in headssum.items(): | ||
r32672 | for l in items: | |||
if l is not None: | ||||
l.sort() | ||||
r32708 | headssum[branch] = items + ([],) | |||
r32707 | # If there are no obsstore, no post processing are needed. | |||
if repo.obsstore: | ||||
r32791 | torev = repo.changelog.rev | |||
Manuel Jacob
|
r45704 | futureheads = {torev(h) for h in outgoing.ancestorsof} | ||
Augie Fackler
|
r44937 | futureheads |= {torev(h) for h in outgoing.commonheads} | ||
r32792 | allfuturecommon = repo.changelog.ancestors(futureheads, inclusive=True) | |||
Gregory Szorc
|
r43376 | for branch, heads in sorted(pycompat.iteritems(headssum)): | ||
r32708 | remoteheads, newheads, unsyncedheads, placeholder = heads | |||
r32707 | result = _postprocessobsolete(pushop, allfuturecommon, newheads) | |||
Augie Fackler
|
r43346 | headssum[branch] = ( | ||
remoteheads, | ||||
sorted(result[0]), | ||||
unsyncedheads, | ||||
sorted(result[1]), | ||||
) | ||||
Pierre-Yves David
|
r17211 | return headssum | ||
Pierre-Yves David
|
r17209 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r17211 | def _oldheadssummary(repo, remoteheads, outgoing, inc=False): | ||
Pierre-Yves David
|
r17209 | """Compute branchmapsummary for repo without branchmap support""" | ||
# 1-4b. old servers: Check for new topological heads. | ||||
# Construct {old,new}map with branch = None (topological branch). | ||||
Pierre-Yves David
|
r18120 | # (code based on update) | ||
Augie Fackler
|
r43346 | knownnode = repo.changelog.hasnode # no nodemap until it is filtered | ||
r32672 | oldheads = sorted(h for h in remoteheads if knownnode(h)) | |||
Pierre-Yves David
|
r17209 | # all nodes in outgoing.missing are children of either: | ||
# - an element of oldheads | ||||
# - another element of outgoing.missing | ||||
# - nullrev | ||||
# This explains why the new head are very simple to compute. | ||||
Augie Fackler
|
r43347 | r = repo.set(b'heads(%ln + %ln)', oldheads, outgoing.missing) | ||
r32672 | newheads = sorted(c.node() for c in r) | |||
Pierre-Yves David
|
r22178 | # set some unsynced head to issue the "unsynced changes" warning | ||
Jordi Gutiérrez Hermoso
|
r24306 | if inc: | ||
r32671 | unsynced = [None] | |||
Jordi Gutiérrez Hermoso
|
r24306 | else: | ||
r32671 | unsynced = [] | |||
r32708 | return {None: (oldheads, newheads, unsynced, [])} | |||
Pierre-Yves David
|
r17209 | |||
Augie Fackler
|
r43346 | |||
Ryan McElroy
|
r26936 | def _nowarnheads(pushop): | ||
Ryan McElroy
|
r26862 | # Compute newly pushed bookmarks. We don't warn about bookmarked heads. | ||
Ryan McElroy
|
r26936 | repo = pushop.repo.unfiltered() | ||
remote = pushop.remote | ||||
Ryan McElroy
|
r26862 | localbookmarks = repo._bookmarks | ||
Gregory Szorc
|
r37655 | |||
with remote.commandexecutor() as e: | ||||
Augie Fackler
|
r43346 | remotebookmarks = e.callcommand( | ||
Augie Fackler
|
r46554 | b'listkeys', | ||
{ | ||||
b'namespace': b'bookmarks', | ||||
}, | ||||
Augie Fackler
|
r43346 | ).result() | ||
Gregory Szorc
|
r37655 | |||
Ryan McElroy
|
r26862 | bookmarkedheads = set() | ||
liscju
|
r29229 | |||
# internal config: bookmarks.pushing | ||||
Augie Fackler
|
r43346 | newbookmarks = [ | ||
localbookmarks.expandname(b) | ||||
Augie Fackler
|
r43347 | for b in pushop.ui.configlist(b'bookmarks', b'pushing') | ||
Augie Fackler
|
r43346 | ] | ||
liscju
|
r29229 | |||
Ryan McElroy
|
r26862 | for bm in localbookmarks: | ||
rnode = remotebookmarks.get(bm) | ||||
if rnode and rnode in repo: | ||||
Augie Fackler
|
r43248 | lctx, rctx = repo[localbookmarks[bm]], repo[rnode] | ||
Ryan McElroy
|
r26862 | if bookmarks.validdest(repo, rctx, lctx): | ||
bookmarkedheads.add(lctx.node()) | ||||
else: | ||||
if bm in newbookmarks and bm not in remotebookmarks: | ||||
Martin von Zweigbergk
|
r37469 | bookmarkedheads.add(localbookmarks[bm]) | ||
Ryan McElroy
|
r26862 | |||
return bookmarkedheads | ||||
Augie Fackler
|
r43346 | |||
Ryan McElroy
|
r26935 | def checkheads(pushop): | ||
Pierre-Yves David
|
r15932 | """Check that a push won't add any outgoing head | ||
Dirkjan Ochtman
|
r11301 | |||
Martin von Zweigbergk
|
r46735 | raise StateError error and display ui message as needed. | ||
Pierre-Yves David
|
r15932 | """ | ||
Ryan McElroy
|
r26935 | |||
repo = pushop.repo.unfiltered() | ||||
remote = pushop.remote | ||||
outgoing = pushop.outgoing | ||||
remoteheads = pushop.remoteheads | ||||
newbranch = pushop.newbranch | ||||
inc = bool(pushop.incoming) | ||||
Pierre-Yves David
|
r17209 | # Check for each named branch if we're creating new remote heads. | ||
# To be a remote head after push, node must be either: | ||||
# - unknown locally | ||||
# - a local outgoing head descended from update | ||||
# - a remote head that's known locally and not | ||||
# ancestral to an outgoing head | ||||
Joerg Sonnenberger
|
r47771 | if remoteheads == [repo.nullid]: | ||
Pierre-Yves David
|
r15932 | # remote is empty, nothing to check. | ||
return | ||||
Dirkjan Ochtman
|
r11301 | |||
Augie Fackler
|
r43347 | if remote.capable(b'branchmap'): | ||
r32706 | headssum = _headssummary(pushop) | |||
Pierre-Yves David
|
r17209 | else: | ||
Pierre-Yves David
|
r17211 | headssum = _oldheadssummary(repo, remoteheads, outgoing, inc) | ||
r32709 | pushop.pushbranchmap = headssum | |||
Augie Fackler
|
r43346 | newbranches = [ | ||
Gregory Szorc
|
r49768 | branch for branch, heads in headssum.items() if heads[0] is None | ||
Augie Fackler
|
r43346 | ] | ||
Pierre-Yves David
|
r17209 | # 1. Check for new branches on the remote. | ||
if newbranches and not newbranch: # new branch requires --new-branch | ||||
Augie Fackler
|
r43347 | branchnames = b', '.join(sorted(newbranches)) | ||
Pulkit Goyal
|
r42402 | # Calculate how many of the new branches are closed branches | ||
closedbranches = set() | ||||
for tag, heads, tip, isclosed in repo.branchmap().iterbranches(): | ||||
if isclosed: | ||||
closedbranches.add(tag) | ||||
Augie Fackler
|
r43346 | closedbranches = closedbranches & set(newbranches) | ||
Taapas Agrawal
|
r42389 | if closedbranches: | ||
Martin von Zweigbergk
|
r46520 | errmsg = _(b"push creates new remote branches: %s (%d closed)") % ( | ||
Augie Fackler
|
r43346 | branchnames, | ||
len(closedbranches), | ||||
) | ||||
Taapas Agrawal
|
r42389 | else: | ||
Martin von Zweigbergk
|
r46520 | errmsg = _(b"push creates new remote branches: %s") % branchnames | ||
Augie Fackler
|
r43347 | hint = _(b"use 'hg push --new-branch' to create new remote branches") | ||
Martin von Zweigbergk
|
r46735 | raise error.StateError(errmsg, hint=hint) | ||
Dirkjan Ochtman
|
r11301 | |||
Ryan McElroy
|
r26862 | # 2. Find heads that we need not warn about | ||
Ryan McElroy
|
r26936 | nowarnheads = _nowarnheads(pushop) | ||
Pierre-Yves David
|
r17212 | |||
# 3. Check for new heads. | ||||
Pierre-Yves David
|
r15932 | # If there are more heads after the push than before, a suitable | ||
# error message, depending on unsynced status, is displayed. | ||||
Pierre-Yves David
|
r26585 | errormsg = None | ||
Gregory Szorc
|
r43376 | for branch, heads in sorted(pycompat.iteritems(headssum)): | ||
r32708 | remoteheads, newheads, unsyncedheads, discardedheads = heads | |||
Pierre-Yves David
|
r17214 | # add unsynced data | ||
Mads Kiilerich
|
r20381 | if remoteheads is None: | ||
FUJIWARA Katsunori
|
r19840 | oldhs = set() | ||
else: | ||||
Mads Kiilerich
|
r20381 | oldhs = set(remoteheads) | ||
oldhs.update(unsyncedheads) | ||||
Augie Fackler
|
r43346 | dhs = None # delta heads, the new heads on branch | ||
r32675 | newhs = set(newheads) | |||
r32673 | newhs.update(unsyncedheads) | |||
r32674 | if unsyncedheads: | |||
if None in unsyncedheads: | ||||
Pierre-Yves David
|
r22178 | # old remote, no heads data | ||
heads = None | ||||
Mads Kiilerich
|
r21198 | else: | ||
Boris Feld
|
r35185 | heads = scmutil.nodesummaries(repo, unsyncedheads) | ||
Pierre-Yves David
|
r22178 | if heads is None: | ||
Augie Fackler
|
r43346 | repo.ui.status( | ||
Martin von Zweigbergk
|
r43387 | _(b"remote has heads that are not known locally\n") | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r22178 | elif branch is None: | ||
Augie Fackler
|
r43346 | repo.ui.status( | ||
Martin von Zweigbergk
|
r43387 | _(b"remote has heads that are not known locally: %s\n") | ||
Augie Fackler
|
r43346 | % heads | ||
) | ||||
Mads Kiilerich
|
r20501 | else: | ||
Augie Fackler
|
r43346 | repo.ui.status( | ||
_( | ||||
Augie Fackler
|
r43347 | b"remote has heads on branch '%s' that are " | ||
b"not known locally: %s\n" | ||||
Augie Fackler
|
r43346 | ) | ||
% (branch, heads) | ||||
) | ||||
Mads Kiilerich
|
r20381 | if remoteheads is None: | ||
if len(newhs) > 1: | ||||
FUJIWARA Katsunori
|
r19840 | dhs = list(newhs) | ||
Pierre-Yves David
|
r26585 | if errormsg is None: | ||
Augie Fackler
|
r43319 | errormsg = ( | ||
Augie Fackler
|
r43347 | _(b"push creates new branch '%s' with multiple heads") | ||
Augie Fackler
|
r43346 | % branch | ||
Augie Fackler
|
r43319 | ) | ||
Augie Fackler
|
r43346 | hint = _( | ||
Augie Fackler
|
r43347 | b"merge or" | ||
b" see 'hg help push' for details about" | ||||
b" pushing new heads" | ||||
Augie Fackler
|
r43346 | ) | ||
FUJIWARA Katsunori
|
r19840 | elif len(newhs) > len(oldhs): | ||
Mads Kiilerich
|
r20381 | # remove bookmarked or existing remote heads from the new heads list | ||
Ryan McElroy
|
r26862 | dhs = sorted(newhs - nowarnheads - oldhs) | ||
Levi Bard
|
r16835 | if dhs: | ||
Pierre-Yves David
|
r26585 | if errormsg is None: | ||
Augie Fackler
|
r43347 | if branch not in (b'default', None): | ||
Augie Fackler
|
r43346 | errormsg = _( | ||
Martin von Zweigbergk
|
r46520 | b"push creates new remote head %s on branch '%s'" | ||
Augie Fackler
|
r46554 | ) % ( | ||
short(dhs[0]), | ||||
branch, | ||||
) | ||||
Stephen Lee
|
r21580 | elif repo[dhs[0]].bookmarks(): | ||
Augie Fackler
|
r43346 | errormsg = _( | ||
Augie Fackler
|
r43347 | b"push creates new remote head %s " | ||
Martin von Zweigbergk
|
r46520 | b"with bookmark '%s'" | ||
Augie Fackler
|
r43346 | ) % (short(dhs[0]), repo[dhs[0]].bookmarks()[0]) | ||
Pierre-Yves David
|
r15932 | else: | ||
Martin von Zweigbergk
|
r46520 | errormsg = _(b"push creates new remote head %s") % short( | ||
Augie Fackler
|
r43346 | dhs[0] | ||
) | ||||
Mads Kiilerich
|
r20381 | if unsyncedheads: | ||
Augie Fackler
|
r43346 | hint = _( | ||
Augie Fackler
|
r43347 | b"pull and merge or" | ||
b" see 'hg help push' for details about" | ||||
b" pushing new heads" | ||||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r15932 | else: | ||
Augie Fackler
|
r43346 | hint = _( | ||
Augie Fackler
|
r43347 | b"merge or" | ||
b" see 'hg help push' for details about" | ||||
b" pushing new heads" | ||||
Augie Fackler
|
r43346 | ) | ||
Mads Kiilerich
|
r20051 | if branch is None: | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b"new remote heads:\n")) | ||
Mads Kiilerich
|
r20051 | else: | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b"new remote heads on branch '%s':\n") % branch) | ||
Pierre-Yves David
|
r15932 | for h in dhs: | ||
Augie Fackler
|
r43347 | repo.ui.note(b" %s\n" % short(h)) | ||
Pierre-Yves David
|
r26585 | if errormsg: | ||
Martin von Zweigbergk
|
r46735 | raise error.StateError(errormsg, hint=hint) | ||
Pierre-Yves David
|
r31586 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r31586 | def _postprocessobsolete(pushop, futurecommon, candidate_newhs): | ||
"""post process the list of new heads with obsolescence information | ||||
Pierre-Yves David
|
r32009 | Exists as a sub-function to contain the complexity and allow extensions to | ||
Pierre-Yves David
|
r31586 | experiment with smarter logic. | ||
Pierre-Yves David
|
r32009 | |||
Pierre-Yves David
|
r31586 | Returns (newheads, discarded_heads) tuple | ||
""" | ||||
Pierre-Yves David
|
r32009 | # known issue | ||
Pierre-Yves David
|
r31586 | # | ||
Pierre-Yves David
|
r32009 | # * We "silently" skip processing on all changeset unknown locally | ||
Pierre-Yves David
|
r31586 | # | ||
Pierre-Yves David
|
r32009 | # * if <nh> is public on the remote, it won't be affected by obsolete | ||
# marker and a new is created | ||||
# define various utilities and containers | ||||
Pierre-Yves David
|
r31586 | repo = pushop.repo | ||
Pierre-Yves David
|
r32009 | unfi = repo.unfiltered() | ||
r43960 | torev = unfi.changelog.index.get_rev | |||
Pierre-Yves David
|
r32009 | public = phases.public | ||
getphase = unfi._phasecache.phase | ||||
Augie Fackler
|
r43346 | ispublic = lambda r: getphase(unfi, r) == public | ||
ispushed = lambda n: torev(n) in futurecommon | ||||
r32789 | hasoutmarker = functools.partial(pushingmarkerfor, unfi.obsstore, ispushed) | |||
Pierre-Yves David
|
r32009 | successorsmarkers = unfi.obsstore.successors | ||
Augie Fackler
|
r43346 | newhs = set() # final set of new heads | ||
discarded = set() # new head of fully replaced branch | ||||
Pierre-Yves David
|
r32009 | |||
Augie Fackler
|
r43346 | localcandidate = set() # candidate heads known locally | ||
unknownheads = set() # candidate heads unknown locally | ||||
Pierre-Yves David
|
r32009 | for h in candidate_newhs: | ||
if h in unfi: | ||||
localcandidate.add(h) | ||||
else: | ||||
if successorsmarkers.get(h) is not None: | ||||
Augie Fackler
|
r43346 | msg = ( | ||
Augie Fackler
|
r43347 | b'checkheads: remote head unknown locally has' | ||
b' local marker: %s\n' | ||||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r32009 | repo.ui.debug(msg % hex(h)) | ||
unknownheads.add(h) | ||||
# fast path the simple case | ||||
if len(localcandidate) == 1: | ||||
return unknownheads | set(candidate_newhs), set() | ||||
r49537 | obsrevs = obsolete.getrevs(unfi, b'obsolete') | |||
futurenonobsolete = frozenset(futurecommon) - obsrevs | ||||
Pierre-Yves David
|
r32009 | # actually process branch replacement | ||
while localcandidate: | ||||
nh = localcandidate.pop() | ||||
r49537 | r = torev(nh) | |||
r44949 | current_branch = unfi[nh].branch() | |||
Pierre-Yves David
|
r32009 | # run this check early to skip the evaluation of the whole branch | ||
r49537 | if ispublic(r) or r not in obsrevs: | |||
Pierre-Yves David
|
r32009 | newhs.add(nh) | ||
continue | ||||
# Get all revs/nodes on the branch exclusive to this head | ||||
# (already filtered heads are "ignored")) | ||||
Augie Fackler
|
r43347 | branchrevs = unfi.revs( | ||
b'only(%n, (%ln+%ln))', nh, localcandidate, newhs | ||||
) | ||||
r44949 | ||||
branchnodes = [] | ||||
for r in branchrevs: | ||||
c = unfi[r] | ||||
if c.branch() == current_branch: | ||||
branchnodes.append(c.node()) | ||||
Pierre-Yves David
|
r32009 | |||
# The branch won't be hidden on the remote if | ||||
# * any part of it is public, | ||||
# * any part of it is considered part of the result by previous logic, | ||||
# * if we have no markers to push to obsolete it. | ||||
Augie Fackler
|
r43346 | if ( | ||
any(ispublic(r) for r in branchrevs) | ||||
r49537 | or any(torev(n) in futurenonobsolete for n in branchnodes) | |||
Augie Fackler
|
r43346 | or any(not hasoutmarker(n) for n in branchnodes) | ||
): | ||||
Pierre-Yves David
|
r31586 | newhs.add(nh) | ||
else: | ||||
Pierre-Yves David
|
r32009 | # note: there is a corner case if there is a merge in the branch. | ||
# we might end up with -more- heads. However, these heads are not | ||||
# "added" by the push, but more by the "removal" on the remote so I | ||||
# think is a okay to ignore them, | ||||
discarded.add(nh) | ||||
newhs |= unknownheads | ||||
Pierre-Yves David
|
r31586 | return newhs, discarded | ||
Pierre-Yves David
|
r32009 | |||
Augie Fackler
|
r43346 | |||
r32789 | def pushingmarkerfor(obsstore, ispushed, node): | |||
Pierre-Yves David
|
r32009 | """true if some markers are to be pushed for node | ||
We cannot just look in to the pushed obsmarkers from the pushop because | ||||
discovery might have filtered relevant markers. In addition listing all | ||||
markers relevant to all changesets in the pushed set would be too expensive | ||||
(O(len(repo))) | ||||
(note: There are cache opportunity in this function. but it would requires | ||||
a two dimensional stack.) | ||||
""" | ||||
successorsmarkers = obsstore.successors | ||||
stack = [node] | ||||
seen = set(stack) | ||||
while stack: | ||||
current = stack.pop() | ||||
r32789 | if ispushed(current): | |||
Pierre-Yves David
|
r32009 | return True | ||
markers = successorsmarkers.get(current, ()) | ||||
# markers fields = ('prec', 'succs', 'flag', 'meta', 'date', 'parents') | ||||
for m in markers: | ||||
Augie Fackler
|
r43346 | nexts = m[1] # successors | ||
if not nexts: # this is a prune marker | ||||
nexts = m[5] or () # parents | ||||
Pierre-Yves David
|
r32009 | for n in nexts: | ||
if n not in seen: | ||||
seen.add(n) | ||||
stack.append(n) | ||||
return False | ||||