rewriteutil.py
135 lines
| 4.2 KiB
| text/x-python
|
PythonLexer
/ mercurial / rewriteutil.py
Pulkit Goyal
|
r35242 | # rewriteutil.py - utility functions for rewriting changesets | ||
# | ||||
# Copyright 2017 Octobus <contact@octobus.net> | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
from __future__ import absolute_import | ||||
Matt Harbison
|
r45993 | import re | ||
Pulkit Goyal
|
r35243 | from .i18n import _ | ||
Pulkit Goyal
|
r35242 | from . import ( | ||
Pulkit Goyal
|
r35243 | error, | ||
node, | ||||
Pulkit Goyal
|
r35242 | obsolete, | ||
Matt Harbison
|
r45993 | obsutil, | ||
Pulkit Goyal
|
r35242 | revset, | ||
Matt Harbison
|
r45993 | scmutil, | ||
Pulkit Goyal
|
r35242 | ) | ||
Augie Fackler
|
r43345 | |||
Matt Harbison
|
r45996 | NODE_RE = re.compile(br'\b[0-9a-f]{6,64}\b') | ||
Matt Harbison
|
r45993 | |||
Augie Fackler
|
r43347 | def precheck(repo, revs, action=b'rewrite'): | ||
Pulkit Goyal
|
r35243 | """check if revs can be rewritten | ||
action is used to control the error message. | ||||
Make sure this function is called after taking the lock. | ||||
""" | ||||
if node.nullrev in revs: | ||||
Augie Fackler
|
r43347 | msg = _(b"cannot %s null changeset") % action | ||
hint = _(b"no changeset checked out") | ||||
Martin von Zweigbergk
|
r46457 | raise error.InputError(msg, hint=hint) | ||
Pulkit Goyal
|
r35243 | |||
if len(repo[None].parents()) > 1: | ||||
Martin von Zweigbergk
|
r46457 | raise error.StateError(_(b"cannot %s while merging") % action) | ||
Pulkit Goyal
|
r35243 | |||
Augie Fackler
|
r43347 | publicrevs = repo.revs(b'%ld and public()', revs) | ||
Pulkit Goyal
|
r35243 | if publicrevs: | ||
Augie Fackler
|
r43347 | msg = _(b"cannot %s public changesets") % action | ||
hint = _(b"see 'hg help phases' for details") | ||||
Martin von Zweigbergk
|
r46457 | raise error.InputError(msg, hint=hint) | ||
Pulkit Goyal
|
r35243 | |||
newunstable = disallowednewunstable(repo, revs) | ||||
if newunstable: | ||||
Martin von Zweigbergk
|
r46457 | raise error.InputError(_(b"cannot %s changeset with children") % action) | ||
Pulkit Goyal
|
r35243 | |||
Augie Fackler
|
r43345 | |||
Pulkit Goyal
|
r35242 | def disallowednewunstable(repo, revs): | ||
"""Checks whether editing the revs will create new unstable changesets and | ||||
are we allowed to create them. | ||||
To allow new unstable changesets, set the config: | ||||
`experimental.evolution.allowunstable=True` | ||||
""" | ||||
allowunstable = obsolete.isenabled(repo, obsolete.allowunstableopt) | ||||
if allowunstable: | ||||
return revset.baseset() | ||||
Augie Fackler
|
r43347 | return repo.revs(b"(%ld::) - %ld", revs, revs) | ||
Manuel Jacob
|
r45682 | |||
def skip_empty_successor(ui, command): | ||||
empty_successor = ui.config(b'rewrite', b'empty-successor') | ||||
if empty_successor == b'skip': | ||||
return True | ||||
elif empty_successor == b'keep': | ||||
return False | ||||
else: | ||||
raise error.ConfigError( | ||||
_( | ||||
b"%s doesn't know how to handle config " | ||||
b"rewrite.empty-successor=%s (only 'skip' and 'keep' are " | ||||
b"supported)" | ||||
) | ||||
% (command, empty_successor) | ||||
) | ||||
Matt Harbison
|
r45993 | |||
Matt Harbison
|
r45994 | def update_hash_refs(repo, commitmsg, pending=None): | ||
Matt Harbison
|
r45993 | """Replace all obsolete commit hashes in the message with the current hash. | ||
If the obsolete commit was split or is divergent, the hash is not replaced | ||||
as there's no way to know which successor to choose. | ||||
Matt Harbison
|
r45994 | |||
For commands that update a series of commits in the current transaction, the | ||||
new obsolete markers can be considered by setting ``pending`` to a mapping | ||||
of ``pending[oldnode] = [successor_node1, successor_node2,..]``. | ||||
Matt Harbison
|
r45993 | """ | ||
Matt Harbison
|
r45994 | if not pending: | ||
pending = {} | ||||
Matt Harbison
|
r45993 | cache = {} | ||
Matt Harbison
|
r45996 | hashes = re.findall(NODE_RE, commitmsg) | ||
Matt Harbison
|
r45993 | unfi = repo.unfiltered() | ||
Matt Harbison
|
r45996 | for h in hashes: | ||
fullnode = scmutil.resolvehexnodeidprefix(unfi, h) | ||||
Matt Harbison
|
r45993 | if fullnode is None: | ||
continue | ||||
ctx = unfi[fullnode] | ||||
if not ctx.obsolete(): | ||||
Matt Harbison
|
r45994 | successors = pending.get(fullnode) | ||
if successors is None: | ||||
continue | ||||
# obsutil.successorssets() returns a list of list of nodes | ||||
successors = [successors] | ||||
else: | ||||
successors = obsutil.successorssets(repo, ctx.node(), cache=cache) | ||||
Matt Harbison
|
r45993 | |||
# We can't make any assumptions about how to update the hash if the | ||||
# cset in question was split or diverged. | ||||
if len(successors) == 1 and len(successors[0]) == 1: | ||||
Matt Harbison
|
r46303 | successor = successors[0][0] | ||
if successor is not None: | ||||
newhash = node.hex(successor) | ||||
commitmsg = commitmsg.replace(h, newhash[: len(h)]) | ||||
else: | ||||
repo.ui.note( | ||||
_( | ||||
b'The stale commit message reference to %s could ' | ||||
b'not be updated\n(The referenced commit was dropped)\n' | ||||
) | ||||
% h | ||||
) | ||||
Matt Harbison
|
r45993 | else: | ||
repo.ui.note( | ||||
_( | ||||
b'The stale commit message reference to %s could ' | ||||
b'not be updated\n' | ||||
) | ||||
Matt Harbison
|
r45996 | % h | ||
Matt Harbison
|
r45993 | ) | ||
return commitmsg | ||||