uncommit.py
322 lines
| 10.1 KiB
| text/x-python
|
PythonLexer
/ hgext / uncommit.py
Pulkit Goyal
|
r34193 | # uncommit - undo the actions of a commit | ||
# | ||||
# Copyright 2011 Peter Arrenbrecht <peter.arrenbrecht@gmail.com> | ||||
# Logilab SA <contact@logilab.fr> | ||||
# Pierre-Yves David <pierre-yves.david@ens-lyon.org> | ||||
# Patrick Mezard <patrick@mezard.eu> | ||||
# Copyright 2016 Facebook, Inc. | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
"""uncommit part or all of a local changeset (EXPERIMENTAL) | ||||
This command undoes the effect of a local commit, returning the affected | ||||
files to their uncommitted state. This means that files modified, added or | ||||
removed in the changeset will be left unchanged, and so will remain modified, | ||||
added and removed in the working directory. | ||||
""" | ||||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Pulkit Goyal
|
r34193 | |||
from mercurial.i18n import _ | ||||
from mercurial import ( | ||||
Pulkit Goyal
|
r34285 | cmdutil, | ||
Pulkit Goyal
|
r34193 | commands, | ||
context, | ||||
Martin von Zweigbergk
|
r41370 | copies as copiesmod, | ||
Pulkit Goyal
|
r34193 | error, | ||
Pulkit Goyal
|
r35177 | obsutil, | ||
r43923 | pathutil, | |||
Pulkit Goyal
|
r35005 | pycompat, | ||
Pulkit Goyal
|
r34193 | registrar, | ||
Pulkit Goyal
|
r35244 | rewriteutil, | ||
Pulkit Goyal
|
r34193 | scmutil, | ||
) | ||||
cmdtable = {} | ||||
command = registrar.command(cmdtable) | ||||
Boris Feld
|
r34759 | configtable = {} | ||
configitem = registrar.configitem(configtable) | ||||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'experimental', | ||
b'uncommitondirtywdir', | ||||
default=False, | ||||
Boris Feld
|
r34759 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'experimental', | ||
b'uncommit.keep', | ||||
default=False, | ||||
Martin von Zweigbergk
|
r41916 | ) | ||
Boris Feld
|
r34759 | |||
Pulkit Goyal
|
r34193 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
# 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' | ||
Pulkit Goyal
|
r34193 | |||
Augie Fackler
|
r43346 | |||
def _commitfiltered( | ||||
repo, ctx, match, keepcommit, message=None, user=None, date=None | ||||
): | ||||
Pulkit Goyal
|
r34193 | """Recommit ctx with changed files not in match. Return the new | ||
node identifier, or None if nothing changed. | ||||
""" | ||||
base = ctx.p1() | ||||
# ctx | ||||
initialfiles = set(ctx.files()) | ||||
Augie Fackler
|
r44937 | exclude = {f for f in initialfiles if match(f)} | ||
Pulkit Goyal
|
r34193 | |||
# No files matched commit, so nothing excluded | ||||
if not exclude: | ||||
return None | ||||
# return the p1 so that we don't create an obsmarker later | ||||
Martin von Zweigbergk
|
r36992 | if not keepcommit: | ||
Martin von Zweigbergk
|
r41442 | return ctx.p1().node() | ||
Pulkit Goyal
|
r34193 | |||
Augie Fackler
|
r43346 | files = initialfiles - exclude | ||
Pulkit Goyal
|
r34193 | # Filter copies | ||
Martin von Zweigbergk
|
r41370 | copied = copiesmod.pathcopies(base, ctx) | ||
Gregory Szorc
|
r49768 | copied = {dst: src for dst, src in copied.items() if dst in files} | ||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r34193 | def filectxfn(repo, memctx, path, contentctx=ctx, redirect=()): | ||
if path not in contentctx: | ||||
return None | ||||
fctx = contentctx[path] | ||||
Augie Fackler
|
r43346 | mctx = context.memfilectx( | ||
repo, | ||||
memctx, | ||||
fctx.path(), | ||||
fctx.data(), | ||||
fctx.islink(), | ||||
fctx.isexec(), | ||||
copysource=copied.get(path), | ||||
) | ||||
Pulkit Goyal
|
r34193 | return mctx | ||
Martin von Zweigbergk
|
r41911 | if not files: | ||
Augie Fackler
|
r43347 | repo.ui.status(_(b"note: keeping empty commit\n")) | ||
Martin von Zweigbergk
|
r41911 | |||
Matt Harbison
|
r43172 | if message is None: | ||
message = ctx.description() | ||||
if not user: | ||||
user = ctx.user() | ||||
if not date: | ||||
date = ctx.date() | ||||
Augie Fackler
|
r43346 | new = context.memctx( | ||
repo, | ||||
Joerg Sonnenberger
|
r47771 | parents=[base.node(), repo.nullid], | ||
Augie Fackler
|
r43346 | text=message, | ||
files=files, | ||||
filectxfn=filectxfn, | ||||
user=user, | ||||
date=date, | ||||
extra=ctx.extra(), | ||||
) | ||||
Martin von Zweigbergk
|
r38442 | return repo.commitctx(new) | ||
Pulkit Goyal
|
r34193 | |||
Augie Fackler
|
r43346 | |||
@command( | ||||
Augie Fackler
|
r43347 | b'uncommit', | ||
Augie Fackler
|
r43346 | [ | ||
Augie Fackler
|
r43347 | (b'', b'keep', None, _(b'allow an empty commit after uncommitting')), | ||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'', | ||
b'allow-dirty-working-copy', | ||||
Augie Fackler
|
r43346 | False, | ||
Augie Fackler
|
r43347 | _(b'allow uncommit with outstanding changes'), | ||
Augie Fackler
|
r43346 | ), | ||
(b'n', b'note', b'', _(b'store a note on uncommit'), _(b'TEXT')), | ||||
] | ||||
+ commands.walkopts | ||||
+ commands.commitopts | ||||
+ commands.commitopts2 | ||||
Matt Harbison
|
r43173 | + commands.commitopts3, | ||
Augie Fackler
|
r43347 | _(b'[OPTION]... [FILE]...'), | ||
Augie Fackler
|
r43346 | helpcategory=command.CATEGORY_CHANGE_MANAGEMENT, | ||
) | ||||
Pulkit Goyal
|
r34193 | def uncommit(ui, repo, *pats, **opts): | ||
"""uncommit part or all of a local changeset | ||||
This command undoes the effect of a local commit, returning the affected | ||||
files to their uncommitted state. This means that files modified or | ||||
deleted in the changeset will be left unchanged, and so will remain | ||||
modified in the working directory. | ||||
Martin von Zweigbergk
|
r36991 | |||
If no files are specified, the commit will be pruned, unless --keep is | ||||
given. | ||||
Pulkit Goyal
|
r34193 | """ | ||
Martin von Zweigbergk
|
r48221 | cmdutil.check_note_size(opts) | ||
Martin von Zweigbergk
|
r48225 | cmdutil.resolve_commit_options(ui, opts) | ||
Matt Harbison
|
r43173 | |||
Pulkit Goyal
|
r34193 | with repo.wlock(), repo.lock(): | ||
Augie Fackler
|
r44042 | st = repo.status() | ||
m, a, r, d = st.modified, st.added, st.removed, st.deleted | ||||
Navaneeth Suresh
|
r42025 | isdirtypath = any(set(m + a + r + d) & set(pats)) | ||
Augie Fackler
|
r43346 | allowdirtywcopy = opts[ | ||
Matt Harbison
|
r51778 | 'allow_dirty_working_copy' | ||
Augie Fackler
|
r43347 | ] or repo.ui.configbool(b'experimental', b'uncommitondirtywdir') | ||
Navaneeth Suresh
|
r42026 | if not allowdirtywcopy and (not pats or isdirtypath): | ||
Augie Fackler
|
r43346 | cmdutil.bailifchanged( | ||
repo, | ||||
Martin von Zweigbergk
|
r43387 | hint=_(b'requires --allow-dirty-working-copy to uncommit'), | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | old = repo[b'.'] | ||
rewriteutil.precheck(repo, [old.rev()], b'uncommit') | ||||
Pulkit Goyal
|
r34193 | if len(old.parents()) > 1: | ||
Martin von Zweigbergk
|
r47192 | raise error.InputError(_(b"cannot uncommit merge changeset")) | ||
Pulkit Goyal
|
r34193 | |||
Matt Harbison
|
r51778 | match = scmutil.match(old, pats, pycompat.byteskwargs(opts)) | ||
Matt Harbison
|
r42218 | |||
# Check all explicitly given files; abort if there's a problem. | ||||
if match.files(): | ||||
s = old.status(old.p1(), match, listclean=True) | ||||
eligible = set(s.added) | set(s.modified) | set(s.removed) | ||||
badfiles = set(match.files()) - eligible | ||||
# Naming a parent directory of an eligible file is OK, even | ||||
# if not everything tracked in that directory can be | ||||
# uncommitted. | ||||
if badfiles: | ||||
r43923 | badfiles -= {f for f in pathutil.dirs(eligible)} | |||
Matt Harbison
|
r42218 | |||
for f in sorted(badfiles): | ||||
if f in s.clean: | ||||
Augie Fackler
|
r43346 | hint = _( | ||
Martin von Zweigbergk
|
r43387 | b"file was not changed in working directory parent" | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r42218 | elif repo.wvfs.exists(f): | ||
hint = _(b"file was untracked in working directory parent") | ||||
else: | ||||
hint = _(b"file does not exist") | ||||
Martin von Zweigbergk
|
r47192 | raise error.InputError( | ||
Augie Fackler
|
r43346 | _(b'cannot uncommit "%s"') % scmutil.getuipathfn(repo)(f), | ||
hint=hint, | ||||
) | ||||
Matt Harbison
|
r42218 | |||
Augie Fackler
|
r43347 | with repo.transaction(b'uncommit'): | ||
Matt Harbison
|
r51778 | if not (opts['message'] or opts['logfile']): | ||
opts['message'] = old.description() | ||||
message = cmdutil.logmessage(ui, pycompat.byteskwargs(opts)) | ||||
Matt Harbison
|
r43172 | |||
Martin von Zweigbergk
|
r41916 | keepcommit = pats | ||
if not keepcommit: | ||||
Matt Harbison
|
r51778 | if opts.get('keep') is not None: | ||
keepcommit = opts.get('keep') | ||||
Martin von Zweigbergk
|
r41916 | else: | ||
Augie Fackler
|
r43347 | keepcommit = ui.configbool( | ||
b'experimental', b'uncommit.keep' | ||||
) | ||||
Augie Fackler
|
r43346 | newid = _commitfiltered( | ||
repo, | ||||
old, | ||||
match, | ||||
keepcommit, | ||||
message=message, | ||||
Matt Harbison
|
r51778 | user=opts.get('user'), | ||
date=opts.get('date'), | ||||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r34193 | if newid is None: | ||
Augie Fackler
|
r43347 | ui.status(_(b"nothing to uncommit\n")) | ||
Pulkit Goyal
|
r34193 | return 1 | ||
mapping = {} | ||||
if newid != old.p1().node(): | ||||
# Move local changes on filtered changeset | ||||
mapping[old.node()] = (newid,) | ||||
else: | ||||
# Fully removed the old commit | ||||
mapping[old.node()] = () | ||||
r50855 | with repo.dirstate.changing_parents(repo): | |||
Martin von Zweigbergk
|
r42103 | scmutil.movedirstate(repo, repo[newid], match) | ||
Pulkit Goyal
|
r35177 | |||
Augie Fackler
|
r43347 | scmutil.cleanupnodes(repo, mapping, b'uncommit', fixphase=True) | ||
Martin von Zweigbergk
|
r41371 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r35177 | def predecessormarkers(ctx): | ||
"""yields the obsolete markers marking the given changeset as a successor""" | ||||
for data in ctx.repo().obsstore.predecessors.get(ctx.node(), ()): | ||||
yield obsutil.marker(ctx.repo(), data) | ||||
Augie Fackler
|
r43346 | |||
@command( | ||||
Augie Fackler
|
r43347 | b'unamend', | ||
Augie Fackler
|
r43346 | [], | ||
helpcategory=command.CATEGORY_CHANGE_MANAGEMENT, | ||||
helpbasic=True, | ||||
) | ||||
Pulkit Goyal
|
r35177 | def unamend(ui, repo, **opts): | ||
Martin von Zweigbergk
|
r35827 | """undo the most recent amend operation on a current changeset | ||
Pulkit Goyal
|
r35177 | |||
This command will roll back to the previous version of a changeset, | ||||
leaving working directory in state in which it was before running | ||||
`hg amend` (e.g. files modified as part of an amend will be | ||||
marked as modified `hg status`) | ||||
""" | ||||
unfi = repo.unfiltered() | ||||
Augie Fackler
|
r43347 | with repo.wlock(), repo.lock(), repo.transaction(b'unamend'): | ||
Pulkit Goyal
|
r35201 | # identify the commit from which to unamend | ||
Augie Fackler
|
r43347 | curctx = repo[b'.'] | ||
Pulkit Goyal
|
r35177 | |||
Augie Fackler
|
r43347 | rewriteutil.precheck(repo, [curctx.rev()], b'unamend') | ||
Martin von Zweigbergk
|
r49517 | if len(curctx.parents()) > 1: | ||
raise error.InputError(_(b"cannot unamend merge changeset")) | ||||
Pulkit Goyal
|
r35177 | |||
Martin von Zweigbergk
|
r49836 | expected_keys = (b'amend_source', b'unamend_source') | ||
if not any(key in curctx.extra() for key in expected_keys): | ||||
raise error.InputError( | ||||
_( | ||||
b"working copy parent was not created by 'hg amend' or " | ||||
b"'hg unamend'" | ||||
) | ||||
) | ||||
Pulkit Goyal
|
r35177 | # identify the commit to which to unamend | ||
markers = list(predecessormarkers(curctx)) | ||||
if len(markers) != 1: | ||||
Augie Fackler
|
r43347 | e = _(b"changeset must have one predecessor, found %i predecessors") | ||
Martin von Zweigbergk
|
r47192 | raise error.InputError(e % len(markers)) | ||
Pulkit Goyal
|
r35177 | |||
prednode = markers[0].prednode() | ||||
predctx = unfi[prednode] | ||||
# add an extra so that we get a new hash | ||||
# note: allowing unamend to undo an unamend is an intentional feature | ||||
extras = predctx.extra() | ||||
Augie Fackler
|
r43347 | extras[b'unamend_source'] = curctx.hex() | ||
Pulkit Goyal
|
r35177 | |||
def filectxfn(repo, ctx_, path): | ||||
try: | ||||
return predctx.filectx(path) | ||||
except KeyError: | ||||
return None | ||||
# Make a new commit same as predctx | ||||
Augie Fackler
|
r43346 | newctx = context.memctx( | ||
repo, | ||||
parents=(predctx.p1(), predctx.p2()), | ||||
text=predctx.description(), | ||||
files=predctx.files(), | ||||
filectxfn=filectxfn, | ||||
user=predctx.user(), | ||||
date=predctx.date(), | ||||
extra=extras, | ||||
) | ||||
Martin von Zweigbergk
|
r38442 | newprednode = repo.commitctx(newctx) | ||
Pulkit Goyal
|
r35177 | newpredctx = repo[newprednode] | ||
dirstate = repo.dirstate | ||||
r50855 | with dirstate.changing_parents(repo): | |||
Martin von Zweigbergk
|
r42103 | scmutil.movedirstate(repo, newpredctx) | ||
Pulkit Goyal
|
r35177 | |||
mapping = {curctx.node(): (newprednode,)} | ||||
Augie Fackler
|
r43347 | scmutil.cleanupnodes(repo, mapping, b'unamend', fixphase=True) | ||