strip.py
292 lines
| 9.2 KiB
| text/x-python
|
PythonLexer
/ hgext / strip.py
Mads Kiilerich
|
r23139 | """strip changesets and their descendants from history | ||
Pierre-Yves David
|
r19826 | |||
Javi Merino
|
r19945 | This extension allows you to strip changesets and all their descendants from the | ||
Pierre-Yves David
|
r19826 | repository. See the command help for details. | ||
""" | ||||
timeless
|
r28377 | from __future__ import absolute_import | ||
Yuya Nishihara
|
r29205 | from mercurial.i18n import _ | ||
Gregory Szorc
|
r43359 | from mercurial.pycompat import getattr | ||
timeless
|
r28377 | from mercurial import ( | ||
bookmarks as bookmarksmod, | ||||
cmdutil, | ||||
error, | ||||
hg, | ||||
lock as lockmod, | ||||
merge, | ||||
node as nodemod, | ||||
Pulkit Goyal
|
r32897 | pycompat, | ||
Yuya Nishihara
|
r32337 | registrar, | ||
timeless
|
r28377 | repair, | ||
scmutil, | ||||
util, | ||||
) | ||||
Augie Fackler
|
r43346 | |||
timeless
|
r28377 | nullid = nodemod.nullid | ||
release = lockmod.release | ||||
Pierre-Yves David
|
r19822 | |||
cmdtable = {} | ||||
Yuya Nishihara
|
r32337 | command = registrar.command(cmdtable) | ||
Augie Fackler
|
r29841 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
Augie Fackler
|
r25186 | # extensions which SHIP WITH MERCURIAL. Non-mainline extensions should | ||
# be specifying the version(s) of Mercurial they are tested with, or | ||||
# leave the attribute unspecified. | ||||
Augie Fackler
|
r43347 | testedwith = b'ships-with-hg-core' | ||
Pierre-Yves David
|
r19823 | |||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r42690 | def checklocalchanges(repo, force=False): | ||
Martin von Zweigbergk
|
r22925 | s = repo.status() | ||
Pierre-Yves David
|
r19824 | if not force: | ||
Taapas Agrawal
|
r42732 | cmdutil.checkunfinished(repo) | ||
Martin von Zweigbergk
|
r42691 | cmdutil.bailifchanged(repo) | ||
Taapas Agrawal
|
r42732 | else: | ||
cmdutil.checkunfinished(repo, skipmerge=True) | ||||
Martin von Zweigbergk
|
r22925 | return s | ||
Pierre-Yves David
|
r19824 | |||
Augie Fackler
|
r43346 | |||
Paul Morelle
|
r34575 | def _findupdatetarget(repo, nodes): | ||
unode, p2 = repo.changelog.parents(nodes[0]) | ||||
Paul Morelle
|
r34622 | currentbranch = repo[None].branch() | ||
Paul Morelle
|
r34575 | |||
Augie Fackler
|
r43346 | if ( | ||
Augie Fackler
|
r43347 | util.safehasattr(repo, b'mq') | ||
Augie Fackler
|
r43346 | and p2 != nullid | ||
and p2 in [x.node for x in repo.mq.applied] | ||||
): | ||||
Paul Morelle
|
r34575 | unode = p2 | ||
Paul Morelle
|
r34622 | elif currentbranch != repo[unode].branch(): | ||
Augie Fackler
|
r43347 | pwdir = b'parents(wdir())' | ||
revset = b'max(((parents(%ln::%r) + %r) - %ln::%r) and branch(%s))' | ||||
Augie Fackler
|
r43346 | branchtarget = repo.revs( | ||
revset, nodes, pwdir, pwdir, nodes, pwdir, currentbranch | ||||
) | ||||
Paul Morelle
|
r34622 | if branchtarget: | ||
cl = repo.changelog | ||||
unode = cl.node(branchtarget.first()) | ||||
Paul Morelle
|
r34575 | |||
return unode | ||||
Augie Fackler
|
r43346 | |||
def strip( | ||||
ui, | ||||
repo, | ||||
revs, | ||||
update=True, | ||||
backup=True, | ||||
force=None, | ||||
bookmarks=None, | ||||
soft=False, | ||||
): | ||||
Martin von Zweigbergk
|
r32919 | with repo.wlock(), repo.lock(): | ||
Pierre-Yves David
|
r19825 | |||
if update: | ||||
checklocalchanges(repo, force=force) | ||||
Paul Morelle
|
r34575 | urev = _findupdatetarget(repo, revs) | ||
Pierre-Yves David
|
r19825 | hg.clean(repo, urev) | ||
FUJIWARA Katsunori
|
r26748 | repo.dirstate.write(repo.currenttransaction()) | ||
Pierre-Yves David
|
r19825 | |||
Boris Feld
|
r41960 | if soft: | ||
repair.softstrip(ui, repo, revs, backup) | ||||
else: | ||||
repair.strip(ui, repo, revs, backup) | ||||
David Soria Parra
|
r21847 | |||
Shubhanshu Agrawal
|
r26972 | repomarks = repo._bookmarks | ||
Shubhanshu Agrawal
|
r27029 | if bookmarks: | ||
Augie Fackler
|
r43347 | with repo.transaction(b'strip') as tr: | ||
Laurent Charignon
|
r27052 | if repo._activebookmark in bookmarks: | ||
bookmarksmod.deactivate(repo) | ||||
Boris Feld
|
r33488 | repomarks.applychanges(repo, tr, [(b, None) for b in bookmarks]) | ||
Bryan O'Sullivan
|
r27872 | for bookmark in sorted(bookmarks): | ||
Augie Fackler
|
r43347 | ui.write(_(b"bookmark '%s' deleted\n") % bookmark) | ||
Pierre-Yves David
|
r19826 | |||
Augie Fackler
|
r43346 | |||
@command( | ||||
Augie Fackler
|
r43347 | b"strip", | ||
Augie Fackler
|
r43346 | [ | ||
( | ||||
Augie Fackler
|
r43347 | b'r', | ||
b'rev', | ||||
Augie Fackler
|
r43346 | [], | ||
_( | ||||
Augie Fackler
|
r43347 | b'strip specified revision (optional, ' | ||
b'can specify revisions without this ' | ||||
b'option)' | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | _(b'REV'), | ||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'f', | ||
b'force', | ||||
Augie Fackler
|
r43346 | None, | ||
_( | ||||
Augie Fackler
|
r43347 | b'force removal of changesets, discard ' | ||
b'uncommitted changes (no backup)' | ||||
Augie Fackler
|
r43346 | ), | ||
), | ||||
Augie Fackler
|
r43347 | (b'', b'no-backup', None, _(b'do not save backup bundle')), | ||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'', | ||
b'nobackup', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b'do not save backup bundle ' b'(DEPRECATED)'), | ||
), | ||||
(b'n', b'', None, _(b'ignored (DEPRECATED)')), | ||||
( | ||||
b'k', | ||||
b'keep', | ||||
None, | ||||
_(b"do not modify working directory during " b"strip"), | ||||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'B', | ||
b'bookmark', | ||||
Augie Fackler
|
r43346 | [], | ||
Augie Fackler
|
r43347 | _(b"remove revs only reachable from given" b" bookmark"), | ||
_(b'BOOKMARK'), | ||||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'', | ||
b'soft', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b"simply drop changesets from visible history (EXPERIMENTAL)"), | ||
Augie Fackler
|
r43346 | ), | ||
], | ||||
Augie Fackler
|
r43347 | _(b'hg strip [-k] [-f] [-B bookmark] [-r] REV...'), | ||
Augie Fackler
|
r43346 | helpcategory=command.CATEGORY_MAINTENANCE, | ||
) | ||||
Pierre-Yves David
|
r19826 | def stripcmd(ui, repo, *revs, **opts): | ||
"""strip changesets and all their descendants from the repository | ||||
The strip command removes the specified changesets and all their | ||||
descendants. If the working directory has uncommitted changes, the | ||||
operation is aborted unless the --force flag is supplied, in which | ||||
case changes will be discarded. | ||||
If a parent of the working directory is stripped, then the working | ||||
directory will automatically be updated to the most recent | ||||
available ancestor of the stripped parent after the operation | ||||
completes. | ||||
Any stripped changesets are stored in ``.hg/strip-backup`` as a | ||||
bundle (see :hg:`help bundle` and :hg:`help unbundle`). They can | ||||
be restored by running :hg:`unbundle .hg/strip-backup/BUNDLE`, | ||||
where BUNDLE is the bundle file created by the strip. Note that | ||||
the local revision numbers will in general be different after the | ||||
restore. | ||||
Use the --no-backup option to discard the backup bundle once the | ||||
operation completes. | ||||
Strip is not a history-rewriting operation and can be used on | ||||
changesets in the public phase. But if the stripped changesets have | ||||
been pushed to a remote repository you will likely pull them again. | ||||
Return 0 on success. | ||||
""" | ||||
Pulkit Goyal
|
r32897 | opts = pycompat.byteskwargs(opts) | ||
Jordi Gutiérrez Hermoso
|
r22057 | backup = True | ||
Augie Fackler
|
r43347 | if opts.get(b'no_backup') or opts.get(b'nobackup'): | ||
Jordi Gutiérrez Hermoso
|
r22057 | backup = False | ||
Pierre-Yves David
|
r19826 | |||
cl = repo.changelog | ||||
Augie Fackler
|
r43347 | revs = list(revs) + opts.get(b'rev') | ||
Pierre-Yves David
|
r19826 | revs = set(scmutil.revrange(repo, revs)) | ||
Bryan O'Sullivan
|
r27839 | with repo.wlock(): | ||
Augie Fackler
|
r43347 | bookmarks = set(opts.get(b'bookmark')) | ||
Shubhanshu Agrawal
|
r27029 | if bookmarks: | ||
Shubhanshu Agrawal
|
r26972 | repomarks = repo._bookmarks | ||
Shubhanshu Agrawal
|
r27029 | if not bookmarks.issubset(repomarks): | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b"bookmark '%s' not found") | ||
% b','.join(sorted(bookmarks - set(repomarks.keys()))) | ||||
Augie Fackler
|
r43346 | ) | ||
Siddharth Agarwal
|
r20096 | |||
# If the requested bookmark is not the only one pointing to a | ||||
# a revision we have to only delete the bookmark and not strip | ||||
# anything. revsets cannot detect that case. | ||||
Shubhanshu Agrawal
|
r27029 | nodetobookmarks = {} | ||
Gregory Szorc
|
r43375 | for mark, node in pycompat.iteritems(repomarks): | ||
Shubhanshu Agrawal
|
r27029 | nodetobookmarks.setdefault(node, []).append(mark) | ||
for marks in nodetobookmarks.values(): | ||||
if bookmarks.issuperset(marks): | ||||
David Demelier
|
r38146 | rsrevs = scmutil.bookmarkrevs(repo, marks[0]) | ||
Shubhanshu Agrawal
|
r27030 | revs.update(set(rsrevs)) | ||
Siddharth Agarwal
|
r20096 | if not revs: | ||
Augie Fackler
|
r43347 | with repo.lock(), repo.transaction(b'bookmark') as tr: | ||
Boris Feld
|
r33488 | bmchanges = [(b, None) for b in bookmarks] | ||
repomarks.applychanges(repo, tr, bmchanges) | ||||
Martin von Zweigbergk
|
r32920 | for bookmark in sorted(bookmarks): | ||
Augie Fackler
|
r43347 | ui.write(_(b"bookmark '%s' deleted\n") % bookmark) | ||
Siddharth Agarwal
|
r20096 | |||
if not revs: | ||||
Augie Fackler
|
r43347 | raise error.Abort(_(b'empty revision set')) | ||
Siddharth Agarwal
|
r20096 | |||
descendants = set(cl.descendants(revs)) | ||||
strippedrevs = revs.union(descendants) | ||||
roots = revs.difference(descendants) | ||||
# if one of the wdir parent is stripped we'll need | ||||
# to update away to an earlier revision | ||||
Augie Fackler
|
r43346 | update = any( | ||
p != nullid and cl.rev(p) in strippedrevs | ||||
for p in repo.dirstate.parents() | ||||
) | ||||
Siddharth Agarwal
|
r20096 | |||
rootnodes = set(cl.node(r) for r in roots) | ||||
Pierre-Yves David
|
r19826 | |||
Siddharth Agarwal
|
r20096 | q = getattr(repo, 'mq', None) | ||
if q is not None and q.applied: | ||||
# refresh queue state if we're about to strip | ||||
# applied patches | ||||
Augie Fackler
|
r43347 | if cl.rev(repo.lookup(b'qtip')) in strippedrevs: | ||
Siddharth Agarwal
|
r20096 | q.applieddirty = True | ||
start = 0 | ||||
end = len(q.applied) | ||||
for i, statusentry in enumerate(q.applied): | ||||
if statusentry.node in rootnodes: | ||||
# if one of the stripped roots is an applied | ||||
# patch, only part of the queue is stripped | ||||
start = i | ||||
break | ||||
del q.applied[start:end] | ||||
q.savedirty() | ||||
revs = sorted(rootnodes) | ||||
Augie Fackler
|
r43347 | if update and opts.get(b'keep'): | ||
Paul Morelle
|
r34575 | urev = _findupdatetarget(repo, revs) | ||
Siddharth Agarwal
|
r20102 | uctx = repo[urev] | ||
Siddharth Agarwal
|
r20096 | |||
Siddharth Agarwal
|
r20102 | # only reset the dirstate for files that would actually change | ||
# between the working context and uctx | ||||
Augie Fackler
|
r35843 | descendantrevs = repo.revs(b"%d::.", uctx.rev()) | ||
Siddharth Agarwal
|
r20102 | changedfiles = [] | ||
for rev in descendantrevs: | ||||
# blindly reset the files, regardless of what actually changed | ||||
changedfiles.extend(repo[rev].files()) | ||||
Siddharth Agarwal
|
r20096 | |||
Siddharth Agarwal
|
r20102 | # reset files that only changed in the dirstate too | ||
dirstate = repo.dirstate | ||||
Augie Fackler
|
r43347 | dirchanges = [f for f in dirstate if dirstate[f] != b'n'] | ||
Siddharth Agarwal
|
r20102 | changedfiles.extend(dirchanges) | ||
Siddharth Agarwal
|
r20096 | |||
Siddharth Agarwal
|
r20102 | repo.dirstate.rebuild(urev, uctx.manifest(), changedfiles) | ||
FUJIWARA Katsunori
|
r26748 | repo.dirstate.write(repo.currenttransaction()) | ||
Matt Mackall
|
r24709 | |||
# clear resolve state | ||||
Augie Fackler
|
r43347 | merge.mergestate.clean(repo, repo[b'.'].node()) | ||
Matt Mackall
|
r24709 | |||
Siddharth Agarwal
|
r20102 | update = False | ||
Siddharth Agarwal
|
r20096 | |||
Augie Fackler
|
r43346 | strip( | ||
ui, | ||||
repo, | ||||
revs, | ||||
backup=backup, | ||||
update=update, | ||||
Augie Fackler
|
r43347 | force=opts.get(b'force'), | ||
Augie Fackler
|
r43346 | bookmarks=bookmarks, | ||
Augie Fackler
|
r43347 | soft=opts[b'soft'], | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r19826 | |||
return 0 | ||||