destutil.py
496 lines
| 17.1 KiB
| text/x-python
|
PythonLexer
/ mercurial / destutil.py
Pierre-Yves David
|
r26569 | # destutil.py - Mercurial utility function for command destination | ||
# | ||||
Raphaël Gomès
|
r47575 | # Copyright Olivia Mackall <olivia@selenic.com> and other | ||
Pierre-Yves David
|
r26569 | # | ||
# 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
|
r27333 | |||
Pierre-Yves David
|
r26569 | from .i18n import _ | ||
Augie Fackler
|
r43346 | from . import bookmarks, error, obsutil, scmutil, stack | ||
Pierre-Yves David
|
r26569 | |||
Augie Fackler
|
r37805 | def orphanpossibledestination(repo, rev): | ||
"""Return all changesets that may be a new parent for orphan `rev`. | ||||
This function works fine on non-orphan revisions, it's just silly | ||||
because there's no destination implied by obsolete markers, so | ||||
it'll return nothing. | ||||
""" | ||||
tonode = repo.changelog.node | ||||
parents = repo.changelog.parentrevs | ||||
torev = repo.changelog.rev | ||||
dest = set() | ||||
tovisit = list(parents(rev)) | ||||
while tovisit: | ||||
r = tovisit.pop() | ||||
succsets = obsutil.successorssets(repo, tonode(r)) | ||||
if not succsets: | ||||
# if there are no successors for r, r was probably pruned | ||||
# and we should walk up to r's parents to try and find | ||||
# some successors. | ||||
tovisit.extend(parents(r)) | ||||
else: | ||||
# We should probably pick only one destination from split | ||||
# (case where '1 < len(ss)'), This could be the currently | ||||
# tipmost, but the correct result is less clear when | ||||
# results of the split have been moved such that they | ||||
# reside on multiple branches. | ||||
for ss in succsets: | ||||
for n in ss: | ||||
dr = torev(n) | ||||
if dr != -1: | ||||
dest.add(dr) | ||||
return dest | ||||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r30962 | def _destupdateobs(repo, clean): | ||
Pierre-Yves David
|
r26723 | """decide of an update destination from obsolescence markers""" | ||
Pierre-Yves David
|
r26569 | node = None | ||
wc = repo[None] | ||||
p1 = wc.p1() | ||||
Pierre-Yves David
|
r26723 | movemark = None | ||
Pierre-Yves David
|
r26569 | |||
if p1.obsolete() and not p1.children(): | ||||
# allow updating to successors | ||||
r33143 | successors = obsutil.successorssets(repo, p1.node()) | |||
Pierre-Yves David
|
r26569 | |||
# behavior of certain cases is as follows, | ||||
# | ||||
# divergent changesets: update to highest rev, similar to what | ||||
# is currently done when there are more than one head | ||||
# (i.e. 'tip') | ||||
# | ||||
# replaced changesets: same as divergent except we know there | ||||
# is no conflict | ||||
# | ||||
r49538 | # pruned changeset: update to the closest non-obsolete ancestor, | |||
# similar to what 'hg prune' currently does | ||||
Pierre-Yves David
|
r26569 | |||
if successors: | ||||
# flatten the list here handles both divergent (len > 1) | ||||
# and the usual case (len = 1) | ||||
successors = [n for sub in successors for n in sub] | ||||
# get the max revision for the given successors set, | ||||
# i.e. the 'tip' of a set | ||||
Augie Fackler
|
r43347 | node = repo.revs(b'max(%ln)', successors).first() | ||
r49536 | else: | |||
r49538 | p1 = p1.p1() | |||
while p1.obsolete(): | ||||
p1 = p1.p1() | ||||
node = p1.node() | ||||
if node is not None and bookmarks.isactivewdirparent(repo): | ||||
movemark = repo[b'.'].node() | ||||
Pierre-Yves David
|
r26723 | return node, movemark, None | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r30962 | def _destupdatebook(repo, clean): | ||
Pierre-Yves David
|
r26724 | """decide on an update destination from active bookmark""" | ||
# we also move the active bookmark, if any | ||||
Martin von Zweigbergk
|
r37377 | node = None | ||
Martin von Zweigbergk
|
r37393 | activemark, movemark = bookmarks.calculateupdate(repo.ui, repo) | ||
Martin von Zweigbergk
|
r37377 | if activemark is not None: | ||
Martin von Zweigbergk
|
r37470 | node = repo._bookmarks[activemark] | ||
Pierre-Yves David
|
r26724 | return node, movemark, activemark | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r30962 | def _destupdatebranch(repo, clean): | ||
FUJIWARA Katsunori
|
r28385 | """decide on an update destination from current branch | ||
This ignores closed branch heads. | ||||
""" | ||||
Pierre-Yves David
|
r26725 | wc = repo[None] | ||
movemark = node = None | ||||
FUJIWARA Katsunori
|
r28235 | currentbranch = wc.branch() | ||
liscju
|
r29284 | |||
if clean: | ||||
Augie Fackler
|
r43347 | currentbranch = repo[b'.'].branch() | ||
liscju
|
r29284 | |||
FUJIWARA Katsunori
|
r28236 | if currentbranch in repo.branchmap(): | ||
FUJIWARA Katsunori
|
r28385 | heads = repo.branchheads(currentbranch) | ||
FUJIWARA Katsunori
|
r28236 | if heads: | ||
Augie Fackler
|
r43347 | node = repo.revs(b'max(.::(%ln))', heads).first() | ||
Pierre-Yves David
|
r26725 | if bookmarks.isactivewdirparent(repo): | ||
Augie Fackler
|
r43347 | movemark = repo[b'.'].node() | ||
elif currentbranch == b'default' and not wc.p1(): | ||||
Yuya Nishihara
|
r28924 | # "null" parent belongs to "default" branch, but it doesn't exist, so | ||
# update to the tipmost non-closed branch head | ||||
Augie Fackler
|
r43347 | node = repo.revs(b'max(head() and not closed())').first() | ||
FUJIWARA Katsunori
|
r28236 | else: | ||
Augie Fackler
|
r43347 | node = repo[b'.'].node() | ||
Pierre-Yves David
|
r26725 | return node, movemark, None | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r30962 | def _destupdatebranchfallback(repo, clean): | ||
FUJIWARA Katsunori
|
r28385 | """decide on an update destination from closed heads in current branch""" | ||
wc = repo[None] | ||||
currentbranch = wc.branch() | ||||
movemark = None | ||||
if currentbranch in repo.branchmap(): | ||||
# here, all descendant branch heads are closed | ||||
heads = repo.branchheads(currentbranch, closed=True) | ||||
Augie Fackler
|
r43347 | assert heads, b"any branch has at least one head" | ||
node = repo.revs(b'max(.::(%ln))', heads).first() | ||||
Martin von Zweigbergk
|
r43387 | assert ( | ||
node is not None | ||||
), b"any revision has at least one descendant branch head" | ||||
FUJIWARA Katsunori
|
r28385 | if bookmarks.isactivewdirparent(repo): | ||
Augie Fackler
|
r43347 | movemark = repo[b'.'].node() | ||
FUJIWARA Katsunori
|
r28385 | else: | ||
# here, no "default" branch, and all branches are closed | ||||
Augie Fackler
|
r43347 | node = repo.lookup(b'tip') | ||
assert node is not None, b"'tip' exists even in empty repository" | ||||
FUJIWARA Katsunori
|
r28385 | return node, movemark, None | ||
Augie Fackler
|
r43346 | |||
Mads Kiilerich
|
r30332 | # order in which each step should be evaluated | ||
Pierre-Yves David
|
r26726 | # steps are run until one finds a destination | ||
Augie Fackler
|
r43347 | destupdatesteps = [b'evolution', b'bookmark', b'branch', b'branchfallback'] | ||
Pierre-Yves David
|
r26726 | # mapping to ease extension overriding steps. | ||
Augie Fackler
|
r43346 | destupdatestepmap = { | ||
Augie Fackler
|
r43347 | b'evolution': _destupdateobs, | ||
b'bookmark': _destupdatebook, | ||||
b'branch': _destupdatebranch, | ||||
b'branchfallback': _destupdatebranchfallback, | ||||
Augie Fackler
|
r43346 | } | ||
Pierre-Yves David
|
r26726 | |||
Martin von Zweigbergk
|
r30962 | def destupdate(repo, clean=False): | ||
Pierre-Yves David
|
r26723 | """destination for bare update operation | ||
return (rev, movemark, activemark) | ||||
- rev: the revision to update to, | ||||
- movemark: node to move the active bookmark from | ||||
(cf bookmark.calculate update), | ||||
- activemark: a bookmark to activate at the end of the update. | ||||
""" | ||||
Pierre-Yves David
|
r26726 | node = movemark = activemark = None | ||
Pierre-Yves David
|
r26723 | |||
Pierre-Yves David
|
r26726 | for step in destupdatesteps: | ||
Martin von Zweigbergk
|
r30962 | node, movemark, activemark = destupdatestepmap[step](repo, clean) | ||
Pierre-Yves David
|
r26726 | if node is not None: | ||
break | ||||
Pierre-Yves David
|
r26628 | rev = repo[node].rev() | ||
Pierre-Yves David
|
r26641 | return rev, movemark, activemark | ||
Pierre-Yves David
|
r26714 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r28102 | msgdestmerge = { | ||
# too many matching divergent bookmark | ||||
Augie Fackler
|
r43347 | b'toomanybookmarks': { | ||
b'merge': ( | ||||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b"multiple matching bookmarks to merge -" | ||
b" please merge with an explicit rev or bookmark" | ||||
Augie Fackler
|
r43346 | ), | ||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads' to see all heads, specify rev with -r"), | ||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | b'rebase': ( | ||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b"multiple matching bookmarks to rebase -" | ||
b" please rebase to an explicit rev or bookmark" | ||||
Augie Fackler
|
r43346 | ), | ||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads' to see all heads, specify destination with -d"), | ||
Augie Fackler
|
r43346 | ), | ||
}, | ||||
Pierre-Yves David
|
r28102 | # no other matching divergent bookmark | ||
Augie Fackler
|
r43347 | b'nootherbookmarks': { | ||
b'merge': ( | ||||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b"no matching bookmark to merge - " | ||
b"please merge with an explicit rev or bookmark" | ||||
Augie Fackler
|
r43346 | ), | ||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads' to see all heads, specify rev with -r"), | ||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | b'rebase': ( | ||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b"no matching bookmark to rebase - " | ||
b"please rebase to an explicit rev or bookmark" | ||||
Augie Fackler
|
r43346 | ), | ||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads' to see all heads, specify destination with -d"), | ||
Augie Fackler
|
r43346 | ), | ||
}, | ||||
Pierre-Yves David
|
r28102 | # branch have too many unbookmarked heads, no obvious destination | ||
Augie Fackler
|
r43347 | b'toomanyheads': { | ||
b'merge': ( | ||||
_(b"branch '%s' has %d heads - please merge with an explicit rev"), | ||||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads .' to see heads, specify rev with -r"), | ||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | b'rebase': ( | ||
_(b"branch '%s' has %d heads - please rebase to an explicit rev"), | ||||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads .' to see heads, specify destination with -d"), | ||
Augie Fackler
|
r43346 | ), | ||
}, | ||||
Pierre-Yves David
|
r28102 | # branch have no other unbookmarked heads | ||
Augie Fackler
|
r43347 | b'bookmarkedheads': { | ||
b'merge': ( | ||||
_(b"heads are bookmarked - please merge with an explicit rev"), | ||||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads' to see all heads, specify rev with -r"), | ||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | b'rebase': ( | ||
_(b"heads are bookmarked - please rebase to an explicit rev"), | ||||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads' to see all heads, specify destination with -d"), | ||
Augie Fackler
|
r43346 | ), | ||
}, | ||||
Pierre-Yves David
|
r28102 | # branch have just a single heads, but there is other branches | ||
Augie Fackler
|
r43347 | b'nootherbranchheads': { | ||
b'merge': ( | ||||
_(b"branch '%s' has one head - please merge with an explicit rev"), | ||||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads' to see all heads, specify rev with -r"), | ||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | b'rebase': ( | ||
_(b"branch '%s' has one head - please rebase to an explicit rev"), | ||||
Kyle Lippincott
|
r43384 | _(b"run 'hg heads' to see all heads, specify destination with -d"), | ||
Augie Fackler
|
r43346 | ), | ||
}, | ||||
Pierre-Yves David
|
r28102 | # repository have a single head | ||
Augie Fackler
|
r43347 | b'nootherheads': { | ||
b'merge': (_(b'nothing to merge'), None), | ||||
b'rebase': (_(b'nothing to rebase'), None), | ||||
Augie Fackler
|
r43346 | }, | ||
Pierre-Yves David
|
r28102 | # repository have a single head and we are not on it | ||
Augie Fackler
|
r43347 | b'nootherheadsbehind': { | ||
b'merge': (_(b'nothing to merge'), _(b"use 'hg update' instead")), | ||||
b'rebase': (_(b'nothing to rebase'), _(b"use 'hg update' instead")), | ||||
Augie Fackler
|
r43346 | }, | ||
Pierre-Yves David
|
r28102 | # We are not on a head | ||
Augie Fackler
|
r43347 | b'notatheads': { | ||
b'merge': ( | ||||
_(b'working directory not at a head revision'), | ||||
_(b"use 'hg update' or merge with an explicit revision"), | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | b'rebase': ( | ||
_(b'working directory not at a head revision'), | ||||
_(b"use 'hg update' or rebase to an explicit revision"), | ||||
Augie Fackler
|
r43346 | ), | ||
}, | ||||
Augie Fackler
|
r43347 | b'emptysourceset': { | ||
b'merge': (_(b'source set is empty'), None), | ||||
b'rebase': (_(b'source set is empty'), None), | ||||
Augie Fackler
|
r43346 | }, | ||
Augie Fackler
|
r43347 | b'multiplebranchessourceset': { | ||
b'merge': (_(b'source set is rooted in multiple branches'), None), | ||||
b'rebase': ( | ||||
_(b'rebaseset is rooted in multiple named branches'), | ||||
_(b'specify an explicit destination with --dest'), | ||||
Augie Fackler
|
r43346 | ), | ||
}, | ||||
} | ||||
Pierre-Yves David
|
r28102 | |||
Augie Fackler
|
r43347 | def _destmergebook(repo, action=b'merge', sourceset=None, destspace=None): | ||
Pierre-Yves David
|
r26727 | """find merge destination in the active bookmark case""" | ||
node = None | ||||
Augie Fackler
|
r32381 | bmheads = bookmarks.headsforactive(repo) | ||
Martin von Zweigbergk
|
r37470 | curhead = repo._bookmarks[repo._activebookmark] | ||
Pierre-Yves David
|
r26727 | if len(bmheads) == 2: | ||
if curhead == bmheads[0]: | ||||
node = bmheads[1] | ||||
else: | ||||
node = bmheads[0] | ||||
elif len(bmheads) > 2: | ||||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'toomanybookmarks'][action] | ||
Pierre-Yves David
|
r28141 | raise error.ManyMergeDestAbort(msg, hint=hint) | ||
Pierre-Yves David
|
r26727 | elif len(bmheads) <= 1: | ||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'nootherbookmarks'][action] | ||
Pierre-Yves David
|
r28141 | raise error.NoMergeDestAbort(msg, hint=hint) | ||
Pierre-Yves David
|
r26727 | assert node is not None | ||
return node | ||||
Augie Fackler
|
r43346 | |||
def _destmergebranch( | ||||
Augie Fackler
|
r43347 | repo, action=b'merge', sourceset=None, onheadcheck=True, destspace=None | ||
Augie Fackler
|
r43346 | ): | ||
Pierre-Yves David
|
r26728 | """find merge destination based on branch heads""" | ||
node = None | ||||
Pierre-Yves David
|
r28139 | |||
if sourceset is None: | ||||
sourceset = [repo[repo.dirstate.p1()].rev()] | ||||
branch = repo.dirstate.branch() | ||||
elif not sourceset: | ||||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'emptysourceset'][action] | ||
Pierre-Yves David
|
r28141 | raise error.NoMergeDestAbort(msg, hint=hint) | ||
Pierre-Yves David
|
r28139 | else: | ||
branch = None | ||||
Augie Fackler
|
r43347 | for ctx in repo.set(b'roots(%ld::%ld)', sourceset, sourceset): | ||
Pierre-Yves David
|
r28139 | if branch is not None and ctx.branch() != branch: | ||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'multiplebranchessourceset'][action] | ||
Pierre-Yves David
|
r28141 | raise error.ManyMergeDestAbort(msg, hint=hint) | ||
Pierre-Yves David
|
r28139 | branch = ctx.branch() | ||
Pierre-Yves David
|
r26728 | bheads = repo.branchheads(branch) | ||
Augie Fackler
|
r43347 | onhead = repo.revs(b'%ld and %ln', sourceset, bheads) | ||
Pierre-Yves David
|
r28161 | if onheadcheck and not onhead: | ||
Pierre-Yves David
|
r28139 | # Case A: working copy if not on a head. (merge only) | ||
Pierre-Yves David
|
r28105 | # | ||
# This is probably a user mistake We bailout pointing at 'hg update' | ||||
Pierre-Yves David
|
r28103 | if len(repo.heads()) <= 1: | ||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'nootherheadsbehind'][action] | ||
Pierre-Yves David
|
r28103 | else: | ||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'notatheads'][action] | ||
Pierre-Yves David
|
r28103 | raise error.Abort(msg, hint=hint) | ||
Pierre-Yves David
|
r28139 | # remove heads descendants of source from the set | ||
Augie Fackler
|
r43347 | bheads = list(repo.revs(b'%ln - (%ld::)', bheads, sourceset)) | ||
Pierre-Yves David
|
r28138 | # filters out bookmarked heads | ||
Augie Fackler
|
r43347 | nbhs = list(repo.revs(b'%ld - bookmark()', bheads)) | ||
Pierre-Yves David
|
r29043 | |||
if destspace is not None: | ||||
# restrict search space | ||||
# used in the 'hg pull --rebase' case, see issue 5214. | ||||
Augie Fackler
|
r43347 | nbhs = list(repo.revs(b'%ld and %ld', destspace, nbhs)) | ||
Pierre-Yves David
|
r29043 | |||
Pierre-Yves David
|
r28138 | if len(nbhs) > 1: | ||
# Case B: There is more than 1 other anonymous heads | ||||
Pierre-Yves David
|
r28105 | # | ||
# This means that there will be more than 1 candidate. This is | ||||
# ambiguous. We abort asking the user to pick as explicit destination | ||||
# instead. | ||||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'toomanyheads'][action] | ||
Pierre-Yves David
|
r28138 | msg %= (branch, len(bheads) + 1) | ||
Pierre-Yves David
|
r28141 | raise error.ManyMergeDestAbort(msg, hint=hint) | ||
Pierre-Yves David
|
r28138 | elif not nbhs: | ||
# Case B: There is no other anonymous heads | ||||
Pierre-Yves David
|
r28105 | # | ||
# This means that there is no natural candidate to merge with. | ||||
# We abort, with various messages for various cases. | ||||
Pierre-Yves David
|
r28138 | if bheads: | ||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'bookmarkedheads'][action] | ||
Pierre-Yves David
|
r28102 | elif len(repo.heads()) > 1: | ||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'nootherbranchheads'][action] | ||
Pierre-Yves David
|
r28102 | msg %= branch | ||
Pierre-Yves David
|
r28161 | elif not onhead: | ||
# if 'onheadcheck == False' (rebase case), | ||||
# this was not caught in Case A. | ||||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'nootherheadsbehind'][action] | ||
Pierre-Yves David
|
r28102 | else: | ||
Augie Fackler
|
r43347 | msg, hint = msgdestmerge[b'nootherheads'][action] | ||
Pierre-Yves David
|
r28141 | raise error.NoMergeDestAbort(msg, hint=hint) | ||
Pierre-Yves David
|
r26728 | else: | ||
node = nbhs[0] | ||||
assert node is not None | ||||
return node | ||||
Augie Fackler
|
r43346 | |||
def destmerge( | ||||
Augie Fackler
|
r43347 | repo, action=b'merge', sourceset=None, onheadcheck=True, destspace=None | ||
Augie Fackler
|
r43346 | ): | ||
Pierre-Yves David
|
r28137 | """return the default destination for a merge | ||
(or raise exception about why it can't pick one) | ||||
:action: the action being performed, controls emitted error message | ||||
""" | ||||
Pierre-Yves David
|
r29043 | # destspace is here to work around issues with `hg pull --rebase` see | ||
# issue5214 for details | ||||
Pierre-Yves David
|
r26714 | if repo._activebookmark: | ||
Augie Fackler
|
r43346 | node = _destmergebook( | ||
repo, action=action, sourceset=sourceset, destspace=destspace | ||||
) | ||||
Pierre-Yves David
|
r26714 | else: | ||
Augie Fackler
|
r43346 | node = _destmergebranch( | ||
repo, | ||||
action=action, | ||||
sourceset=sourceset, | ||||
onheadcheck=onheadcheck, | ||||
destspace=destspace, | ||||
) | ||||
Pierre-Yves David
|
r26714 | return repo[node].rev() | ||
Gregory Szorc
|
r27262 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r27262 | def desthistedit(ui, repo): | ||
"""Default base revision to edit for `hg histedit`.""" | ||||
Augie Fackler
|
r43347 | default = ui.config(b'histedit', b'defaultrev') | ||
Boris Feld
|
r37021 | |||
if default is None: | ||||
revs = stack.getstack(repo) | ||||
elif default: | ||||
Gregory Szorc
|
r27559 | revs = scmutil.revrange(repo, [default]) | ||
rdamazio@google.com
|
r41202 | else: | ||
Martin von Zweigbergk
|
r46447 | raise error.ConfigError( | ||
Augie Fackler
|
r43347 | _(b"config option histedit.defaultrev can't be empty") | ||
) | ||||
Boris Feld
|
r37021 | |||
if revs: | ||||
Boris Feld
|
r37408 | # Take the first revision of the revset as the root | ||
return revs.min() | ||||
Gregory Szorc
|
r27262 | |||
return None | ||||
Pierre-Yves David
|
r28029 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r33194 | def stackbase(ui, repo): | ||
Boris Feld
|
r37020 | revs = stack.getstack(repo) | ||
Boris Feld
|
r37022 | return revs.first() if revs else None | ||
Gregory Szorc
|
r33194 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r28029 | def _statusotherbook(ui, repo): | ||
Augie Fackler
|
r32381 | bmheads = bookmarks.headsforactive(repo) | ||
Martin von Zweigbergk
|
r37470 | curhead = repo._bookmarks[repo._activebookmark] | ||
Augie Fackler
|
r43347 | if repo.revs(b'%n and parents()', curhead): | ||
Pierre-Yves David
|
r28029 | # we are on the active bookmark | ||
bmheads = [b for b in bmheads if curhead != b] | ||||
if bmheads: | ||||
Augie Fackler
|
r43347 | msg = _(b'%i other divergent bookmarks for "%s"\n') | ||
Pierre-Yves David
|
r28029 | ui.status(msg % (len(bmheads), repo._activebookmark)) | ||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r28029 | def _statusotherbranchheads(ui, repo): | ||
currentbranch = repo.dirstate.branch() | ||||
FUJIWARA Katsunori
|
r28266 | allheads = repo.branchheads(currentbranch, closed=True) | ||
Pierre-Yves David
|
r28029 | heads = repo.branchheads(currentbranch) | ||
Augie Fackler
|
r43347 | if repo.revs(b'%ln and parents()', allheads): | ||
FUJIWARA Katsunori
|
r28266 | # we are on a head, even though it might be closed | ||
FUJIWARA Katsunori
|
r28385 | # | ||
# on closed otherheads | ||||
# ========= ========== | ||||
# o 0 all heads for current branch are closed | ||||
# N only descendant branch heads are closed | ||||
# x 0 there is only one non-closed branch head | ||||
# N there are some non-closed branch heads | ||||
# ========= ========== | ||||
Augie Fackler
|
r43347 | otherheads = repo.revs(b'%ln - parents()', heads) | ||
if repo[b'.'].closesbranch(): | ||||
Augie Fackler
|
r43346 | ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b'no open descendant heads on branch "%s", ' | ||
b'updating to a closed head\n' | ||||
Augie Fackler
|
r43346 | ) | ||
% currentbranch | ||||
) | ||||
FUJIWARA Katsunori
|
r28385 | if otherheads: | ||
Augie Fackler
|
r43346 | ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b"(committing will reopen the head, " | ||
b"use 'hg heads .' to see %i other heads)\n" | ||||
Augie Fackler
|
r43346 | ) | ||
% (len(otherheads)) | ||||
) | ||||
FUJIWARA Katsunori
|
r28385 | else: | ||
Augie Fackler
|
r43346 | ui.warn( | ||
Augie Fackler
|
r43347 | _(b'(committing will reopen branch "%s")\n') % currentbranch | ||
Augie Fackler
|
r43346 | ) | ||
FUJIWARA Katsunori
|
r28385 | elif otherheads: | ||
Augie Fackler
|
r43347 | curhead = repo[b'.'] | ||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b'updated to "%s: %s"\n') | ||
% (curhead, curhead.description().split(b'\n')[0]) | ||||
Augie Fackler
|
r43346 | ) | ||
ui.status( | ||||
Augie Fackler
|
r43347 | _(b'%i other heads for branch "%s"\n') | ||
Augie Fackler
|
r43346 | % (len(otherheads), currentbranch) | ||
) | ||||
Pierre-Yves David
|
r28029 | |||
def statusotherdests(ui, repo): | ||||
"""Print message about other head""" | ||||
# XXX we should probably include a hint: | ||||
# - about what to do | ||||
# - how to see such heads | ||||
if repo._activebookmark: | ||||
_statusotherbook(ui, repo) | ||||
else: | ||||
_statusotherbranchheads(ui, repo) | ||||