commit.py
558 lines
| 18.5 KiB
| text/x-python
|
PythonLexer
/ mercurial / commit.py
r45759 | # commit.py - fonction to perform commit | |||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
import errno | ||||
from .i18n import _ | ||||
from .node import ( | ||||
hex, | ||||
nullrev, | ||||
) | ||||
from . import ( | ||||
r45760 | context, | |||
mergestate, | ||||
r45759 | metadata, | |||
phases, | ||||
scmutil, | ||||
subrepoutil, | ||||
) | ||||
r45791 | def _write_copy_meta(repo): | |||
"""return a (changelog, filelog) boolean tuple | ||||
changelog: copy related information should be stored in the changeset | ||||
filelof: copy related information should be written in the file revision | ||||
""" | ||||
if repo.filecopiesmode == b'changeset-sidedata': | ||||
writechangesetcopy = True | ||||
writefilecopymeta = True | ||||
else: | ||||
writecopiesto = repo.ui.config(b'experimental', b'copies.write-to') | ||||
writefilecopymeta = writecopiesto != b'changeset-only' | ||||
writechangesetcopy = writecopiesto in ( | ||||
b'changeset-only', | ||||
b'compatibility', | ||||
) | ||||
return writechangesetcopy, writefilecopymeta | ||||
r45759 | def commitctx(repo, ctx, error=False, origctx=None): | |||
"""Add a new revision to the target repository. | ||||
Revision information is passed via the context argument. | ||||
ctx.files() should list all files involved in this commit, i.e. | ||||
modified/added/removed files. On merge, it may be wider than the | ||||
ctx.files() to be committed, since any file nodes derived directly | ||||
from p1 or p2 are excluded from the committed ctx.files(). | ||||
origctx is for convert to work around the problem that bug | ||||
fixes to the files list in changesets change hashes. For | ||||
convert to be the identity, it can pass an origctx and this | ||||
function will use the same files list when it makes sense to | ||||
do so. | ||||
""" | ||||
repo = repo.unfiltered() | ||||
p1, p2 = ctx.p1(), ctx.p2() | ||||
user = ctx.user() | ||||
r45796 | with repo.lock(), repo.transaction(b"commit") as tr: | |||
r45883 | mn, files = _prepare_files(tr, ctx, error=error, origctx=origctx) | |||
r45759 | ||||
r45808 | extra = ctx.extra().copy() | |||
r45809 | if extra is not None: | |||
for name in ( | ||||
b'p1copies', | ||||
b'p2copies', | ||||
b'filesadded', | ||||
b'filesremoved', | ||||
): | ||||
extra.pop(name, None) | ||||
if repo.changelog._copiesstorage == b'extra': | ||||
r45883 | extra = _extra_with_copies(repo, extra, files) | |||
r45809 | ||||
Dan Villiom Podlaski Christiansen
|
r46379 | # save the tip to check whether we actually committed anything | ||
oldtip = repo.changelog.tiprev() | ||||
r45759 | # update changelog | |||
repo.ui.note(_(b"committing changelog\n")) | ||||
repo.changelog.delayupdate(tr) | ||||
n = repo.changelog.add( | ||||
mn, | ||||
r45884 | files, | |||
r45759 | ctx.description(), | |||
r45788 | tr, | |||
r45759 | p1.node(), | |||
p2.node(), | ||||
user, | ||||
ctx.date(), | ||||
r45808 | extra, | |||
r45759 | ) | |||
Joerg Sonnenberger
|
r47066 | rev = repo[n].rev() | ||
Joerg Sonnenberger
|
r47083 | if oldtip != repo.changelog.tiprev(): | ||
repo.register_changeset(rev, repo.changelog.changelogrevision(rev)) | ||||
r45759 | xp1, xp2 = p1.hex(), p2 and p2.hex() or b'' | |||
repo.hook( | ||||
Augie Fackler
|
r46554 | b'pretxncommit', | ||
throw=True, | ||||
node=hex(n), | ||||
parent1=xp1, | ||||
parent2=xp2, | ||||
r45759 | ) | |||
# set the new commit is proper phase | ||||
targetphase = subrepoutil.newcommitphase(repo.ui, ctx) | ||||
Dan Villiom Podlaski Christiansen
|
r46379 | |||
# prevent unmarking changesets as public on recommit | ||||
Joerg Sonnenberger
|
r47066 | waspublic = oldtip == repo.changelog.tiprev() and not repo[rev].phase() | ||
Dan Villiom Podlaski Christiansen
|
r46379 | |||
if targetphase and not waspublic: | ||||
r45759 | # retract boundary do not alter parent changeset. | |||
# if a parent have higher the resulting phase will | ||||
# be compliant anyway | ||||
# | ||||
# if minimal phase was 0 we don't need to retract anything | ||||
Joerg Sonnenberger
|
r47066 | phases.registernew(repo, tr, targetphase, [rev]) | ||
r45759 | return n | |||
r45760 | ||||
r45797 | def _prepare_files(tr, ctx, error=False, origctx=None): | |||
repo = ctx.repo() | ||||
p1 = ctx.p1() | ||||
writechangesetcopy, writefilecopymeta = _write_copy_meta(repo) | ||||
Pulkit Goyal
|
r46300 | files = metadata.ChangingFiles() | ||
Pulkit Goyal
|
r46298 | ms = mergestate.mergestate.read(repo) | ||
salvaged = _get_salvaged(repo, ms, ctx) | ||||
Pulkit Goyal
|
r46300 | for s in salvaged: | ||
files.mark_salvaged(s) | ||||
r45797 | ||||
r49565 | narrow_files = {} | |||
if not ctx.repo().narrowmatch().always(): | ||||
for f, e in ms.allextras().items(): | ||||
action = e.get(b'outside-narrow-merge-action') | ||||
if action is not None: | ||||
narrow_files[f] = action | ||||
if ctx.manifestnode() and not narrow_files: | ||||
r45797 | # reuse an existing manifest revision | |||
repo.ui.debug(b'reusing known manifest\n') | ||||
mn = ctx.manifestnode() | ||||
r45888 | files.update_touched(ctx.files()) | |||
r45797 | if writechangesetcopy: | |||
r45888 | files.update_added(ctx.filesadded()) | |||
files.update_removed(ctx.filesremoved()) | ||||
r49565 | elif not ctx.files() and not narrow_files: | |||
r45797 | repo.ui.debug(b'reusing manifest from p1 (no file change)\n') | |||
mn = p1.manifestnode() | ||||
else: | ||||
r49565 | mn = _process_files(tr, ctx, ms, files, narrow_files, error=error) | |||
r45797 | ||||
if origctx and origctx.manifestnode() == mn: | ||||
r45888 | origfiles = origctx.files() | |||
assert files.touched.issubset(origfiles) | ||||
files.update_touched(origfiles) | ||||
r45797 | ||||
r45886 | if writechangesetcopy: | |||
files.update_copies_from_p1(ctx.p1copies()) | ||||
files.update_copies_from_p2(ctx.p2copies()) | ||||
r45883 | ||||
Pulkit Goyal
|
r46297 | return mn, files | ||
def _get_salvaged(repo, ms, ctx): | ||||
Augie Fackler
|
r46554 | """returns a list of salvaged files | ||
Pulkit Goyal
|
r46297 | |||
returns empty list if config option which process salvaged files are | ||||
Augie Fackler
|
r46554 | not enabled""" | ||
Pulkit Goyal
|
r46297 | salvaged = [] | ||
copy_sd = repo.filecopiesmode == b'changeset-sidedata' | ||||
r46254 | if copy_sd and len(ctx.parents()) > 1: | |||
if ms.active(): | ||||
Pulkit Goyal
|
r46307 | for fname in sorted(ms.allextras().keys()): | ||
r46254 | might_removed = ms.extras(fname).get(b'merge-removal-candidate') | |||
if might_removed == b'yes': | ||||
if fname in ctx: | ||||
Pulkit Goyal
|
r46297 | salvaged.append(fname) | ||
return salvaged | ||||
r45797 | ||||
r49565 | def _process_files(tr, ctx, ms, files, narrow_files=None, error=False): | |||
r45794 | repo = ctx.repo() | |||
p1 = ctx.p1() | ||||
p2 = ctx.p2() | ||||
writechangesetcopy, writefilecopymeta = _write_copy_meta(repo) | ||||
m1ctx = p1.manifestctx() | ||||
m2ctx = p2.manifestctx() | ||||
mctx = m1ctx.copy() | ||||
m = mctx.read() | ||||
m1 = m1ctx.read() | ||||
m2 = m2ctx.read() | ||||
# check in files | ||||
added = [] | ||||
removed = list(ctx.removed()) | ||||
linkrev = len(repo) | ||||
repo.ui.note(_(b"committing files:\n")) | ||||
uipathfn = scmutil.getuipathfn(repo) | ||||
r49565 | all_files = ctx.modified() + ctx.added() | |||
all_files.extend(narrow_files.keys()) | ||||
all_files.sort() | ||||
for f in all_files: | ||||
r45794 | repo.ui.note(uipathfn(f) + b"\n") | |||
r49565 | if f in narrow_files: | |||
narrow_action = narrow_files.get(f) | ||||
r49592 | if narrow_action == mergestate.CHANGE_REMOVED: | |||
files.mark_removed(f) | ||||
removed.append(f) | ||||
elif narrow_action == mergestate.CHANGE_ADDED: | ||||
files.mark_added(f) | ||||
added.append(f) | ||||
m[f] = m2[f] | ||||
flags = m2ctx.find(f)[1] or b'' | ||||
m.setflag(f, flags) | ||||
elif narrow_action == mergestate.CHANGE_MODIFIED: | ||||
r49565 | files.mark_touched(f) | |||
added.append(f) | ||||
m[f] = m2[f] | ||||
flags = m2ctx.find(f)[1] or b'' | ||||
m.setflag(f, flags) | ||||
else: | ||||
msg = _(b"corrupted mergestate, unknown narrow action: %b") | ||||
hint = _(b"restart the merge") | ||||
raise error.Abort(msg, hint=hint) | ||||
continue | ||||
r45794 | try: | |||
fctx = ctx[f] | ||||
if fctx is None: | ||||
removed.append(f) | ||||
else: | ||||
added.append(f) | ||||
m[f], is_touched = _filecommit( | ||||
Pulkit Goyal
|
r46014 | repo, fctx, m1, m2, linkrev, tr, writefilecopymeta, ms | ||
r45794 | ) | |||
if is_touched: | ||||
if is_touched == 'added': | ||||
r45892 | files.mark_added(f) | |||
r46187 | elif is_touched == 'merged': | |||
files.mark_merged(f) | ||||
r45892 | else: | |||
files.mark_touched(f) | ||||
r45794 | m.setflag(f, fctx.flags()) | |||
except OSError: | ||||
repo.ui.warn(_(b"trouble committing %s!\n") % uipathfn(f)) | ||||
raise | ||||
except IOError as inst: | ||||
errcode = getattr(inst, 'errno', errno.ENOENT) | ||||
if error or errcode and errcode != errno.ENOENT: | ||||
repo.ui.warn(_(b"trouble committing %s!\n") % uipathfn(f)) | ||||
raise | ||||
# update manifest | ||||
removed = [f for f in removed if f in m1 or f in m2] | ||||
drop = sorted([f for f in removed if f in m]) | ||||
for f in drop: | ||||
del m[f] | ||||
r45891 | if p2.rev() == nullrev: | |||
files.update_removed(removed) | ||||
else: | ||||
r45794 | rf = metadata.get_removal_filter(ctx, (p1, p2, m1, m2)) | |||
r45891 | for f in removed: | |||
if not rf(f): | ||||
files.mark_removed(f) | ||||
r45794 | ||||
r49565 | mn = _commit_manifest( | |||
tr, | ||||
linkrev, | ||||
ctx, | ||||
mctx, | ||||
m, | ||||
files.touched, | ||||
added, | ||||
drop, | ||||
bool(narrow_files), | ||||
) | ||||
r45889 | ||||
Pulkit Goyal
|
r46300 | return mn | ||
r45794 | ||||
r45760 | def _filecommit( | |||
Augie Fackler
|
r46554 | repo, | ||
fctx, | ||||
manifest1, | ||||
manifest2, | ||||
linkrev, | ||||
tr, | ||||
includecopymeta, | ||||
ms, | ||||
r45760 | ): | |||
""" | ||||
commit an individual file as part of a larger transaction | ||||
input: | ||||
fctx: a file context with the content we are trying to commit | ||||
manifest1: manifest of changeset first parent | ||||
manifest2: manifest of changeset second parent | ||||
linkrev: revision number of the changeset being created | ||||
tr: current transation | ||||
Pulkit Goyal
|
r46002 | includecopymeta: boolean, set to False to skip storing the copy data | ||
r45760 | (only used by the Google specific feature of using | |||
changeset extra as copy source of truth). | ||||
Pulkit Goyal
|
r46014 | ms: mergestate object | ||
r45760 | ||||
output: (filenode, touched) | ||||
filenode: the filenode that should be used by this changeset | ||||
r45787 | touched: one of: None (mean untouched), 'added' or 'modified' | |||
r45760 | """ | |||
fname = fctx.path() | ||||
Joerg Sonnenberger
|
r47771 | fparent1 = manifest1.get(fname, repo.nullid) | ||
fparent2 = manifest2.get(fname, repo.nullid) | ||||
r45760 | touched = None | |||
Joerg Sonnenberger
|
r47771 | if fparent1 == fparent2 == repo.nullid: | ||
r45760 | touched = 'added' | |||
if isinstance(fctx, context.filectx): | ||||
# This block fast path most comparisons which are usually done. It | ||||
# assumes that bare filectx is used and no merge happened, hence no | ||||
# need to create a new file revision in this case. | ||||
node = fctx.filenode() | ||||
if node in [fparent1, fparent2]: | ||||
repo.ui.debug(b'reusing %s filelog entry\n' % fname) | ||||
if ( | ||||
Joerg Sonnenberger
|
r47771 | fparent1 != repo.nullid | ||
and manifest1.flags(fname) != fctx.flags() | ||||
r45760 | ) or ( | |||
Joerg Sonnenberger
|
r47771 | fparent2 != repo.nullid | ||
and manifest2.flags(fname) != fctx.flags() | ||||
r45760 | ): | |||
touched = 'modified' | ||||
return node, touched | ||||
flog = repo.file(fname) | ||||
meta = {} | ||||
cfname = fctx.copysource() | ||||
fnode = None | ||||
if cfname and cfname != fname: | ||||
# Mark the new revision of this file as a copy of another | ||||
# file. This copy data will effectively act as a parent | ||||
# of this new revision. If this is a merge, the first | ||||
# parent will be the nullid (meaning "look up the copy data") | ||||
# and the second one will be the other parent. For example: | ||||
# | ||||
# 0 --- 1 --- 3 rev1 changes file foo | ||||
# \ / rev2 renames foo to bar and changes it | ||||
# \- 2 -/ rev3 should have bar with all changes and | ||||
# should record that bar descends from | ||||
# bar in rev2 and foo in rev1 | ||||
# | ||||
# this allows this merge to succeed: | ||||
# | ||||
# 0 --- 1 --- 3 rev4 reverts the content change from rev2 | ||||
# \ / merging rev3 and rev4 should use bar@rev2 | ||||
# \- 2 --- 4 as the merge base | ||||
# | ||||
cnode = manifest1.get(cfname) | ||||
newfparent = fparent2 | ||||
if manifest2: # branch merge | ||||
Joerg Sonnenberger
|
r47771 | if ( | ||
fparent2 == repo.nullid or cnode is None | ||||
): # copied on remote side | ||||
r45760 | if cfname in manifest2: | |||
cnode = manifest2[cfname] | ||||
newfparent = fparent1 | ||||
# Here, we used to search backwards through history to try to find | ||||
# where the file copy came from if the source of a copy was not in | ||||
# the parent directory. However, this doesn't actually make sense to | ||||
# do (what does a copy from something not in your working copy even | ||||
# mean?) and it causes bugs (eg, issue4476). Instead, we will warn | ||||
# the user that copy information was dropped, so if they didn't | ||||
# expect this outcome it can be fixed, but this is the correct | ||||
# behavior in this circumstance. | ||||
if cnode: | ||||
repo.ui.debug(b" %s: copy %s:%s\n" % (fname, cfname, hex(cnode))) | ||||
if includecopymeta: | ||||
meta[b"copy"] = cfname | ||||
meta[b"copyrev"] = hex(cnode) | ||||
Joerg Sonnenberger
|
r47771 | fparent1, fparent2 = repo.nullid, newfparent | ||
r45760 | else: | |||
repo.ui.warn( | ||||
_( | ||||
b"warning: can't find ancestor for '%s' " | ||||
b"copied from '%s'!\n" | ||||
) | ||||
% (fname, cfname) | ||||
) | ||||
Joerg Sonnenberger
|
r47771 | elif fparent1 == repo.nullid: | ||
fparent1, fparent2 = fparent2, repo.nullid | ||||
elif fparent2 != repo.nullid: | ||||
Pulkit Goyal
|
r47437 | if ms.active() and ms.extras(fname).get(b'filenode-source') == b'other': | ||
Joerg Sonnenberger
|
r47771 | fparent1, fparent2 = fparent2, repo.nullid | ||
Pulkit Goyal
|
r47567 | elif ms.active() and ms.extras(fname).get(b'merged') != b'yes': | ||
Joerg Sonnenberger
|
r47771 | fparent1, fparent2 = fparent1, repo.nullid | ||
Pulkit Goyal
|
r47437 | # is one parent an ancestor of the other? | ||
else: | ||||
fparentancestors = flog.commonancestorsheads(fparent1, fparent2) | ||||
if fparent1 in fparentancestors: | ||||
Joerg Sonnenberger
|
r47771 | fparent1, fparent2 = fparent2, repo.nullid | ||
Pulkit Goyal
|
r47437 | elif fparent2 in fparentancestors: | ||
Joerg Sonnenberger
|
r47771 | fparent2 = repo.nullid | ||
r45760 | ||||
Pulkit Goyal
|
r46159 | force_new_node = False | ||
# The file might have been deleted by merge code and user explicitly choose | ||||
# to revert the file and keep it. The other case can be where there is | ||||
# change-delete or delete-change conflict and user explicitly choose to keep | ||||
# the file. The goal is to create a new filenode for users explicit choices | ||||
if ( | ||||
repo.ui.configbool(b'experimental', b'merge-track-salvaged') | ||||
and ms.active() | ||||
and ms.extras(fname).get(b'merge-removal-candidate') == b'yes' | ||||
): | ||||
force_new_node = True | ||||
r45760 | # is the file changed? | |||
text = fctx.data() | ||||
Joerg Sonnenberger
|
r47771 | if ( | ||
fparent2 != repo.nullid | ||||
Valentin Gatien-Baron
|
r48770 | or fparent1 == repo.nullid | ||
Joerg Sonnenberger
|
r47771 | or meta | ||
or flog.cmp(fparent1, text) | ||||
or force_new_node | ||||
): | ||||
r45760 | if touched is None: # do not overwrite added | |||
Joerg Sonnenberger
|
r47771 | if fparent2 == repo.nullid: | ||
r46187 | touched = 'modified' | |||
else: | ||||
touched = 'merged' | ||||
r45760 | fnode = flog.add(text, meta, tr, linkrev, fparent1, fparent2) | |||
# are just the flags changed during merge? | ||||
elif fname in manifest1 and manifest1.flags(fname) != fctx.flags(): | ||||
touched = 'modified' | ||||
fnode = fparent1 | ||||
else: | ||||
fnode = fparent1 | ||||
return fnode, touched | ||||
r45789 | ||||
r49565 | def _commit_manifest( | |||
tr, | ||||
linkrev, | ||||
ctx, | ||||
mctx, | ||||
manifest, | ||||
files, | ||||
added, | ||||
drop, | ||||
has_some_narrow_action=False, | ||||
): | ||||
r45789 | """make a new manifest entry (or reuse a new one) | |||
given an initialised manifest context and precomputed list of | ||||
- files: files affected by the commit | ||||
- added: new entries in the manifest | ||||
- drop: entries present in parents but absent of this one | ||||
Create a new manifest revision, reuse existing ones if possible. | ||||
Return the nodeid of the manifest revision. | ||||
""" | ||||
repo = ctx.repo() | ||||
md = None | ||||
# all this is cached, so it is find to get them all from the ctx. | ||||
p1 = ctx.p1() | ||||
p2 = ctx.p2() | ||||
m1ctx = p1.manifestctx() | ||||
m1 = m1ctx.read() | ||||
if not files: | ||||
# if no "files" actually changed in terms of the changelog, | ||||
# try hard to detect unmodified manifest entry so that the | ||||
# exact same commit can be reproduced later on convert. | ||||
md = m1.diff(manifest, scmutil.matchfiles(repo, ctx.files())) | ||||
if not files and md: | ||||
repo.ui.debug( | ||||
b'not reusing manifest (no file change in ' | ||||
b'changelog, but manifest differs)\n' | ||||
) | ||||
if files or md: | ||||
repo.ui.note(_(b"committing manifest\n")) | ||||
# we're using narrowmatch here since it's already applied at | ||||
# other stages (such as dirstate.walk), so we're already | ||||
# ignoring things outside of narrowspec in most cases. The | ||||
# one case where we might have files outside the narrowspec | ||||
# at this point is merges, and we already error out in the | ||||
# case where the merge has files outside of the narrowspec, | ||||
# so this is safe. | ||||
r49565 | if has_some_narrow_action: | |||
match = None | ||||
else: | ||||
match = repo.narrowmatch() | ||||
r45789 | mn = mctx.write( | |||
tr, | ||||
linkrev, | ||||
p1.manifestnode(), | ||||
p2.manifestnode(), | ||||
added, | ||||
drop, | ||||
r49565 | match=match, | |||
r45789 | ) | |||
else: | ||||
repo.ui.debug( | ||||
b'reusing manifest from p1 (listed files ' b'actually unchanged)\n' | ||||
) | ||||
mn = p1.manifestnode() | ||||
return mn | ||||
r45809 | ||||
r45883 | def _extra_with_copies(repo, extra, files): | |||
r45809 | """encode copy information into a `extra` dictionnary""" | |||
r45883 | p1copies = files.copied_from_p1 | |||
p2copies = files.copied_from_p2 | ||||
filesadded = files.added | ||||
filesremoved = files.removed | ||||
files = sorted(files.touched) | ||||
r45811 | if not _write_copy_meta(repo)[1]: | |||
# If writing only to changeset extras, use None to indicate that | ||||
# no entry should be written. If writing to both, write an empty | ||||
# entry to prevent the reader from falling back to reading | ||||
# filelogs. | ||||
p1copies = p1copies or None | ||||
p2copies = p2copies or None | ||||
filesadded = filesadded or None | ||||
filesremoved = filesremoved or None | ||||
r45809 | extrasentries = p1copies, p2copies, filesadded, filesremoved | |||
if extra is None and any(x is not None for x in extrasentries): | ||||
extra = {} | ||||
if p1copies is not None: | ||||
p1copies = metadata.encodecopies(files, p1copies) | ||||
extra[b'p1copies'] = p1copies | ||||
if p2copies is not None: | ||||
p2copies = metadata.encodecopies(files, p2copies) | ||||
extra[b'p2copies'] = p2copies | ||||
if filesadded is not None: | ||||
filesadded = metadata.encodefileindices(files, filesadded) | ||||
extra[b'filesadded'] = filesadded | ||||
if filesremoved is not None: | ||||
filesremoved = metadata.encodefileindices(files, filesremoved) | ||||
extra[b'filesremoved'] = filesremoved | ||||
return extra | ||||