merge.py
2650 lines
| 95.7 KiB
| text/x-python
|
PythonLexer
/ mercurial / merge.py
Matt Mackall
|
r2775 | # merge.py - directory-level update/merge handling for Mercurial | ||
# | ||||
Raphaël Gomès
|
r47575 | # Copyright 2006, 2007 Olivia Mackall <olivia@selenic.com> | ||
Matt Mackall
|
r2775 | # | ||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Matt Mackall
|
r2775 | |||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Gregory Szorc
|
r25959 | |||
Pulkit Goyal
|
r45849 | import collections | ||
Raphaël Gomès
|
r52953 | import os | ||
Pierre-Yves David
|
r20590 | import struct | ||
Matt Harbison
|
r52622 | import typing | ||
Raphaël Gomès
|
r52951 | from typing import Dict, Optional, Tuple | ||
Pierre-Yves David
|
r20590 | |||
Gregory Szorc
|
r25959 | from .i18n import _ | ||
Joerg Sonnenberger
|
r47771 | from .node import nullrev | ||
Augie Fackler
|
r43346 | from .thirdparty import attr | ||
Matt Harbison
|
r52622 | |||
# Force pytype to use the non-vendored package | ||||
if typing.TYPE_CHECKING: | ||||
# noinspection PyPackageRequirements | ||||
import attr | ||||
Matt Harbison
|
r47519 | from .utils import stringutil | ||
Simon Sapin
|
r49079 | from .dirstateutils import timestamp | ||
Gregory Szorc
|
r25959 | from . import ( | ||
copies, | ||||
Matt Harbison
|
r39843 | encoding, | ||
Pierre-Yves David
|
r26587 | error, | ||
Gregory Szorc
|
r25959 | filemerge, | ||
Durham Goode
|
r31257 | match as matchmod, | ||
Augie Fackler
|
r45383 | mergestate as mergestatemod, | ||
r33147 | obsutil, | |||
Martin von Zweigbergk
|
r44032 | pathutil, | ||
Raphaël Gomès
|
r49660 | policy, | ||
Pulkit Goyal
|
r30519 | pycompat, | ||
Siddharth Agarwal
|
r27656 | scmutil, | ||
Yuya Nishihara
|
r36026 | subrepoutil, | ||
Gregory Szorc
|
r25959 | util, | ||
worker, | ||||
) | ||||
Matt Mackall
|
r6512 | |||
Raphaël Gomès
|
r52953 | rust_update_mod = policy.importrust("update") | ||
Pierre-Yves David
|
r20590 | _pack = struct.pack | ||
_unpack = struct.unpack | ||||
Augie Fackler
|
r43346 | |||
Siddharth Agarwal
|
r27740 | def _getcheckunknownconfig(repo, section, name): | ||
Boris Feld
|
r34523 | config = repo.ui.config(section, name) | ||
Augie Fackler
|
r43347 | valid = [b'abort', b'ignore', b'warn'] | ||
Siddharth Agarwal
|
r27740 | if config not in valid: | ||
Augie Fackler
|
r43347 | validstr = b', '.join([b"'" + v + b"'" for v in valid]) | ||
r49547 | msg = _(b"%s.%s not valid ('%s' is none of %s)") | |||
msg %= (section, name, config, validstr) | ||||
raise error.ConfigError(msg) | ||||
Siddharth Agarwal
|
r27740 | return config | ||
Augie Fackler
|
r43346 | |||
Arseniy Alekseyev
|
r50805 | def _checkunknownfile(dirstate, wvfs, dircache, wctx, mctx, f, f2=None): | ||
Phil Cohen
|
r35289 | if wctx.isinmemory(): | ||
# Nothing to do in IMM because nothing in the "working copy" can be an | ||||
# unknown file. | ||||
# | ||||
# Note that we should bail out here, not in ``_checkunknownfiles()``, | ||||
# because that function does other useful work. | ||||
return False | ||||
Martin von Zweigbergk
|
r23653 | if f2 is None: | ||
f2 = f | ||||
Augie Fackler
|
r43346 | return ( | ||
Arseniy Alekseyev
|
r50805 | wvfs.isfileorlink_checkdir(dircache, f) | ||
and dirstate.normalize(f) not in dirstate | ||||
Augie Fackler
|
r43346 | and mctx[f2].cmp(wctx[f]) | ||
) | ||||
Matt Mackall
|
r16093 | |||
Gregory Szorc
|
r49801 | class _unknowndirschecker: | ||
Mark Thomas
|
r34551 | """ | ||
Look for any unknown files or directories that may have a path conflict | ||||
with a file. If any path prefix of the file exists as a file or link, | ||||
then it conflicts. If the file itself is a directory that contains any | ||||
file that is not tracked, then it conflicts. | ||||
Returns the shortest path at which a conflict occurs, or None if there is | ||||
no conflict. | ||||
""" | ||||
Augie Fackler
|
r43346 | |||
Mark Thomas
|
r35181 | def __init__(self): | ||
# A set of paths known to be good. This prevents repeated checking of | ||||
# dirs. It will be updated with any new dirs that are checked and found | ||||
# to be safe. | ||||
self._unknowndircache = set() | ||||
Mark Thomas
|
r34551 | |||
Mark Thomas
|
r35181 | # A set of paths that are known to be absent. This prevents repeated | ||
# checking of subdirectories that are known not to exist. It will be | ||||
# updated with any new dirs that are checked and found to be absent. | ||||
self._missingdircache = set() | ||||
Mark Thomas
|
r34551 | |||
Phil Cohen
|
r35289 | def __call__(self, repo, wctx, f): | ||
if wctx.isinmemory(): | ||||
# Nothing to do in IMM for the same reason as ``_checkunknownfile``. | ||||
return False | ||||
Mark Thomas
|
r35181 | # Check for path prefixes that exist as unknown files. | ||
Martin von Zweigbergk
|
r44032 | for p in reversed(list(pathutil.finddirs(f))): | ||
Mark Thomas
|
r35181 | if p in self._missingdircache: | ||
return | ||||
if p in self._unknowndircache: | ||||
continue | ||||
if repo.wvfs.audit.check(p): | ||||
Augie Fackler
|
r43346 | if ( | ||
repo.wvfs.isfileorlink(p) | ||||
and repo.dirstate.normalize(p) not in repo.dirstate | ||||
): | ||||
Mark Thomas
|
r35181 | return p | ||
if not repo.wvfs.lexists(p): | ||||
self._missingdircache.add(p) | ||||
return | ||||
self._unknowndircache.add(p) | ||||
# Check if the file conflicts with a directory containing unknown files. | ||||
if repo.wvfs.audit.check(f) and repo.wvfs.isdir(f): | ||||
# Does the directory contain any files that are not in the dirstate? | ||||
for p, dirs, files in repo.wvfs.walk(f): | ||||
for fn in files: | ||||
Matt Harbison
|
r37104 | relf = util.pconvert(repo.wvfs.reljoin(p, fn)) | ||
Matt Harbison
|
r37105 | relf = repo.dirstate.normalize(relf, isknown=True) | ||
Mark Thomas
|
r35181 | if relf not in repo.dirstate: | ||
return f | ||||
return None | ||||
Mark Thomas
|
r34551 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r45844 | def _checkunknownfiles(repo, wctx, mctx, force, mresult, mergeforce): | ||
Martin von Zweigbergk
|
r23655 | """ | ||
Considers any actions that care about the presence of conflicting unknown | ||||
files. For some actions, the result is to abort; for others, it is to | ||||
choose a different action. | ||||
""" | ||||
Mark Thomas
|
r34552 | fileconflicts = set() | ||
Mark Thomas
|
r34553 | pathconflicts = set() | ||
Siddharth Agarwal
|
r28018 | warnconflicts = set() | ||
abortconflicts = set() | ||||
Augie Fackler
|
r43347 | unknownconfig = _getcheckunknownconfig(repo, b'merge', b'checkunknown') | ||
ignoredconfig = _getcheckunknownconfig(repo, b'merge', b'checkignored') | ||||
pathconfig = repo.ui.configbool( | ||||
b'experimental', b'merge.checkpathconflicts' | ||||
) | ||||
Arseniy Alekseyev
|
r50804 | dircache = dict() | ||
Arseniy Alekseyev
|
r50805 | dirstate = repo.dirstate | ||
wvfs = repo.wvfs | ||||
Raphaël Gomès
|
r52953 | # wouldn't it be easier to loop over unknown files (and dirs)? | ||
Martin von Zweigbergk
|
r23655 | if not force: | ||
Augie Fackler
|
r43346 | |||
Siddharth Agarwal
|
r27741 | def collectconflicts(conflicts, config): | ||
Augie Fackler
|
r43347 | if config == b'abort': | ||
Siddharth Agarwal
|
r27741 | abortconflicts.update(conflicts) | ||
Augie Fackler
|
r43347 | elif config == b'warn': | ||
Siddharth Agarwal
|
r27741 | warnconflicts.update(conflicts) | ||
Mark Thomas
|
r35181 | checkunknowndirs = _unknowndirschecker() | ||
Arseniy Alekseyev
|
r50804 | for f in mresult.files( | ||
( | ||||
mergestatemod.ACTION_CREATED, | ||||
mergestatemod.ACTION_DELETED_CHANGED, | ||||
) | ||||
): | ||||
Arseniy Alekseyev
|
r50805 | if _checkunknownfile(dirstate, wvfs, dircache, wctx, mctx, f): | ||
Arseniy Alekseyev
|
r50804 | fileconflicts.add(f) | ||
elif pathconfig and f not in wctx: | ||||
path = checkunknowndirs(repo, wctx, f) | ||||
if path is not None: | ||||
pathconflicts.add(path) | ||||
Pulkit Goyal
|
r45850 | for f, args, msg in mresult.getactions( | ||
[mergestatemod.ACTION_LOCAL_DIR_RENAME_GET] | ||||
): | ||||
Arseniy Alekseyev
|
r50805 | if _checkunknownfile( | ||
dirstate, wvfs, dircache, wctx, mctx, f, args[0] | ||||
): | ||||
Pulkit Goyal
|
r45850 | fileconflicts.add(f) | ||
Martin von Zweigbergk
|
r23655 | |||
Mark Thomas
|
r34553 | allconflicts = fileconflicts | pathconflicts | ||
Augie Fackler
|
r43346 | ignoredconflicts = {c for c in allconflicts if repo.dirstate._ignore(c)} | ||
Mark Thomas
|
r34552 | unknownconflicts = allconflicts - ignoredconflicts | ||
Siddharth Agarwal
|
r27742 | collectconflicts(ignoredconflicts, ignoredconfig) | ||
collectconflicts(unknownconflicts, unknownconfig) | ||||
Siddharth Agarwal
|
r28022 | else: | ||
Pulkit Goyal
|
r45900 | for f, args, msg in list( | ||
mresult.getactions([mergestatemod.ACTION_CREATED_MERGE]) | ||||
Pulkit Goyal
|
r45850 | ): | ||
fl2, anc = args | ||||
Arseniy Alekseyev
|
r50805 | different = _checkunknownfile( | ||
dirstate, wvfs, dircache, wctx, mctx, f | ||||
) | ||||
Pulkit Goyal
|
r45850 | if repo.dirstate._ignore(f): | ||
config = ignoredconfig | ||||
else: | ||||
config = unknownconfig | ||||
Siddharth Agarwal
|
r28022 | |||
Pulkit Goyal
|
r45850 | # The behavior when force is True is described by this table: | ||
# config different mergeforce | action backup | ||||
# * n * | get n | ||||
# * y y | merge - | ||||
# abort y n | merge - (1) | ||||
# warn y n | warn + get y | ||||
# ignore y n | get y | ||||
# | ||||
# (1) this is probably the wrong behavior here -- we should | ||||
# probably abort, but some actions like rebases currently | ||||
# don't like an abort happening in the middle of | ||||
# merge.update. | ||||
if not different: | ||||
mresult.addfile( | ||||
f, | ||||
mergestatemod.ACTION_GET, | ||||
(fl2, False), | ||||
b'remote created', | ||||
) | ||||
elif mergeforce or config == b'abort': | ||||
mresult.addfile( | ||||
f, | ||||
mergestatemod.ACTION_MERGE, | ||||
(f, f, None, False, anc), | ||||
b'remote differs from untracked local', | ||||
) | ||||
elif config == b'abort': | ||||
abortconflicts.add(f) | ||||
else: | ||||
if config == b'warn': | ||||
warnconflicts.add(f) | ||||
mresult.addfile( | ||||
Augie Fackler
|
r46554 | f, | ||
mergestatemod.ACTION_GET, | ||||
(fl2, True), | ||||
b'remote created', | ||||
Pulkit Goyal
|
r45850 | ) | ||
Siddharth Agarwal
|
r27741 | |||
Siddharth Agarwal
|
r28018 | for f in sorted(abortconflicts): | ||
Mark Thomas
|
r34554 | warn = repo.ui.warn | ||
if f in pathconflicts: | ||||
if repo.wvfs.isfileorlink(f): | ||||
Augie Fackler
|
r43347 | warn(_(b"%s: untracked file conflicts with directory\n") % f) | ||
Mark Thomas
|
r34554 | else: | ||
Augie Fackler
|
r43347 | warn(_(b"%s: untracked directory conflicts with file\n") % f) | ||
Mark Thomas
|
r34554 | else: | ||
Augie Fackler
|
r43347 | warn(_(b"%s: untracked file differs\n") % f) | ||
Siddharth Agarwal
|
r28018 | if abortconflicts: | ||
Martin von Zweigbergk
|
r47146 | raise error.StateError( | ||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b"untracked files in working directory " | ||
b"differ from files in requested revision" | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Siddharth Agarwal
|
r28018 | |||
for f in sorted(warnconflicts): | ||||
Mark Thomas
|
r34554 | if repo.wvfs.isfileorlink(f): | ||
Augie Fackler
|
r43347 | repo.ui.warn(_(b"%s: replacing untracked file\n") % f) | ||
Mark Thomas
|
r34554 | else: | ||
Augie Fackler
|
r43347 | repo.ui.warn(_(b"%s: replacing untracked files in directory\n") % f) | ||
Martin von Zweigbergk
|
r23655 | |||
Arseniy Alekseyev
|
r50806 | def transformargs(f, args): | ||
Pulkit Goyal
|
r45850 | backup = ( | ||
f in fileconflicts | ||||
Arseniy Alekseyev
|
r50781 | or pathconflicts | ||
and ( | ||||
f in pathconflicts | ||||
or any(p in pathconflicts for p in pathutil.finddirs(f)) | ||||
) | ||||
Pulkit Goyal
|
r45850 | ) | ||
(flags,) = args | ||||
Arseniy Alekseyev
|
r50806 | return (flags, backup) | ||
mresult.mapaction( | ||||
mergestatemod.ACTION_CREATED, mergestatemod.ACTION_GET, transformargs | ||||
) | ||||
Martin von Zweigbergk
|
r23655 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r45907 | def _forgetremoved(wctx, mctx, branchmerge, mresult): | ||
Matt Mackall
|
r3107 | """ | ||
Forget removed files | ||||
If we're jumping between revisions (as opposed to merging), and if | ||||
neither the working directory nor the target rev has the file, | ||||
then we need to remove it from the dirstate, to prevent the | ||||
dirstate from listing the file when it is no longer in the | ||||
manifest. | ||||
Alexis S. L. Carvalho
|
r6242 | |||
If we're merging, and the other revision has removed a file | ||||
that is not present in the working directory, we need to mark it | ||||
as removed. | ||||
Matt Mackall
|
r3107 | """ | ||
Augie Fackler
|
r45383 | m = mergestatemod.ACTION_FORGET | ||
Mads Kiilerich
|
r21545 | if branchmerge: | ||
Augie Fackler
|
r45383 | m = mergestatemod.ACTION_REMOVE | ||
Alexis S. L. Carvalho
|
r6242 | for f in wctx.deleted(): | ||
Matt Mackall
|
r6272 | if f not in mctx: | ||
Pulkit Goyal
|
r45907 | mresult.addfile(f, m, None, b"forget deleted") | ||
Alexis S. L. Carvalho
|
r6242 | |||
if not branchmerge: | ||||
for f in wctx.removed(): | ||||
Matt Mackall
|
r6272 | if f not in mctx: | ||
Pulkit Goyal
|
r45907 | mresult.addfile( | ||
Augie Fackler
|
r46554 | f, | ||
mergestatemod.ACTION_FORGET, | ||||
None, | ||||
b"forget removed", | ||||
Augie Fackler
|
r45383 | ) | ||
Matt Mackall
|
r3107 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r45893 | def _checkcollision(repo, wmf, mresult): | ||
Martin von Zweigbergk
|
r38056 | """ | ||
Check for case-folding collisions. | ||||
""" | ||||
# If the repo is narrowed, filter out files outside the narrowspec. | ||||
narrowmatch = repo.narrowmatch() | ||||
if not narrowmatch.always(): | ||||
Augie Fackler
|
r44769 | pmmf = set(wmf.walk(narrowmatch)) | ||
Pulkit Goyal
|
r45893 | if mresult: | ||
Pulkit Goyal
|
r45905 | for f in list(mresult.files()): | ||
Pulkit Goyal
|
r45893 | if not narrowmatch(f): | ||
mresult.removefile(f) | ||||
Augie Fackler
|
r44769 | else: | ||
# build provisional merged manifest up | ||||
pmmf = set(wmf) | ||||
FUJIWARA Katsunori
|
r19105 | |||
Pulkit Goyal
|
r45893 | if mresult: | ||
Gregory Szorc
|
r37130 | # KEEP and EXEC are no-op | ||
Pulkit Goyal
|
r45905 | for f in mresult.files( | ||
Pulkit Goyal
|
r45893 | ( | ||
mergestatemod.ACTION_ADD, | ||||
mergestatemod.ACTION_ADD_MODIFIED, | ||||
mergestatemod.ACTION_FORGET, | ||||
mergestatemod.ACTION_GET, | ||||
mergestatemod.ACTION_CHANGED_DELETED, | ||||
mergestatemod.ACTION_DELETED_CHANGED, | ||||
) | ||||
Augie Fackler
|
r43346 | ): | ||
Pulkit Goyal
|
r45893 | pmmf.add(f) | ||
Pulkit Goyal
|
r45905 | for f in mresult.files((mergestatemod.ACTION_REMOVE,)): | ||
Mads Kiilerich
|
r21545 | pmmf.discard(f) | ||
Pulkit Goyal
|
r45893 | for f, args, msg in mresult.getactions( | ||
[mergestatemod.ACTION_DIR_RENAME_MOVE_LOCAL] | ||||
): | ||||
Mads Kiilerich
|
r21545 | f2, flags = args | ||
pmmf.discard(f2) | ||||
pmmf.add(f) | ||||
Pulkit Goyal
|
r45905 | for f in mresult.files((mergestatemod.ACTION_LOCAL_DIR_RENAME_GET,)): | ||
Mads Kiilerich
|
r21545 | pmmf.add(f) | ||
Pulkit Goyal
|
r45893 | for f, args, msg in mresult.getactions([mergestatemod.ACTION_MERGE]): | ||
Mads Kiilerich
|
r21545 | f1, f2, fa, move, anc = args | ||
if move: | ||||
pmmf.discard(f1) | ||||
pmmf.add(f) | ||||
FUJIWARA Katsunori
|
r19105 | |||
# check case-folding collision in provisional merged manifest | ||||
foldmap = {} | ||||
Alex Gaynor
|
r33807 | for f in pmmf: | ||
FUJIWARA Katsunori
|
r19105 | fold = util.normcase(f) | ||
if fold in foldmap: | ||||
r49548 | msg = _(b"case-folding collision between %s and %s") | |||
msg %= (f, foldmap[fold]) | ||||
raise error.StateError(msg) | ||||
FUJIWARA Katsunori
|
r19105 | foldmap[fold] = f | ||
Mads Kiilerich
|
r26661 | # check case-folding of directories | ||
Augie Fackler
|
r43347 | foldprefix = unfoldprefix = lastfull = b'' | ||
Mads Kiilerich
|
r26661 | for fold, f in sorted(foldmap.items()): | ||
if fold.startswith(foldprefix) and not f.startswith(unfoldprefix): | ||||
# the folded prefix matches but actual casing is different | ||||
r49549 | msg = _(b"case-folding collision between %s and directory of %s") | |||
msg %= (lastfull, f) | ||||
raise error.StateError(msg) | ||||
Augie Fackler
|
r43347 | foldprefix = fold + b'/' | ||
unfoldprefix = f + b'/' | ||||
Mads Kiilerich
|
r26661 | lastfull = f | ||
Augie Fackler
|
r43346 | |||
Mark Thomas
|
r34556 | def _filesindirs(repo, manifest, dirs): | ||
""" | ||||
Generator that yields pairs of all the files in the manifest that are found | ||||
inside the directories listed in dirs, and which directory they are found | ||||
in. | ||||
""" | ||||
for f in manifest: | ||||
Martin von Zweigbergk
|
r44032 | for p in pathutil.finddirs(f): | ||
Mark Thomas
|
r34556 | if p in dirs: | ||
yield f, p | ||||
break | ||||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r45841 | def checkpathconflicts(repo, wctx, mctx, mresult): | ||
Mark Thomas
|
r34556 | """ | ||
Check if any actions introduce path conflicts in the repository, updating | ||||
actions to record or handle the path conflict accordingly. | ||||
""" | ||||
mf = wctx.manifest() | ||||
# The set of local files that conflict with a remote directory. | ||||
localconflicts = set() | ||||
# The set of directories that conflict with a remote file, and so may cause | ||||
# conflicts if they still contain any files after the merge. | ||||
remoteconflicts = set() | ||||
# The set of directories that appear as both a file and a directory in the | ||||
# remote manifest. These indicate an invalid remote manifest, which | ||||
# can't be updated to cleanly. | ||||
invalidconflicts = set() | ||||
Mark Thomas
|
r35182 | # The set of directories that contain files that are being created. | ||
createdfiledirs = set() | ||||
Mark Thomas
|
r34556 | # The set of files deleted by all the actions. | ||
deletedfiles = set() | ||||
Pulkit Goyal
|
r45905 | for f in mresult.files( | ||
Pulkit Goyal
|
r45902 | ( | ||
Augie Fackler
|
r45383 | mergestatemod.ACTION_CREATED, | ||
mergestatemod.ACTION_DELETED_CHANGED, | ||||
mergestatemod.ACTION_MERGE, | ||||
mergestatemod.ACTION_CREATED_MERGE, | ||||
Pulkit Goyal
|
r45902 | ) | ||
): | ||||
# This action may create a new local file. | ||||
createdfiledirs.update(pathutil.finddirs(f)) | ||||
if mf.hasdir(f): | ||||
# The file aliases a local directory. This might be ok if all | ||||
# the files in the local directory are being deleted. This | ||||
# will be checked once we know what all the deleted files are. | ||||
remoteconflicts.add(f) | ||||
# Track the names of all deleted files. | ||||
Pulkit Goyal
|
r45905 | for f in mresult.files((mergestatemod.ACTION_REMOVE,)): | ||
Pulkit Goyal
|
r45902 | deletedfiles.add(f) | ||
Raphaël Gomès
|
r52596 | for f, args, msg in mresult.getactions((mergestatemod.ACTION_MERGE,)): | ||
Pulkit Goyal
|
r45902 | f1, f2, fa, move, anc = args | ||
if move: | ||||
deletedfiles.add(f1) | ||||
Raphaël Gomès
|
r52596 | for f, args, msg in mresult.getactions( | ||
Pulkit Goyal
|
r45902 | (mergestatemod.ACTION_DIR_RENAME_MOVE_LOCAL,) | ||
): | ||||
f2, flags = args | ||||
deletedfiles.add(f2) | ||||
Mark Thomas
|
r34556 | |||
Mark Thomas
|
r35182 | # Check all directories that contain created files for path conflicts. | ||
for p in createdfiledirs: | ||||
if p in mf: | ||||
if p in mctx: | ||||
# A file is in a directory which aliases both a local | ||||
# and a remote file. This is an internal inconsistency | ||||
# within the remote manifest. | ||||
invalidconflicts.add(p) | ||||
else: | ||||
# A file is in a directory which aliases a local file. | ||||
# We will need to rename the local file. | ||||
localconflicts.add(p) | ||||
Pulkit Goyal
|
r45904 | pd = mresult.getfile(p) | ||
if pd and pd[0] in ( | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_CREATED, | ||
mergestatemod.ACTION_DELETED_CHANGED, | ||||
mergestatemod.ACTION_MERGE, | ||||
mergestatemod.ACTION_CREATED_MERGE, | ||||
Augie Fackler
|
r43346 | ): | ||
Mark Thomas
|
r35182 | # The file is in a directory which aliases a remote file. | ||
# This is an internal inconsistency within the remote | ||||
# manifest. | ||||
invalidconflicts.add(p) | ||||
Mark Thomas
|
r34556 | # Rename all local conflicting files that have not been deleted. | ||
for p in localconflicts: | ||||
if p not in deletedfiles: | ||||
Augie Fackler
|
r43347 | ctxname = bytes(wctx).rstrip(b'+') | ||
Pulkit Goyal
|
r45905 | pnew = util.safename(p, ctxname, wctx, set(mresult.files())) | ||
Martin von Zweigbergk
|
r45465 | porig = wctx[p].copysource() or p | ||
Pulkit Goyal
|
r45841 | mresult.addfile( | ||
pnew, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_PATH_CONFLICT_RESOLVE, | ||
Martin von Zweigbergk
|
r45465 | (p, porig), | ||
Augie Fackler
|
r43347 | b'local path conflict', | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r45841 | mresult.addfile( | ||
p, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_PATH_CONFLICT, | ||
(pnew, b'l'), | ||||
b'path conflict', | ||||
) | ||||
Mark Thomas
|
r34556 | |||
if remoteconflicts: | ||||
# Check if all files in the conflicting directories have been removed. | ||||
Augie Fackler
|
r43347 | ctxname = bytes(mctx).rstrip(b'+') | ||
Mark Thomas
|
r34556 | for f, p in _filesindirs(repo, mf, remoteconflicts): | ||
if f not in deletedfiles: | ||||
Pulkit Goyal
|
r45904 | m, args, msg = mresult.getfile(p) | ||
Pulkit Goyal
|
r45905 | pnew = util.safename(p, ctxname, wctx, set(mresult.files())) | ||
Augie Fackler
|
r45383 | if m in ( | ||
mergestatemod.ACTION_DELETED_CHANGED, | ||||
mergestatemod.ACTION_MERGE, | ||||
): | ||||
Mark Thomas
|
r34556 | # Action was merge, just update target. | ||
Pulkit Goyal
|
r45841 | mresult.addfile(pnew, m, args, msg) | ||
Mark Thomas
|
r34556 | else: | ||
# Action was create, change to renamed get action. | ||||
fl = args[0] | ||||
Pulkit Goyal
|
r45841 | mresult.addfile( | ||
pnew, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_LOCAL_DIR_RENAME_GET, | ||
Augie Fackler
|
r43346 | (p, fl), | ||
Augie Fackler
|
r43347 | b'remote path conflict', | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r45841 | mresult.addfile( | ||
p, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_PATH_CONFLICT, | ||
r49557 | (pnew, b'r'), | |||
Augie Fackler
|
r43347 | b'path conflict', | ||
Augie Fackler
|
r43346 | ) | ||
Mark Thomas
|
r34556 | remoteconflicts.remove(p) | ||
break | ||||
if invalidconflicts: | ||||
for p in invalidconflicts: | ||||
Augie Fackler
|
r43347 | repo.ui.warn(_(b"%s: is both a file and a directory\n") % p) | ||
Martin von Zweigbergk
|
r47146 | raise error.StateError( | ||
_(b"destination manifest contains path conflicts") | ||||
) | ||||
Mark Thomas
|
r34556 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r45842 | def _filternarrowactions(narrowmatch, branchmerge, mresult): | ||
Martin von Zweigbergk
|
r38055 | """ | ||
Filters out actions that can ignored because the repo is narrowed. | ||||
Raise an exception if the merge cannot be completed because the repo is | ||||
narrowed. | ||||
""" | ||||
# We mutate the items in the dict during iteration, so iterate | ||||
# over a copy. | ||||
Manuel Jacob
|
r52711 | for f, action in list(mresult.filemap()): | ||
Martin von Zweigbergk
|
r38055 | if narrowmatch(f): | ||
pass | ||||
elif not branchmerge: | ||||
Pulkit Goyal
|
r45842 | mresult.removefile(f) # just updating, ignore changes outside clone | ||
r49562 | elif action[0].no_op: | |||
Pulkit Goyal
|
r45842 | mresult.removefile(f) # merge does not affect file | ||
r49565 | elif action[0].narrow_safe: | |||
r49592 | if not f.endswith(b'/'): | |||
r49565 | mresult.removefile(f) # merge won't affect on-disk files | |||
mresult.addcommitinfo( | ||||
f, b'outside-narrow-merge-action', action[0].changes | ||||
) | ||||
else: # TODO: handle the tree case | ||||
msg = _( | ||||
b'merge affects file \'%s\' outside narrow, ' | ||||
b'which is not yet supported' | ||||
) | ||||
hint = _(b'merging in the other direction may work') | ||||
raise error.Abort(msg % f, hint=hint) | ||||
Martin von Zweigbergk
|
r38055 | else: | ||
r49551 | msg = _(b'conflict in file \'%s\' is outside narrow clone') | |||
raise error.StateError(msg % f) | ||||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r38055 | |||
Gregory Szorc
|
r49801 | class mergeresult: | ||
Augie Fackler
|
r46554 | """An object representing result of merging manifests. | ||
Pulkit Goyal
|
r45831 | |||
It has information about what actions need to be performed on dirstate | ||||
Augie Fackler
|
r46554 | mapping of divergent renames and other such cases.""" | ||
Pulkit Goyal
|
r45831 | |||
Pulkit Goyal
|
r45838 | def __init__(self): | ||
Pulkit Goyal
|
r45831 | """ | ||
Pulkit Goyal
|
r45848 | filemapping: dict of filename as keys and action related info as values | ||
Pulkit Goyal
|
r45831 | diverge: mapping of source name -> list of dest name for | ||
divergent renames | ||||
renamedelete: mapping of source name -> list of destinations for files | ||||
deleted on one side and renamed on other. | ||||
Pulkit Goyal
|
r45832 | commitinfo: dict containing data which should be used on commit | ||
contains a filename -> info mapping | ||||
Pulkit Goyal
|
r45856 | actionmapping: dict of action names as keys and values are dict of | ||
filename as key and related data as values | ||||
Pulkit Goyal
|
r45831 | """ | ||
Pulkit Goyal
|
r45848 | self._filemapping = {} | ||
Pulkit Goyal
|
r45838 | self._diverge = {} | ||
self._renamedelete = {} | ||||
Pulkit Goyal
|
r45943 | self._commitinfo = collections.defaultdict(dict) | ||
Pulkit Goyal
|
r45856 | self._actionmapping = collections.defaultdict(dict) | ||
Pulkit Goyal
|
r45831 | |||
Pulkit Goyal
|
r45944 | def updatevalues(self, diverge, renamedelete): | ||
Pulkit Goyal
|
r45831 | self._diverge = diverge | ||
self._renamedelete = renamedelete | ||||
Pulkit Goyal
|
r45839 | def addfile(self, filename, action, data, message): | ||
Augie Fackler
|
r46554 | """adds a new file to the mergeresult object | ||
Pulkit Goyal
|
r45839 | |||
filename: file which we are adding | ||||
action: one of mergestatemod.ACTION_* | ||||
data: a tuple of information like fctx and ctx related to this merge | ||||
message: a message about the merge | ||||
""" | ||||
Pulkit Goyal
|
r45849 | # if the file already existed, we need to delete it's old | ||
# entry form _actionmapping too | ||||
if filename in self._filemapping: | ||||
a, d, m = self._filemapping[filename] | ||||
Pulkit Goyal
|
r45856 | del self._actionmapping[a][filename] | ||
Pulkit Goyal
|
r45849 | |||
Pulkit Goyal
|
r45848 | self._filemapping[filename] = (action, data, message) | ||
Pulkit Goyal
|
r45856 | self._actionmapping[action][filename] = (data, message) | ||
Pulkit Goyal
|
r45839 | |||
Arseniy Alekseyev
|
r50806 | def mapaction(self, actionfrom, actionto, transform): | ||
"""changes all occurrences of action `actionfrom` into `actionto`, | ||||
transforming its args with the function `transform`. | ||||
""" | ||||
orig = self._actionmapping[actionfrom] | ||||
del self._actionmapping[actionfrom] | ||||
dest = self._actionmapping[actionto] | ||||
for f, (data, msg) in orig.items(): | ||||
data = transform(f, data) | ||||
self._filemapping[f] = (actionto, data, msg) | ||||
dest[f] = (data, msg) | ||||
Pulkit Goyal
|
r45904 | def getfile(self, filename, default_return=None): | ||
Augie Fackler
|
r46554 | """returns (action, args, msg) about this file | ||
Pulkit Goyal
|
r45904 | |||
Augie Fackler
|
r46554 | returns default_return if the file is not present""" | ||
Pulkit Goyal
|
r45904 | if filename in self._filemapping: | ||
return self._filemapping[filename] | ||||
return default_return | ||||
Pulkit Goyal
|
r45905 | def files(self, actions=None): | ||
Augie Fackler
|
r46554 | """returns files on which provided action needs to perfromed | ||
Pulkit Goyal
|
r45905 | |||
If actions is None, all files are returned | ||||
""" | ||||
# TODO: think whether we should return renamedelete and | ||||
# diverge filenames also | ||||
if actions is None: | ||||
for f in self._filemapping: | ||||
yield f | ||||
else: | ||||
for a in actions: | ||||
for f in self._actionmapping[a]: | ||||
yield f | ||||
Pulkit Goyal
|
r45840 | def removefile(self, filename): | ||
Augie Fackler
|
r46554 | """removes a file from the mergeresult object as the file might | ||
not merging anymore""" | ||||
Pulkit Goyal
|
r45849 | action, data, message = self._filemapping[filename] | ||
Pulkit Goyal
|
r45848 | del self._filemapping[filename] | ||
Pulkit Goyal
|
r45856 | del self._actionmapping[action][filename] | ||
Pulkit Goyal
|
r45849 | |||
Pulkit Goyal
|
r45896 | def getactions(self, actions, sort=False): | ||
Augie Fackler
|
r46554 | """get list of files which are marked with these actions | ||
Pulkit Goyal
|
r45896 | if sort is true, files for each action is sorted and then added | ||
Pulkit Goyal
|
r45849 | |||
Returns a list of tuple of form (filename, data, message) | ||||
""" | ||||
for a in actions: | ||||
Pulkit Goyal
|
r45896 | if sort: | ||
for f in sorted(self._actionmapping[a]): | ||||
args, msg = self._actionmapping[a][f] | ||||
Pulkit Goyal
|
r45900 | yield f, args, msg | ||
Pulkit Goyal
|
r45896 | else: | ||
Gregory Szorc
|
r49782 | for f, (args, msg) in self._actionmapping[a].items(): | ||
Pulkit Goyal
|
r45900 | yield f, args, msg | ||
Pulkit Goyal
|
r45840 | |||
Pulkit Goyal
|
r45898 | def len(self, actions=None): | ||
Augie Fackler
|
r46554 | """returns number of files which needs actions | ||
Pulkit Goyal
|
r45898 | |||
if actions is passed, total of number of files in that action | ||||
Augie Fackler
|
r46554 | only is returned""" | ||
Pulkit Goyal
|
r45898 | |||
if actions is None: | ||||
return len(self._filemapping) | ||||
return sum(len(self._actionmapping[a]) for a in actions) | ||||
Pulkit Goyal
|
r45906 | def filemap(self, sort=False): | ||
Manuel Jacob
|
r52711 | if sort: | ||
Gregory Szorc
|
r49768 | for key, val in sorted(self._filemapping.items()): | ||
Pulkit Goyal
|
r45906 | yield key, val | ||
else: | ||||
Gregory Szorc
|
r49768 | for key, val in self._filemapping.items(): | ||
Pulkit Goyal
|
r45906 | yield key, val | ||
Pulkit Goyal
|
r45831 | |||
Pulkit Goyal
|
r45944 | def addcommitinfo(self, filename, key, value): | ||
Augie Fackler
|
r46554 | """adds key-value information about filename which will be required | ||
while committing this merge""" | ||||
Pulkit Goyal
|
r45944 | self._commitinfo[filename][key] = value | ||
Pulkit Goyal
|
r45831 | @property | ||
def diverge(self): | ||||
return self._diverge | ||||
@property | ||||
def renamedelete(self): | ||||
return self._renamedelete | ||||
Pulkit Goyal
|
r45832 | @property | ||
def commitinfo(self): | ||||
return self._commitinfo | ||||
Pulkit Goyal
|
r45836 | @property | ||
def actionsdict(self): | ||||
Augie Fackler
|
r46554 | """returns a dictionary of actions to be perfomed with action as key | ||
and a list of files and related arguments as values""" | ||||
Pulkit Goyal
|
r45908 | res = collections.defaultdict(list) | ||
Gregory Szorc
|
r49768 | for a, d in self._actionmapping.items(): | ||
for f, (args, msg) in d.items(): | ||||
Pulkit Goyal
|
r45856 | res[a].append((f, args, msg)) | ||
return res | ||||
Pulkit Goyal
|
r45836 | |||
Pulkit Goyal
|
r45831 | def setactions(self, actions): | ||
Pulkit Goyal
|
r45848 | self._filemapping = actions | ||
Pulkit Goyal
|
r45856 | self._actionmapping = collections.defaultdict(dict) | ||
Gregory Szorc
|
r49768 | for f, (act, data, msg) in self._filemapping.items(): | ||
Pulkit Goyal
|
r45856 | self._actionmapping[act][f] = data, msg | ||
Pulkit Goyal
|
r45831 | |||
Pulkit Goyal
|
r45835 | def hasconflicts(self): | ||
Augie Fackler
|
r46554 | """tells whether this merge resulted in some actions which can | ||
result in conflicts or not""" | ||||
Pulkit Goyal
|
r45850 | for a in self._actionmapping.keys(): | ||
if ( | ||||
a | ||||
not in ( | ||||
mergestatemod.ACTION_GET, | ||||
mergestatemod.ACTION_EXEC, | ||||
mergestatemod.ACTION_REMOVE, | ||||
mergestatemod.ACTION_PATH_CONFLICT_RESOLVE, | ||||
) | ||||
and self._actionmapping[a] | ||||
r49562 | and not a.no_op | |||
Pulkit Goyal
|
r45835 | ): | ||
return True | ||||
return False | ||||
Pulkit Goyal
|
r45831 | |||
Augie Fackler
|
r43346 | def manifestmerge( | ||
repo, | ||||
wctx, | ||||
p2, | ||||
pa, | ||||
branchmerge, | ||||
force, | ||||
matcher, | ||||
acceptremote, | ||||
followcopies, | ||||
forcefulldiff=False, | ||||
): | ||||
Matt Mackall
|
r3105 | """ | ||
Yuya Nishihara
|
r30096 | Merge wctx and p2 with ancestor pa and generate merge action list | ||
Matt Mackall
|
r3315 | |||
Siddharth Agarwal
|
r18605 | branchmerge and force are as passed in to update | ||
Augie Fackler
|
r27346 | matcher = matcher to filter file lists | ||
Durham Goode
|
r18778 | acceptremote = accept the incoming changes without prompting | ||
Pulkit Goyal
|
r45728 | |||
Pulkit Goyal
|
r45831 | Returns an object of mergeresult class | ||
Matt Mackall
|
r3105 | """ | ||
Pulkit Goyal
|
r45839 | mresult = mergeresult() | ||
Augie Fackler
|
r27346 | if matcher is not None and matcher.always(): | ||
matcher = None | ||||
Matt Mackall
|
r3105 | |||
Siddharth Agarwal
|
r18651 | # manifests fetched in order are going to be faster, so prime the caches | ||
Augie Fackler
|
r43346 | [ | ||
x.manifest() | ||||
for x in sorted(wctx.parents() + [p2, pa], key=scmutil.intrev) | ||||
] | ||||
Siddharth Agarwal
|
r18651 | |||
Martin von Zweigbergk
|
r44682 | branch_copies1 = copies.branch_copies() | ||
branch_copies2 = copies.branch_copies() | ||||
diverge = {} | ||||
Pulkit Goyal
|
r45832 | # information from merge which is needed at commit time | ||
# for example choosing filelog of which parent to commit | ||||
# TODO: use specific constants in future for this mapping | ||||
Siddharth Agarwal
|
r18651 | if followcopies: | ||
Martin von Zweigbergk
|
r44682 | branch_copies1, branch_copies2, diverge = copies.mergecopies( | ||
repo, wctx, p2, pa | ||||
) | ||||
Matt Mackall
|
r8753 | |||
Pulkit Goyal
|
r32641 | boolbm = pycompat.bytestr(bool(branchmerge)) | ||
boolf = pycompat.bytestr(bool(force)) | ||||
boolm = pycompat.bytestr(bool(matcher)) | ||||
Augie Fackler
|
r43347 | repo.ui.note(_(b"resolving manifests\n")) | ||
Augie Fackler
|
r43346 | repo.ui.debug( | ||
Augie Fackler
|
r43347 | b" branchmerge: %s, force: %s, partial: %s\n" % (boolbm, boolf, boolm) | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | repo.ui.debug(b" ancestor: %s, local: %s, remote: %s\n" % (pa, wctx, p2)) | ||
Matt Mackall
|
r8753 | |||
Bryan O'Sullivan
|
r18611 | m1, m2, ma = wctx.manifest(), p2.manifest(), pa.manifest() | ||
Martin von Zweigbergk
|
r44682 | copied1 = set(branch_copies1.copy.values()) | ||
copied1.update(branch_copies1.movewithdir.values()) | ||||
copied2 = set(branch_copies2.copy.values()) | ||||
copied2.update(branch_copies2.movewithdir.values()) | ||||
Matt Mackall
|
r3295 | |||
Augie Fackler
|
r43347 | if b'.hgsubstate' in m1 and wctx.rev() is None: | ||
Yuya Nishihara
|
r38444 | # Check whether sub state is modified, and overwrite the manifest | ||
# to flag the change. If wctx is a committed revision, we shouldn't | ||||
# care for the dirty state of the working directory. | ||||
Martin von Zweigbergk
|
r28226 | if any(wctx.sub(s).dirty() for s in wctx.substate): | ||
Joerg Sonnenberger
|
r47771 | m1[b'.hgsubstate'] = repo.nodeconstants.modifiednodeid | ||
Matt Mackall
|
r9783 | |||
Durham Goode
|
r32151 | # Don't use m2-vs-ma optimization if: | ||
# - ma is the same as m1 or m2, which we're just going to diff again later | ||||
# - The caller specifically asks for a full diff, which is useful during bid | ||||
# merge. | ||||
Pulkit Goyal
|
r46154 | # - we are tracking salvaged files specifically hence should process all | ||
# files | ||||
if ( | ||||
pa not in ([wctx, p2] + wctx.parents()) | ||||
and not forcefulldiff | ||||
r46253 | and not ( | |||
repo.ui.configbool(b'experimental', b'merge-track-salvaged') | ||||
or repo.filecopiesmode == b'changeset-sidedata' | ||||
) | ||||
Pulkit Goyal
|
r46154 | ): | ||
Durham Goode
|
r32151 | # Identify which files are relevant to the merge, so we can limit the | ||
# total m1-vs-m2 diff to just those files. This has significant | ||||
# performance benefits in large repositories. | ||||
relevantfiles = set(ma.diff(m2).keys()) | ||||
# For copied and moved files, we need to add the source file too. | ||||
Gregory Szorc
|
r49768 | for copykey, copyvalue in branch_copies1.copy.items(): | ||
Durham Goode
|
r32151 | if copyvalue in relevantfiles: | ||
relevantfiles.add(copykey) | ||||
Martin von Zweigbergk
|
r44682 | for movedirkey in branch_copies1.movewithdir: | ||
Durham Goode
|
r32151 | relevantfiles.add(movedirkey) | ||
Martin von Zweigbergk
|
r32498 | filesmatcher = scmutil.matchfiles(repo, relevantfiles) | ||
matcher = matchmod.intersectmatchers(matcher, filesmatcher) | ||||
Durham Goode
|
r32151 | |||
Durham Goode
|
r31257 | diff = m1.diff(m2, match=matcher) | ||
Gregory Szorc
|
r49768 | for f, ((n1, fl1), (n2, fl2)) in diff.items(): | ||
Augie Fackler
|
r43346 | if n1 and n2: # file exists on both local and remote side | ||
Martin von Zweigbergk
|
r23396 | if f not in ma: | ||
Martin von Zweigbergk
|
r44682 | # TODO: what if they're renamed from different sources? | ||
fa = branch_copies1.copy.get( | ||||
f, None | ||||
) or branch_copies2.copy.get(f, None) | ||||
Pulkit Goyal
|
r45853 | args, msg = None, None | ||
Martin von Zweigbergk
|
r23397 | if fa is not None: | ||
Pulkit Goyal
|
r45853 | args = (f, f, fa, False, pa.node()) | ||
msg = b'both renamed from %s' % fa | ||||
Mads Kiilerich
|
r18338 | else: | ||
Pulkit Goyal
|
r45853 | args = (f, f, None, False, pa.node()) | ||
msg = b'both created' | ||||
mresult.addfile(f, mergestatemod.ACTION_MERGE, args, msg) | ||||
Martin von Zweigbergk
|
r46421 | elif f in branch_copies1.copy: | ||
fa = branch_copies1.copy[f] | ||||
mresult.addfile( | ||||
f, | ||||
mergestatemod.ACTION_MERGE, | ||||
(f, fa, fa, False, pa.node()), | ||||
b'local replaced from %s' % fa, | ||||
) | ||||
elif f in branch_copies2.copy: | ||||
fa = branch_copies2.copy[f] | ||||
mresult.addfile( | ||||
f, | ||||
mergestatemod.ACTION_MERGE, | ||||
(fa, f, fa, False, pa.node()), | ||||
b'other replaced from %s' % fa, | ||||
) | ||||
Matt Mackall
|
r16094 | else: | ||
Martin von Zweigbergk
|
r23396 | a = ma[f] | ||
fla = ma.flags(f) | ||||
Augie Fackler
|
r43347 | nol = b'l' not in fl1 + fl2 + fla | ||
Martin von Zweigbergk
|
r23395 | if n2 == a and fl2 == fla: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
Augie Fackler
|
r46554 | f, | ||
mergestatemod.ACTION_KEEP, | ||||
(), | ||||
b'remote unchanged', | ||||
Augie Fackler
|
r45383 | ) | ||
Augie Fackler
|
r43346 | elif n1 == a and fl1 == fla: # local unchanged - use remote | ||
if n1 == n2: # optimization: keep local content | ||||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_EXEC, | ||
Augie Fackler
|
r43347 | (fl2,), | ||
b'update permissions', | ||||
) | ||||
Martin von Zweigbergk
|
r23395 | else: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Pulkit Goyal
|
r45834 | mergestatemod.ACTION_GET, | ||
Augie Fackler
|
r43346 | (fl2, False), | ||
Augie Fackler
|
r43347 | b'remote is newer', | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r45832 | if branchmerge: | ||
Pulkit Goyal
|
r45944 | mresult.addcommitinfo( | ||
f, b'filenode-source', b'other' | ||||
) | ||||
Augie Fackler
|
r43346 | elif nol and n2 == a: # remote only changed 'x' | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_EXEC, | ||
(fl2,), | ||||
b'update permissions', | ||||
) | ||||
Augie Fackler
|
r43346 | elif nol and n1 == a: # local only changed 'x' | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Pulkit Goyal
|
r45834 | mergestatemod.ACTION_GET, | ||
Pulkit Goyal
|
r45178 | (fl1, False), | ||
b'remote is newer', | ||||
) | ||||
Pulkit Goyal
|
r45832 | if branchmerge: | ||
Pulkit Goyal
|
r45944 | mresult.addcommitinfo(f, b'filenode-source', b'other') | ||
Augie Fackler
|
r43346 | else: # both changed something | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_MERGE, | ||
Augie Fackler
|
r43346 | (f, f, f, False, pa.node()), | ||
Augie Fackler
|
r43347 | b'versions differ', | ||
Augie Fackler
|
r43346 | ) | ||
elif n1: # file exists only on local side | ||||
Martin von Zweigbergk
|
r44682 | if f in copied2: | ||
Augie Fackler
|
r43346 | pass # we'll deal with it on m2 side | ||
Martin von Zweigbergk
|
r44682 | elif ( | ||
f in branch_copies1.movewithdir | ||||
): # directory rename, move local | ||||
f2 = branch_copies1.movewithdir[f] | ||||
Durham Goode
|
r31515 | if f2 in m2: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f2, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_MERGE, | ||
Augie Fackler
|
r43346 | (f, f2, None, True, pa.node()), | ||
Augie Fackler
|
r43347 | b'remote directory rename, both created', | ||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r23475 | else: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f2, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_DIR_RENAME_MOVE_LOCAL, | ||
Augie Fackler
|
r43346 | (f, fl1), | ||
Augie Fackler
|
r43347 | b'remote directory rename - move from %s' % f, | ||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r44682 | elif f in branch_copies1.copy: | ||
f2 = branch_copies1.copy[f] | ||||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_MERGE, | ||
Augie Fackler
|
r43346 | (f, f2, f2, False, pa.node()), | ||
Augie Fackler
|
r43347 | b'local copied/moved from %s' % f2, | ||
Augie Fackler
|
r43346 | ) | ||
elif f in ma: # clean, a different, no remote | ||||
Martin von Zweigbergk
|
r23473 | if n1 != ma[f]: | ||
if acceptremote: | ||||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_REMOVE, | ||
None, | ||||
b'remote delete', | ||||
) | ||||
Martin von Zweigbergk
|
r23473 | else: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_CHANGED_DELETED, | ||
Augie Fackler
|
r43346 | (f, None, f, False, pa.node()), | ||
Augie Fackler
|
r43347 | b'prompt changed/deleted', | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r46158 | if branchmerge: | ||
mresult.addcommitinfo( | ||||
f, b'merge-removal-candidate', b'yes' | ||||
) | ||||
Joerg Sonnenberger
|
r47771 | elif n1 == repo.nodeconstants.addednodeid: | ||
Joerg Sonnenberger
|
r45678 | # This file was locally added. We should forget it instead of | ||
Martin von Zweigbergk
|
r23473 | # deleting it. | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
Augie Fackler
|
r46554 | f, | ||
mergestatemod.ACTION_FORGET, | ||||
None, | ||||
b'remote deleted', | ||||
Augie Fackler
|
r45383 | ) | ||
Mads Kiilerich
|
r20639 | else: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
Augie Fackler
|
r46554 | f, | ||
mergestatemod.ACTION_REMOVE, | ||||
None, | ||||
b'other deleted', | ||||
Augie Fackler
|
r45383 | ) | ||
Pulkit Goyal
|
r46161 | if branchmerge: | ||
# the file must be absent after merging, | ||||
# howeber the user might make | ||||
# the file reappear using revert and if they does, | ||||
# we force create a new node | ||||
mresult.addcommitinfo( | ||||
f, b'merge-removal-candidate', b'yes' | ||||
) | ||||
Pulkit Goyal
|
r46041 | else: # file not in ancestor, not in remote | ||
mresult.addfile( | ||||
f, | ||||
Pulkit Goyal
|
r46095 | mergestatemod.ACTION_KEEP_NEW, | ||
Pulkit Goyal
|
r46041 | None, | ||
b'ancestor missing, remote missing', | ||||
) | ||||
Augie Fackler
|
r43346 | elif n2: # file exists only on remote side | ||
Martin von Zweigbergk
|
r44682 | if f in copied1: | ||
Augie Fackler
|
r43346 | pass # we'll deal with it on m1 side | ||
Martin von Zweigbergk
|
r44682 | elif f in branch_copies2.movewithdir: | ||
f2 = branch_copies2.movewithdir[f] | ||||
Durham Goode
|
r31515 | if f2 in m1: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f2, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_MERGE, | ||
Augie Fackler
|
r43346 | (f2, f, None, False, pa.node()), | ||
Augie Fackler
|
r43347 | b'local directory rename, both created', | ||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r23476 | else: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f2, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_LOCAL_DIR_RENAME_GET, | ||
Augie Fackler
|
r43346 | (f, fl2), | ||
Augie Fackler
|
r43347 | b'local directory rename - get from %s' % f, | ||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r44682 | elif f in branch_copies2.copy: | ||
f2 = branch_copies2.copy[f] | ||||
Pulkit Goyal
|
r45853 | msg, args = None, None | ||
Durham Goode
|
r31515 | if f2 in m2: | ||
Pulkit Goyal
|
r45853 | args = (f2, f, f2, False, pa.node()) | ||
msg = b'remote copied from %s' % f2 | ||||
Martin von Zweigbergk
|
r23473 | else: | ||
Pulkit Goyal
|
r45853 | args = (f2, f, f2, True, pa.node()) | ||
msg = b'remote moved from %s' % f2 | ||||
mresult.addfile(f, mergestatemod.ACTION_MERGE, args, msg) | ||||
Martin von Zweigbergk
|
r23473 | elif f not in ma: | ||
# local unknown, remote created: the logic is described by the | ||||
# following table: | ||||
# | ||||
# force branchmerge different | action | ||||
Martin von Zweigbergk
|
r23651 | # n * * | create | ||
Martin von Zweigbergk
|
r23650 | # y n * | create | ||
# y y n | create | ||||
Martin von Zweigbergk
|
r23473 | # y y y | merge | ||
# | ||||
# Checking whether the files are different is expensive, so we | ||||
# don't do that when we can avoid it. | ||||
Martin von Zweigbergk
|
r23649 | if not force: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_CREATED, | ||
(fl2,), | ||||
b'remote created', | ||||
) | ||||
Martin von Zweigbergk
|
r23649 | elif not branchmerge: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_CREATED, | ||
(fl2,), | ||||
b'remote created', | ||||
) | ||||
Martin von Zweigbergk
|
r23473 | else: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_CREATED_MERGE, | ||
Augie Fackler
|
r43346 | (fl2, pa.node()), | ||
Augie Fackler
|
r43347 | b'remote created, get or merge', | ||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r23473 | elif n2 != ma[f]: | ||
Gábor Stefanik
|
r30581 | df = None | ||
Martin von Zweigbergk
|
r44682 | for d in branch_copies1.dirmove: | ||
Gábor Stefanik
|
r30581 | if f.startswith(d): | ||
# new file added in a directory that was moved | ||||
Martin von Zweigbergk
|
r44682 | df = branch_copies1.dirmove[d] + f[len(d) :] | ||
Gábor Stefanik
|
r30581 | break | ||
Durham Goode
|
r31515 | if df is not None and df in m1: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
df, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_MERGE, | ||
Augie Fackler
|
r43346 | (df, f, f, False, pa.node()), | ||
Augie Fackler
|
r43347 | b'local directory rename - respect move ' | ||
b'from %s' % f, | ||||
Augie Fackler
|
r43346 | ) | ||
Gábor Stefanik
|
r30581 | elif acceptremote: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_CREATED, | ||
(fl2,), | ||||
b'remote recreating', | ||||
) | ||||
Siddharth Agarwal
|
r18606 | else: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_DELETED_CHANGED, | ||
Augie Fackler
|
r43346 | (None, f, f, False, pa.node()), | ||
Augie Fackler
|
r43347 | b'prompt deleted/changed', | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r46158 | if branchmerge: | ||
mresult.addcommitinfo( | ||||
f, b'merge-removal-candidate', b'yes' | ||||
) | ||||
Pulkit Goyal
|
r46040 | else: | ||
mresult.addfile( | ||||
f, | ||||
mergestatemod.ACTION_KEEP_ABSENT, | ||||
None, | ||||
b'local not present, remote unchanged', | ||||
) | ||||
Pulkit Goyal
|
r46161 | if branchmerge: | ||
# the file must be absent after merging | ||||
# however the user might make | ||||
# the file reappear using revert and if they does, | ||||
# we force create a new node | ||||
mresult.addcommitinfo(f, b'merge-removal-candidate', b'yes') | ||||
Martin von Zweigbergk
|
r23651 | |||
Augie Fackler
|
r43347 | if repo.ui.configbool(b'experimental', b'merge.checkpathconflicts'): | ||
Siddharth Agarwal
|
r34942 | # If we are merging, look for path conflicts. | ||
Pulkit Goyal
|
r45841 | checkpathconflicts(repo, wctx, p2, mresult) | ||
Mark Thomas
|
r34556 | |||
Martin von Zweigbergk
|
r38055 | narrowmatch = repo.narrowmatch() | ||
if not narrowmatch.always(): | ||||
# Updates "actions" in place | ||||
Pulkit Goyal
|
r45842 | _filternarrowactions(narrowmatch, branchmerge, mresult) | ||
Martin von Zweigbergk
|
r38055 | |||
Martin von Zweigbergk
|
r44682 | renamedelete = branch_copies1.renamedelete | ||
renamedelete.update(branch_copies2.renamedelete) | ||||
Pulkit Goyal
|
r45944 | mresult.updatevalues(diverge, renamedelete) | ||
Pulkit Goyal
|
r45838 | return mresult | ||
Matt Mackall
|
r3105 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r45843 | def _resolvetrivial(repo, wctx, mctx, ancestor, mresult): | ||
Martin von Zweigbergk
|
r23531 | """Resolves false conflicts where the nodeid changed but the content | ||
Augie Fackler
|
r46554 | remained the same.""" | ||
Augie Fackler
|
r36334 | # We force a copy of actions.items() because we're going to mutate | ||
# actions as we resolve trivial conflicts. | ||||
Pulkit Goyal
|
r45905 | for f in list(mresult.files((mergestatemod.ACTION_CHANGED_DELETED,))): | ||
Pulkit Goyal
|
r45850 | if f in ancestor and not wctx[f].cmp(ancestor[f]): | ||
Martin von Zweigbergk
|
r23531 | # local did change but ended up with same content | ||
Pulkit Goyal
|
r45843 | mresult.addfile( | ||
f, mergestatemod.ACTION_REMOVE, None, b'prompt same' | ||||
) | ||||
Pulkit Goyal
|
r45850 | |||
Pulkit Goyal
|
r45905 | for f in list(mresult.files((mergestatemod.ACTION_DELETED_CHANGED,))): | ||
Pulkit Goyal
|
r45850 | if f in ancestor and not mctx[f].cmp(ancestor[f]): | ||
Martin von Zweigbergk
|
r23531 | # remote did change but ended up with same content | ||
Pulkit Goyal
|
r45843 | mresult.removefile(f) # don't get = keep local deleted | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r23531 | |||
Augie Fackler
|
r43346 | def calculateupdates( | ||
repo, | ||||
wctx, | ||||
mctx, | ||||
ancestors, | ||||
branchmerge, | ||||
force, | ||||
acceptremote, | ||||
followcopies, | ||||
matcher=None, | ||||
mergeforce=False, | ||||
): | ||||
Pulkit Goyal
|
r45728 | """ | ||
Calculate the actions needed to merge mctx into wctx using ancestors | ||||
Uses manifestmerge() to merge manifest and get list of actions required to | ||||
perform for merging two manifests. If there are multiple ancestors, uses bid | ||||
merge if enabled. | ||||
Also filters out actions which are unrequired if repository is sparse. | ||||
Pulkit Goyal
|
r45831 | Returns mergeresult object same as manifestmerge(). | ||
Pulkit Goyal
|
r45728 | """ | ||
Gregory Szorc
|
r33323 | # Avoid cycle. | ||
from . import sparse | ||||
Pulkit Goyal
|
r45838 | mresult = None | ||
Augie Fackler
|
r43346 | if len(ancestors) == 1: # default | ||
Pulkit Goyal
|
r45831 | mresult = manifestmerge( | ||
Augie Fackler
|
r43346 | repo, | ||
wctx, | ||||
mctx, | ||||
ancestors[0], | ||||
branchmerge, | ||||
force, | ||||
matcher, | ||||
acceptremote, | ||||
followcopies, | ||||
) | ||||
Pulkit Goyal
|
r45844 | _checkunknownfiles(repo, wctx, mctx, force, mresult, mergeforce) | ||
Arseniy Alekseyev
|
r50778 | if repo.ui.configbool(b'devel', b'debug.abort-update'): | ||
exit(1) | ||||
Martin von Zweigbergk
|
r23385 | |||
Augie Fackler
|
r43346 | else: # only when merge.preferancestor=* - the default | ||
Martin von Zweigbergk
|
r23385 | repo.ui.note( | ||
Augie Fackler
|
r43347 | _(b"note: merging %s and %s using bids from ancestors %s\n") | ||
Augie Fackler
|
r43346 | % ( | ||
wctx, | ||||
mctx, | ||||
Augie Fackler
|
r43347 | _(b' and ').join(pycompat.bytestr(anc) for anc in ancestors), | ||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Martin von Zweigbergk
|
r23385 | |||
Pulkit Goyal
|
r45837 | # mapping filename to bids (action method to list af actions) | ||
# {FILENAME1 : BID1, FILENAME2 : BID2} | ||||
# BID is another dictionary which contains | ||||
# mapping of following form: | ||||
# {ACTION_X : [info, ..], ACTION_Y : [info, ..]} | ||||
fbids = {} | ||||
Pulkit Goyal
|
r46042 | mresult = mergeresult() | ||
Martin von Zweigbergk
|
r23526 | diverge, renamedelete = None, None | ||
Martin von Zweigbergk
|
r23385 | for ancestor in ancestors: | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b'\ncalculating bids for ancestor %s\n') % ancestor) | ||
Pulkit Goyal
|
r45831 | mresult1 = manifestmerge( | ||
Augie Fackler
|
r43346 | repo, | ||
wctx, | ||||
mctx, | ||||
ancestor, | ||||
branchmerge, | ||||
force, | ||||
matcher, | ||||
acceptremote, | ||||
followcopies, | ||||
forcefulldiff=True, | ||||
) | ||||
Pulkit Goyal
|
r45844 | _checkunknownfiles(repo, wctx, mctx, force, mresult1, mergeforce) | ||
Matt Mackall
|
r26318 | |||
# Track the shortest set of warning on the theory that bid | ||||
# merge will correctly incorporate more information | ||||
Pulkit Goyal
|
r45831 | if diverge is None or len(mresult1.diverge) < len(diverge): | ||
diverge = mresult1.diverge | ||||
if renamedelete is None or len(renamedelete) < len( | ||||
mresult1.renamedelete | ||||
): | ||||
renamedelete = mresult1.renamedelete | ||||
Matt Mackall
|
r26318 | |||
Pulkit Goyal
|
r46042 | # blindly update final mergeresult commitinfo with what we get | ||
# from mergeresult object for each ancestor | ||||
# TODO: some commitinfo depends on what bid merge choose and hence | ||||
# we will need to make commitinfo also depend on bid merge logic | ||||
mresult._commitinfo.update(mresult1._commitinfo) | ||||
Pulkit Goyal
|
r45906 | for f, a in mresult1.filemap(sort=True): | ||
Martin von Zweigbergk
|
r23638 | m, args, msg = a | ||
r49560 | repo.ui.debug(b' %s: %s -> %s\n' % (f, msg, m.__bytes__())) | |||
Martin von Zweigbergk
|
r23638 | if f in fbids: | ||
d = fbids[f] | ||||
if m in d: | ||||
d[m].append(a) | ||||
Martin von Zweigbergk
|
r23385 | else: | ||
Martin von Zweigbergk
|
r23638 | d[m] = [a] | ||
else: | ||||
fbids[f] = {m: [a]} | ||||
Martin von Zweigbergk
|
r23385 | |||
Pulkit Goyal
|
r45837 | # Call for bids | ||
Martin von Zweigbergk
|
r23385 | # Pick the best bid for each file | ||
Pulkit Goyal
|
r46018 | repo.ui.note( | ||
_(b'\nauction for merging merge bids (%d ancestors)\n') | ||||
% len(ancestors) | ||||
) | ||||
Martin von Zweigbergk
|
r23385 | for f, bids in sorted(fbids.items()): | ||
Pulkit Goyal
|
r46031 | if repo.ui.debugflag: | ||
repo.ui.debug(b" list of bids for %s:\n" % f) | ||||
for m, l in sorted(bids.items()): | ||||
for _f, args, msg in l: | ||||
r49560 | repo.ui.debug(b' %s -> %s\n' % (msg, m.__bytes__())) | |||
Martin von Zweigbergk
|
r23385 | # bids is a mapping from action method to list af actions | ||
# Consensus? | ||||
Augie Fackler
|
r43346 | if len(bids) == 1: # all bids are the same kind of method | ||
Pulkit Goyal
|
r34350 | m, l = list(bids.items())[0] | ||
Augie Fackler
|
r43346 | if all(a == l[0] for a in l[1:]): # len(bids) is > 1 | ||
r49560 | repo.ui.note( | |||
_(b" %s: consensus for %s\n") % (f, m.__bytes__()) | ||||
) | ||||
Pulkit Goyal
|
r45839 | mresult.addfile(f, *l[0]) | ||
Martin von Zweigbergk
|
r23385 | continue | ||
# If keep is an option, just do it. | ||||
Augie Fackler
|
r45383 | if mergestatemod.ACTION_KEEP in bids: | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b" %s: picking 'keep' action\n") % f) | ||
Pulkit Goyal
|
r45839 | mresult.addfile(f, *bids[mergestatemod.ACTION_KEEP][0]) | ||
Martin von Zweigbergk
|
r23385 | continue | ||
Pulkit Goyal
|
r46039 | # If keep absent is an option, just do that | ||
if mergestatemod.ACTION_KEEP_ABSENT in bids: | ||||
repo.ui.note(_(b" %s: picking 'keep absent' action\n") % f) | ||||
mresult.addfile(f, *bids[mergestatemod.ACTION_KEEP_ABSENT][0]) | ||||
continue | ||||
Pulkit Goyal
|
r46194 | # ACTION_KEEP_NEW and ACTION_CHANGED_DELETED are conflicting actions | ||
# as one say that file is new while other says that file was present | ||||
# earlier too and has a change delete conflict | ||||
# Let's fall back to conflicting ACTION_CHANGED_DELETED and let user | ||||
# do the right thing | ||||
if ( | ||||
mergestatemod.ACTION_CHANGED_DELETED in bids | ||||
and mergestatemod.ACTION_KEEP_NEW in bids | ||||
): | ||||
repo.ui.note(_(b" %s: picking 'changed/deleted' action\n") % f) | ||||
mresult.addfile( | ||||
f, *bids[mergestatemod.ACTION_CHANGED_DELETED][0] | ||||
) | ||||
continue | ||||
Pulkit Goyal
|
r46095 | # If keep new is an option, let's just do that | ||
if mergestatemod.ACTION_KEEP_NEW in bids: | ||||
repo.ui.note(_(b" %s: picking 'keep new' action\n") % f) | ||||
mresult.addfile(f, *bids[mergestatemod.ACTION_KEEP_NEW][0]) | ||||
continue | ||||
Pulkit Goyal
|
r46192 | # ACTION_GET and ACTION_DELETE_CHANGED are conflicting actions as | ||
# one action states the file is newer/created on remote side and | ||||
# other states that file is deleted locally and changed on remote | ||||
# side. Let's fallback and rely on a conflicting action to let user | ||||
# do the right thing | ||||
if ( | ||||
mergestatemod.ACTION_DELETED_CHANGED in bids | ||||
and mergestatemod.ACTION_GET in bids | ||||
): | ||||
repo.ui.note(_(b" %s: picking 'delete/changed' action\n") % f) | ||||
mresult.addfile( | ||||
f, *bids[mergestatemod.ACTION_DELETED_CHANGED][0] | ||||
) | ||||
continue | ||||
Martin von Zweigbergk
|
r23385 | # If there are gets and they all agree [how could they not?], do it. | ||
Augie Fackler
|
r45383 | if mergestatemod.ACTION_GET in bids: | ||
ga0 = bids[mergestatemod.ACTION_GET][0] | ||||
if all(a == ga0 for a in bids[mergestatemod.ACTION_GET][1:]): | ||||
Augie Fackler
|
r43347 | repo.ui.note(_(b" %s: picking 'get' action\n") % f) | ||
Pulkit Goyal
|
r45839 | mresult.addfile(f, *ga0) | ||
Martin von Zweigbergk
|
r23385 | continue | ||
# TODO: Consider other simple actions such as mode changes | ||||
# Handle inefficient democrazy. | ||||
Augie Fackler
|
r43347 | repo.ui.note(_(b' %s: multiple bids for merge action:\n') % f) | ||
Martin von Zweigbergk
|
r23385 | for m, l in sorted(bids.items()): | ||
for _f, args, msg in l: | ||||
r49560 | repo.ui.note(b' %s -> %s\n' % (msg, m.__bytes__())) | |||
Martin von Zweigbergk
|
r23385 | # Pick random action. TODO: Instead, prompt user when resolving | ||
Pulkit Goyal
|
r34350 | m, l = list(bids.items())[0] | ||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
r49560 | _(b' %s: ambiguous merge - picked %s action\n') | |||
% (f, m.__bytes__()) | ||||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r45839 | mresult.addfile(f, *l[0]) | ||
Martin von Zweigbergk
|
r23385 | continue | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b'end of auction\n\n')) | ||
Pulkit Goyal
|
r45944 | mresult.updatevalues(diverge, renamedelete) | ||
Martin von Zweigbergk
|
r23385 | |||
Martin von Zweigbergk
|
r23640 | if wctx.rev() is None: | ||
Pulkit Goyal
|
r45907 | _forgetremoved(wctx, mctx, branchmerge, mresult) | ||
Martin von Zweigbergk
|
r23640 | |||
Pulkit Goyal
|
r45847 | sparse.filterupdatesactions(repo, wctx, mctx, branchmerge, mresult) | ||
Pulkit Goyal
|
r45843 | _resolvetrivial(repo, wctx, mctx, ancestors[0], mresult) | ||
Gregory Szorc
|
r33323 | |||
Pulkit Goyal
|
r45831 | return mresult | ||
Martin von Zweigbergk
|
r23385 | |||
Augie Fackler
|
r43346 | |||
Phil Cohen
|
r34144 | def _getcwd(): | ||
try: | ||||
Matt Harbison
|
r39843 | return encoding.getcwd() | ||
Manuel Jacob
|
r50201 | except FileNotFoundError: | ||
return None | ||||
Phil Cohen
|
r34144 | |||
Augie Fackler
|
r43346 | |||
Phil Cohen
|
r33081 | def batchremove(repo, wctx, actions): | ||
Mads Kiilerich
|
r21392 | """apply removes to the working directory | ||
Bryan O'Sullivan
|
r18630 | |||
yields tuples for progress updates | ||||
""" | ||||
Bryan O'Sullivan
|
r18640 | verbose = repo.ui.verbose | ||
Phil Cohen
|
r34144 | cwd = _getcwd() | ||
Bryan O'Sullivan
|
r18633 | i = 0 | ||
Mads Kiilerich
|
r21545 | for f, args, msg in actions: | ||
Augie Fackler
|
r43347 | repo.ui.debug(b" %s: %s -> r\n" % (f, msg)) | ||
Mads Kiilerich
|
r21551 | if verbose: | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b"removing %s\n") % f) | ||
Phil Cohen
|
r33086 | wctx[f].audit() | ||
Mads Kiilerich
|
r21551 | try: | ||
Phil Cohen
|
r33082 | wctx[f].remove(ignoremissing=True) | ||
Gregory Szorc
|
r25660 | except OSError as inst: | ||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
Martin von Zweigbergk
|
r46280 | _(b"update failed to remove %s: %s!\n") | ||
Matt Harbison
|
r47519 | % (f, stringutil.forcebytestr(inst.strerror)) | ||
Augie Fackler
|
r43346 | ) | ||
Mads Kiilerich
|
r21392 | if i == 100: | ||
yield i, f | ||||
i = 0 | ||||
i += 1 | ||||
if i > 0: | ||||
yield i, f | ||||
Phil Cohen
|
r34144 | |||
if cwd and not _getcwd(): | ||||
# cwd was removed in the course of removing files; print a helpful | ||||
# warning. | ||||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b"current directory was removed\n" | ||
b"(consider changing to repo root: %s)\n" | ||||
Augie Fackler
|
r43346 | ) | ||
% repo.root | ||||
) | ||||
Mads Kiilerich
|
r21392 | |||
Valentin Gatien-Baron
|
r42656 | def batchget(repo, mctx, wctx, wantfiledata, actions): | ||
Mads Kiilerich
|
r21392 | """apply gets to the working directory | ||
mctx is the context to get from | ||||
Valentin Gatien-Baron
|
r42656 | Yields arbitrarily many (False, tuple) for progress updates, followed by | ||
exactly one (True, filedata). When wantfiledata is false, filedata is an | ||||
Valentin Gatien-Baron
|
r42722 | empty dict. When wantfiledata is true, filedata[f] is a triple (mode, size, | ||
mtime) of the file f written for each action. | ||||
Mads Kiilerich
|
r21392 | """ | ||
Valentin Gatien-Baron
|
r42722 | filedata = {} | ||
Mads Kiilerich
|
r21392 | verbose = repo.ui.verbose | ||
fctx = mctx.filectx | ||||
Siddharth Agarwal
|
r27656 | ui = repo.ui | ||
Mads Kiilerich
|
r21392 | i = 0 | ||
Gregory Szorc
|
r28200 | with repo.wvfs.backgroundclosing(ui, expectedcount=len(actions)): | ||
Gregory Szorc
|
r28199 | for f, (flags, backup), msg in actions: | ||
Augie Fackler
|
r43347 | repo.ui.debug(b" %s: %s -> g\n" % (f, msg)) | ||
Gregory Szorc
|
r28199 | if verbose: | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b"getting %s\n") % f) | ||
Siddharth Agarwal
|
r27656 | |||
Gregory Szorc
|
r28199 | if backup: | ||
Mark Thomas
|
r34550 | # If a file or directory exists with the same name, back that | ||
# up. Otherwise, look to see if there is a file that conflicts | ||||
# with a directory this file is in, and if so, back that up. | ||||
Martin von Zweigbergk
|
r41711 | conflicting = f | ||
Mark Thomas
|
r34550 | if not repo.wvfs.lexists(f): | ||
Martin von Zweigbergk
|
r44032 | for p in pathutil.finddirs(f): | ||
Mark Thomas
|
r34550 | if repo.wvfs.isfileorlink(p): | ||
Martin von Zweigbergk
|
r41711 | conflicting = p | ||
Mark Thomas
|
r34550 | break | ||
Martin von Zweigbergk
|
r41711 | if repo.wvfs.lexists(conflicting): | ||
Martin von Zweigbergk
|
r41750 | orig = scmutil.backuppath(ui, repo, conflicting) | ||
util.rename(repo.wjoin(conflicting), orig) | ||||
Valentin Gatien-Baron
|
r42656 | wfctx = wctx[f] | ||
wfctx.clearunknown() | ||||
Augie Fackler
|
r43347 | atomictemp = ui.configbool(b"experimental", b"update.atomic-file") | ||
Augie Fackler
|
r43346 | size = wfctx.write( | ||
fctx(f).data(), | ||||
flags, | ||||
backgroundclose=True, | ||||
atomictemp=atomictemp, | ||||
) | ||||
Valentin Gatien-Baron
|
r42656 | if wantfiledata: | ||
r49200 | # XXX note that there is a race window between the time we | |||
# write the clean data into the file and we stats it. So another | ||||
# writing process meddling with the file content right after we | ||||
# wrote it could cause bad stat data to be gathered. | ||||
# | ||||
# They are 2 data we gather here | ||||
# - the mode: | ||||
# That we actually just wrote, we should not need to read | ||||
# it from disk, (except not all mode might have survived | ||||
# the disk round-trip, which is another issue: we should | ||||
# not depends on this) | ||||
# - the mtime, | ||||
# On system that support nanosecond precision, the mtime | ||||
# could be accurate enough to tell the two writes appart. | ||||
# However gathering it in a racy way make the mtime we | ||||
# gather "unreliable". | ||||
# | ||||
# (note: we get the size from the data we write, which is sane) | ||||
# | ||||
# So in theory the data returned here are fully racy, but in | ||||
# practice "it works mostly fine". | ||||
# | ||||
# Do not be surprised if you end up reading this while looking | ||||
# for the causes of some buggy status. Feel free to improve | ||||
# this in the future, but we cannot simply stop gathering | ||||
# information. Otherwise `hg status` call made after a large `hg | ||||
# update` runs would have to redo a similar amount of work to | ||||
# restore and compare all files content. | ||||
Valentin Gatien-Baron
|
r42656 | s = wfctx.lstat() | ||
mode = s.st_mode | ||||
Simon Sapin
|
r49079 | mtime = timestamp.mtime_of(s) | ||
# for dirstate.update_file's parentfiledata argument: | ||||
filedata[f] = (mode, size, mtime) | ||||
Gregory Szorc
|
r28199 | if i == 100: | ||
Valentin Gatien-Baron
|
r42656 | yield False, (i, f) | ||
Gregory Szorc
|
r28199 | i = 0 | ||
i += 1 | ||||
Bryan O'Sullivan
|
r18633 | if i > 0: | ||
Valentin Gatien-Baron
|
r42656 | yield False, (i, f) | ||
yield True, filedata | ||||
Bryan O'Sullivan
|
r18630 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r45895 | def _prefetchfiles(repo, ctx, mresult): | ||
Matt Harbison
|
r37780 | """Invoke ``scmutil.prefetchfiles()`` for the files relevant to the dict | ||
Matt Harbison
|
r36159 | of merge actions. ``ctx`` is the context being merged in.""" | ||
# Skipping 'a', 'am', 'f', 'r', 'dm', 'e', 'k', 'p' and 'pr', because they | ||||
# don't touch the context to be merged in. 'cd' is skipped, because | ||||
# changed/deleted never resolves to something from the remote side. | ||||
Pulkit Goyal
|
r45905 | files = mresult.files( | ||
Pulkit Goyal
|
r45895 | [ | ||
Augie Fackler
|
r45383 | mergestatemod.ACTION_GET, | ||
mergestatemod.ACTION_DELETED_CHANGED, | ||||
mergestatemod.ACTION_LOCAL_DIR_RENAME_GET, | ||||
mergestatemod.ACTION_MERGE, | ||||
Pulkit Goyal
|
r45895 | ] | ||
Pulkit Goyal
|
r45905 | ) | ||
Pulkit Goyal
|
r45895 | |||
Matt Harbison
|
r37780 | prefetch = scmutil.prefetchfiles | ||
matchfiles = scmutil.matchfiles | ||||
Augie Fackler
|
r43346 | prefetch( | ||
Augie Fackler
|
r46554 | repo, | ||
[ | ||||
( | ||||
ctx.rev(), | ||||
matchfiles(repo, files), | ||||
) | ||||
], | ||||
Augie Fackler
|
r43346 | ) | ||
Phil Cohen
|
r34126 | |||
Gregory Szorc
|
r37125 | @attr.s(frozen=True) | ||
Gregory Szorc
|
r49801 | class updateresult: | ||
Gregory Szorc
|
r37125 | updatedcount = attr.ib() | ||
mergedcount = attr.ib() | ||||
removedcount = attr.ib() | ||||
unresolvedcount = attr.ib() | ||||
Gregory Szorc
|
r37143 | def isempty(self): | ||
Augie Fackler
|
r43346 | return not ( | ||
self.updatedcount | ||||
or self.mergedcount | ||||
or self.removedcount | ||||
or self.unresolvedcount | ||||
) | ||||
Gregory Szorc
|
r37143 | |||
Augie Fackler
|
r43346 | def applyupdates( | ||
Augie Fackler
|
r46554 | repo, | ||
mresult, | ||||
wctx, | ||||
mctx, | ||||
overwrite, | ||||
wantfiledata, | ||||
labels=None, | ||||
Augie Fackler
|
r43346 | ): | ||
Peter Arrenbrecht
|
r11454 | """apply the merge action list to the working directory | ||
Pulkit Goyal
|
r45894 | mresult is a mergeresult object representing result of the merge | ||
Peter Arrenbrecht
|
r11454 | wctx is the working copy context | ||
mctx is the context to be merged into the working copy | ||||
Greg Ward
|
r13162 | |||
Valentin Gatien-Baron
|
r42656 | Return a tuple of (counts, filedata), where counts is a tuple | ||
(updated, merged, removed, unresolved) that describes how many | ||||
files were affected by the update, and filedata is as described in | ||||
batchget. | ||||
Peter Arrenbrecht
|
r11454 | """ | ||
Matt Mackall
|
r3315 | |||
Pulkit Goyal
|
r45895 | _prefetchfiles(repo, mctx, mresult) | ||
Matt Harbison
|
r36159 | |||
Siddharth Agarwal
|
r27078 | updated, merged, removed = 0, 0, 0 | ||
Martin von Zweigbergk
|
r46071 | ms = wctx.mergestate(clean=True) | ||
Martin von Zweigbergk
|
r46065 | ms.start(wctx.p1().node(), mctx.node(), labels) | ||
Pulkit Goyal
|
r45178 | |||
Gregory Szorc
|
r49768 | for f, op in mresult.commitinfo.items(): | ||
Pulkit Goyal
|
r45833 | # the other side of filenode was choosen while merging, store this in | ||
# mergestate so that it can be reused on commit | ||||
Pulkit Goyal
|
r45945 | ms.addcommitinfo(f, op) | ||
Pulkit Goyal
|
r45833 | |||
r49562 | num_no_op = mresult.len(mergestatemod.MergeAction.NO_OP_ACTIONS) | |||
numupdates = mresult.len() - num_no_op | ||||
Augie Fackler
|
r43346 | progress = repo.ui.makeprogress( | ||
Augie Fackler
|
r43347 | _(b'updating'), unit=_(b'files'), total=numupdates | ||
Augie Fackler
|
r43346 | ) | ||
Bryan O'Sullivan
|
r18630 | |||
Pulkit Goyal
|
r45897 | if b'.hgsubstate' in mresult._actionmapping[mergestatemod.ACTION_REMOVE]: | ||
Yuya Nishihara
|
r36026 | subrepoutil.submerge(repo, wctx, mctx, wctx, overwrite, labels) | ||
Bryan O'Sullivan
|
r18632 | |||
Mark Thomas
|
r34548 | # record path conflicts | ||
Pulkit Goyal
|
r45897 | for f, args, msg in mresult.getactions( | ||
[mergestatemod.ACTION_PATH_CONFLICT], sort=True | ||||
): | ||||
Mark Thomas
|
r34548 | f1, fo = args | ||
s = repo.ui.status | ||||
Augie Fackler
|
r43346 | s( | ||
_( | ||||
Augie Fackler
|
r43347 | b"%s: path conflict - a file or link has the same name as a " | ||
b"directory\n" | ||||
Augie Fackler
|
r43346 | ) | ||
% f | ||||
) | ||||
Augie Fackler
|
r43347 | if fo == b'l': | ||
s(_(b"the local file has been renamed to %s\n") % f1) | ||||
Mark Thomas
|
r34548 | else: | ||
Augie Fackler
|
r43347 | s(_(b"the remote file has been renamed to %s\n") % f1) | ||
s(_(b"resolve manually then use 'hg resolve --mark %s'\n") % f) | ||||
Pulkit Goyal
|
r45719 | ms.addpathconflict(f, f1, fo) | ||
Martin von Zweigbergk
|
r38364 | progress.increment(item=f) | ||
Mark Thomas
|
r34548 | |||
Phil Cohen
|
r34787 | # When merging in-memory, we can't support worker processes, so set the | ||
# per-item cost at 0 in that case. | ||||
cost = 0 if wctx.isinmemory() else 0.001 | ||||
Mark Thomas
|
r34549 | # remove in parallel (must come before resolving path conflicts and getting) | ||
Augie Fackler
|
r43346 | prog = worker.worker( | ||
Augie Fackler
|
r45383 | repo.ui, | ||
cost, | ||||
batchremove, | ||||
(repo, wctx), | ||||
Pulkit Goyal
|
r45900 | list(mresult.getactions([mergestatemod.ACTION_REMOVE], sort=True)), | ||
Augie Fackler
|
r43346 | ) | ||
FUJIWARA Katsunori
|
r19095 | for i, item in prog: | ||
Martin von Zweigbergk
|
r38364 | progress.increment(step=i, item=item) | ||
Pulkit Goyal
|
r45898 | removed = mresult.len((mergestatemod.ACTION_REMOVE,)) | ||
Mads Kiilerich
|
r21390 | |||
Mark Thomas
|
r34549 | # resolve path conflicts (must come before getting) | ||
Pulkit Goyal
|
r45897 | for f, args, msg in mresult.getactions( | ||
[mergestatemod.ACTION_PATH_CONFLICT_RESOLVE], sort=True | ||||
): | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b" %s: %s -> pr\n" % (f, msg)) | ||
Martin von Zweigbergk
|
r45465 | (f0, origf0) = args | ||
Mark Thomas
|
r34549 | if wctx[f0].lexists(): | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b"moving %s to %s\n") % (f0, f)) | ||
Mark Thomas
|
r34549 | wctx[f].audit() | ||
wctx[f].write(wctx.filectx(f0).data(), wctx.filectx(f0).flags()) | ||||
wctx[f0].remove() | ||||
Martin von Zweigbergk
|
r38364 | progress.increment(item=f) | ||
Mark Thomas
|
r34549 | |||
Gregory Szorc
|
r38755 | # get in parallel. | ||
Augie Fackler
|
r43346 | threadsafe = repo.ui.configbool( | ||
Augie Fackler
|
r43347 | b'experimental', b'worker.wdir-get-thread-safe' | ||
Augie Fackler
|
r43346 | ) | ||
prog = worker.worker( | ||||
repo.ui, | ||||
cost, | ||||
batchget, | ||||
(repo, mctx, wctx, wantfiledata), | ||||
Pulkit Goyal
|
r45900 | list(mresult.getactions([mergestatemod.ACTION_GET], sort=True)), | ||
Augie Fackler
|
r43346 | threadsafe=threadsafe, | ||
hasretval=True, | ||||
) | ||||
Valentin Gatien-Baron
|
r42722 | getfiledata = {} | ||
Valentin Gatien-Baron
|
r42656 | for final, res in prog: | ||
if final: | ||||
getfiledata = res | ||||
else: | ||||
i, item = res | ||||
progress.increment(step=i, item=item) | ||||
Bryan O'Sullivan
|
r18630 | |||
Pulkit Goyal
|
r45897 | if b'.hgsubstate' in mresult._actionmapping[mergestatemod.ACTION_GET]: | ||
Yuya Nishihara
|
r36026 | subrepoutil.submerge(repo, wctx, mctx, wctx, overwrite, labels) | ||
Bryan O'Sullivan
|
r18632 | |||
Mads Kiilerich
|
r21551 | # forget (manifest only, just log it) (must come first) | ||
Pulkit Goyal
|
r45897 | for f, args, msg in mresult.getactions( | ||
(mergestatemod.ACTION_FORGET,), sort=True | ||||
): | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b" %s: %s -> f\n" % (f, msg)) | ||
Martin von Zweigbergk
|
r38364 | progress.increment(item=f) | ||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # re-add (manifest only, just log it) | ||
Pulkit Goyal
|
r45897 | for f, args, msg in mresult.getactions( | ||
(mergestatemod.ACTION_ADD,), sort=True | ||||
): | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b" %s: %s -> a\n" % (f, msg)) | ||
Martin von Zweigbergk
|
r38364 | progress.increment(item=f) | ||
Mads Kiilerich
|
r21391 | |||
Siddharth Agarwal
|
r27131 | # re-add/mark as modified (manifest only, just log it) | ||
Pulkit Goyal
|
r45897 | for f, args, msg in mresult.getactions( | ||
(mergestatemod.ACTION_ADD_MODIFIED,), sort=True | ||||
): | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b" %s: %s -> am\n" % (f, msg)) | ||
Martin von Zweigbergk
|
r38364 | progress.increment(item=f) | ||
Siddharth Agarwal
|
r27131 | |||
Mads Kiilerich
|
r21551 | # keep (noop, just log it) | ||
r49562 | for a in mergestatemod.MergeAction.NO_OP_ACTIONS: | |||
Pulkit Goyal
|
r46095 | for f, args, msg in mresult.getactions((a,), sort=True): | ||
r49560 | repo.ui.debug(b" %s: %s -> %s\n" % (f, msg, a.__bytes__())) | |||
Pulkit Goyal
|
r46095 | # no progress | ||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # directory rename, move local | ||
Pulkit Goyal
|
r45897 | for f, args, msg in mresult.getactions( | ||
(mergestatemod.ACTION_DIR_RENAME_MOVE_LOCAL,), sort=True | ||||
): | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b" %s: %s -> dm\n" % (f, msg)) | ||
Martin von Zweigbergk
|
r38364 | progress.increment(item=f) | ||
Mads Kiilerich
|
r21551 | f0, flags = args | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b"moving %s to %s\n") % (f0, f)) | ||
Phil Cohen
|
r33086 | wctx[f].audit() | ||
Phil Cohen
|
r33083 | wctx[f].write(wctx.filectx(f0).data(), flags) | ||
Phil Cohen
|
r33082 | wctx[f0].remove() | ||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # local directory rename, get | ||
Pulkit Goyal
|
r45897 | for f, args, msg in mresult.getactions( | ||
(mergestatemod.ACTION_LOCAL_DIR_RENAME_GET,), sort=True | ||||
): | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b" %s: %s -> dg\n" % (f, msg)) | ||
Martin von Zweigbergk
|
r38364 | progress.increment(item=f) | ||
Mads Kiilerich
|
r21551 | f0, flags = args | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b"getting %s to %s\n") % (f0, f)) | ||
Phil Cohen
|
r33083 | wctx[f].write(mctx.filectx(f0).data(), flags) | ||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # exec | ||
Pulkit Goyal
|
r45897 | for f, args, msg in mresult.getactions( | ||
(mergestatemod.ACTION_EXEC,), sort=True | ||||
): | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b" %s: %s -> e\n" % (f, msg)) | ||
Martin von Zweigbergk
|
r38364 | progress.increment(item=f) | ||
Augie Fackler
|
r43346 | (flags,) = args | ||
Phil Cohen
|
r33086 | wctx[f].audit() | ||
Augie Fackler
|
r43347 | wctx[f].setflags(b'l' in flags, b'x' in flags) | ||
Mads Kiilerich
|
r21391 | |||
Pulkit Goyal
|
r46060 | moves = [] | ||
# 'cd' and 'dc' actions are treated like other merge conflicts | ||||
mergeactions = list( | ||||
mresult.getactions( | ||||
[ | ||||
mergestatemod.ACTION_CHANGED_DELETED, | ||||
mergestatemod.ACTION_DELETED_CHANGED, | ||||
mergestatemod.ACTION_MERGE, | ||||
], | ||||
sort=True, | ||||
) | ||||
) | ||||
for f, args, msg in mergeactions: | ||||
f1, f2, fa, move, anc = args | ||||
if f == b'.hgsubstate': # merged internally | ||||
continue | ||||
if f1 is None: | ||||
fcl = filemerge.absentfilectx(wctx, fa) | ||||
else: | ||||
repo.ui.debug(b" preserving %s for resolve of %s\n" % (f1, f)) | ||||
fcl = wctx[f1] | ||||
if f2 is None: | ||||
fco = filemerge.absentfilectx(mctx, fa) | ||||
else: | ||||
fco = mctx[f2] | ||||
actx = repo[anc] | ||||
if fa in actx: | ||||
fca = actx[fa] | ||||
else: | ||||
# TODO: move to absentfilectx | ||||
fca = repo.filectx(f1, fileid=nullrev) | ||||
ms.add(fcl, fco, fca, f) | ||||
if f1 != f and move: | ||||
moves.append(f1) | ||||
# remove renamed files after safely stored | ||||
for f in moves: | ||||
if wctx[f].lexists(): | ||||
repo.ui.debug(b"removing %s\n" % f) | ||||
wctx[f].audit() | ||||
wctx[f].remove() | ||||
Pulkit Goyal
|
r45901 | # these actions updates the file | ||
updated = mresult.len( | ||||
( | ||||
mergestatemod.ACTION_GET, | ||||
mergestatemod.ACTION_EXEC, | ||||
mergestatemod.ACTION_LOCAL_DIR_RENAME_GET, | ||||
mergestatemod.ACTION_DIR_RENAME_MOVE_LOCAL, | ||||
) | ||||
) | ||||
Siddharth Agarwal
|
r26786 | |||
Ryan McElroy
|
r34681 | try: | ||
for f, args, msg in mergeactions: | ||||
Martin von Zweigbergk
|
r49256 | repo.ui.debug(b" %s: %s -> m\n" % (f, msg)) | ||
Pulkit Goyal
|
r47567 | ms.addcommitinfo(f, {b'merged': b'yes'}) | ||
Martin von Zweigbergk
|
r38364 | progress.increment(item=f) | ||
Augie Fackler
|
r43347 | if f == b'.hgsubstate': # subrepo states need updating | ||
Augie Fackler
|
r43346 | subrepoutil.submerge( | ||
repo, wctx, mctx, wctx.ancestor(mctx), overwrite, labels | ||||
) | ||||
Ryan McElroy
|
r34681 | continue | ||
wctx[f].audit() | ||||
ms.resolve(f, wctx) | ||||
Siddharth Agarwal
|
r26292 | |||
Kyle Lippincott
|
r49092 | except error.InterventionRequired: | ||
# If the user has merge.on-failure=halt, catch the error and close the | ||||
# merge state "properly". | ||||
pass | ||||
Ryan McElroy
|
r34681 | finally: | ||
ms.commit() | ||||
Siddharth Agarwal
|
r26787 | |||
Siddharth Agarwal
|
r27078 | unresolved = ms.unresolvedcount() | ||
msupdated, msmerged, msremoved = ms.counts() | ||||
updated += msupdated | ||||
merged += msmerged | ||||
removed += msremoved | ||||
Siddharth Agarwal
|
r27080 | |||
extraactions = ms.actions() | ||||
Martin von Zweigbergk
|
r38392 | progress.complete() | ||
Martin von Zweigbergk
|
r48163 | return ( | ||
updateresult(updated, merged, removed, unresolved), | ||||
getfiledata, | ||||
extraactions, | ||||
Augie Fackler
|
r45383 | ) | ||
Matt Mackall
|
r3111 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r45727 | def _advertisefsmonitor(repo, num_gets, p1node): | ||
# Advertise fsmonitor when its presence could be useful. | ||||
# | ||||
# We only advertise when performing an update from an empty working | ||||
# directory. This typically only occurs during initial clone. | ||||
# | ||||
# We give users a mechanism to disable the warning in case it is | ||||
# annoying. | ||||
# | ||||
# We only allow on Linux and MacOS because that's where fsmonitor is | ||||
# considered stable. | ||||
fsmonitorwarning = repo.ui.configbool(b'fsmonitor', b'warn_when_unused') | ||||
fsmonitorthreshold = repo.ui.configint( | ||||
b'fsmonitor', b'warn_update_file_count' | ||||
) | ||||
Valentin Gatien-Baron
|
r46034 | # avoid cycle dirstate -> sparse -> merge -> dirstate | ||
Raphaël Gomès
|
r49660 | dirstate_rustmod = policy.importrust("dirstate") | ||
Valentin Gatien-Baron
|
r46034 | |||
Raphaël Gomès
|
r49660 | if dirstate_rustmod is not None: | ||
Valentin Gatien-Baron
|
r46034 | # When using rust status, fsmonitor becomes necessary at higher sizes | ||
fsmonitorthreshold = repo.ui.configint( | ||||
Augie Fackler
|
r46554 | b'fsmonitor', | ||
b'warn_update_file_count_rust', | ||||
Valentin Gatien-Baron
|
r46034 | ) | ||
Pulkit Goyal
|
r45727 | try: | ||
# avoid cycle: extensions -> cmdutil -> merge | ||||
from . import extensions | ||||
extensions.find(b'fsmonitor') | ||||
fsmonitorenabled = repo.ui.config(b'fsmonitor', b'mode') != b'off' | ||||
# We intentionally don't look at whether fsmonitor has disabled | ||||
# itself because a) fsmonitor may have already printed a warning | ||||
# b) we only care about the config state here. | ||||
except KeyError: | ||||
fsmonitorenabled = False | ||||
if ( | ||||
fsmonitorwarning | ||||
and not fsmonitorenabled | ||||
Joerg Sonnenberger
|
r47771 | and p1node == repo.nullid | ||
Pulkit Goyal
|
r45727 | and num_gets >= fsmonitorthreshold | ||
and pycompat.sysplatform.startswith((b'linux', b'darwin')) | ||||
): | ||||
repo.ui.warn( | ||||
_( | ||||
b'(warning: large working directory being used without ' | ||||
b'fsmonitor enabled; enable fsmonitor to improve performance; ' | ||||
b'see "hg help -e fsmonitor")\n' | ||||
) | ||||
) | ||||
Augie Fackler
|
r43347 | UPDATECHECK_ABORT = b'abort' # handled at higher layers | ||
UPDATECHECK_NONE = b'none' | ||||
UPDATECHECK_LINEAR = b'linear' | ||||
UPDATECHECK_NO_CONFLICT = b'noconflict' | ||||
Augie Fackler
|
r43240 | |||
Raphaël Gomès
|
r52953 | # Let extensions turn off any Rust code in the update code if that interferes | ||
# will their patching. | ||||
# This being `True` does not mean that you have Rust extensions installed or | ||||
# that the Rust path will be taken for any given invocation. | ||||
MAYBE_USE_RUST_UPDATE = True | ||||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r46134 | def _update( | ||
Augie Fackler
|
r43346 | repo, | ||
node, | ||||
branchmerge, | ||||
force, | ||||
ancestor=None, | ||||
mergeancestor=False, | ||||
labels=None, | ||||
matcher=None, | ||||
mergeforce=False, | ||||
Martin von Zweigbergk
|
r44744 | updatedirstate=True, | ||
Augie Fackler
|
r43346 | updatecheck=None, | ||
wc=None, | ||||
): | ||||
Matt Mackall
|
r3315 | """ | ||
Perform a merge between the working directory and the given node | ||||
Martin von Zweigbergk
|
r30902 | node = the node to update to | ||
Matt Mackall
|
r3315 | branchmerge = whether to merge between branches | ||
force = whether to force branch merging or file overwriting | ||||
Augie Fackler
|
r27344 | matcher = a matcher to filter file lists (dirstate not updated) | ||
Durham Goode
|
r18778 | mergeancestor = whether it is merging with an ancestor. If true, | ||
we should accept the incoming changes for any prompts that occur. | ||||
If false, merging with an ancestor (fast-forward) is only allowed | ||||
between different named branches. This flag is used by rebase extension | ||||
as a temporary fix and should be avoided in general. | ||||
Martin von Zweigbergk
|
r49601 | labels = labels to use for local, other, and base | ||
Siddharth Agarwal
|
r28020 | mergeforce = whether the merge was run with 'merge --force' (deprecated): if | ||
this is True, then 'force' should be True as well. | ||||
Stuart W Marks
|
r9716 | |||
muxator
|
r34920 | The table below shows all the behaviors of the update command given the | ||
-c/--check and -C/--clean or no options, whether the working directory is | ||||
dirty, whether a revision is specified, and the relationship of the parent | ||||
rev to the target rev (linear or not). Match from top first. The -n | ||||
option doesn't exist on the command line, but represents the | ||||
Martin von Zweigbergk
|
r31168 | experimental.updatecheck=noconflict option. | ||
Stuart W Marks
|
r9716 | |||
Adrian Buehlmann
|
r12279 | This logic is tested by test-update-branches.t. | ||
Stuart W Marks
|
r9716 | |||
Martin von Zweigbergk
|
r31168 | -c -C -n -m dirty rev linear | result | ||
y y * * * * * | (1) | ||||
y * y * * * * | (1) | ||||
y * * y * * * | (1) | ||||
* y y * * * * | (1) | ||||
* y * y * * * | (1) | ||||
* * y y * * * | (1) | ||||
* * * * * n n | x | ||||
* * * * n * * | ok | ||||
n n n n y * y | merge | ||||
n n n n y y n | (2) | ||||
n n n y y * * | merge | ||||
n n y n y * * | merge if no conflict | ||||
n y n n y * * | discard | ||||
y n n n y * * | (3) | ||||
Stuart W Marks
|
r9716 | |||
x = can't happen | ||||
* = don't-care | ||||
Martin von Zweigbergk
|
r31161 | 1 = incompatible options (checked in commands.py) | ||
2 = abort: uncommitted changes (commit or update --clean to discard changes) | ||||
3 = abort: uncommitted changes (checked in commands.py) | ||||
Greg Ward
|
r13162 | |||
Phil Cohen
|
r34303 | The merge is performed inside ``wc``, a workingctx-like objects. It defaults | ||
to repo[None] if None is passed. | ||||
Greg Ward
|
r13162 | Return the same tuple as applyupdates(). | ||
Matt Mackall
|
r3315 | """ | ||
Gregory Szorc
|
r33321 | # Avoid cycle. | ||
from . import sparse | ||||
Matt Mackall
|
r2815 | |||
Martin von Zweigbergk
|
r31166 | # This function used to find the default destination if node was None, but | ||
Martin von Zweigbergk
|
r30902 | # that's now in destutil.py. | ||
assert node is not None | ||||
Martin von Zweigbergk
|
r31166 | if not branchmerge and not force: | ||
# TODO: remove the default once all callers that pass branchmerge=False | ||||
# and force=False pass a value for updatecheck. We may want to allow | ||||
# updatecheck='abort' to better suppport some of these callers. | ||||
if updatecheck is None: | ||||
Augie Fackler
|
r43240 | updatecheck = UPDATECHECK_LINEAR | ||
r49552 | okay = (UPDATECHECK_NONE, UPDATECHECK_LINEAR, UPDATECHECK_NO_CONFLICT) | |||
if updatecheck not in okay: | ||||
msg = r'Invalid updatecheck %r (can accept %r)' | ||||
msg %= (updatecheck, okay) | ||||
raise ValueError(msg) | ||||
Martin von Zweigbergk
|
r45536 | if wc is not None and wc.isinmemory(): | ||
maybe_wlock = util.nullcontextmanager() | ||||
else: | ||||
maybe_wlock = repo.wlock() | ||||
with maybe_wlock: | ||||
Phil Cohen
|
r34303 | if wc is None: | ||
wc = repo[None] | ||||
Sean Farley
|
r20279 | pl = wc.parents() | ||
p1 = pl[0] | ||||
Martin von Zweigbergk
|
r42601 | p2 = repo[node] | ||
Mads Kiilerich
|
r23405 | if ancestor is not None: | ||
Mads Kiilerich
|
r21081 | pas = [repo[ancestor]] | ||
Martin von Zweigbergk
|
r42602 | else: | ||
Augie Fackler
|
r43347 | if repo.ui.configlist(b'merge', b'preferancestor') == [b'*']: | ||
Mads Kiilerich
|
r21128 | cahs = repo.changelog.commonancestorsheads(p1.node(), p2.node()) | ||
Joerg Sonnenberger
|
r47771 | pas = [repo[anc] for anc in (sorted(cahs) or [repo.nullid])] | ||
Mads Kiilerich
|
r21128 | else: | ||
Mads Kiilerich
|
r22179 | pas = [p1.ancestor(p2, warn=branchmerge)] | ||
Matt Mackall
|
r13874 | |||
Augie Fackler
|
r36195 | fp1, fp2, xp1, xp2 = p1.node(), p2.node(), bytes(p1), bytes(p2) | ||
Matt Mackall
|
r3314 | |||
Martin von Zweigbergk
|
r42601 | overwrite = force and not branchmerge | ||
Matt Mackall
|
r4915 | ### check phase | ||
Martin von Zweigbergk
|
r27316 | if not overwrite: | ||
if len(pl) > 1: | ||||
Martin von Zweigbergk
|
r47146 | raise error.StateError(_(b"outstanding uncommitted merge")) | ||
Martin von Zweigbergk
|
r46071 | ms = wc.mergestate() | ||
Augie Fackler
|
r47070 | if ms.unresolvedcount(): | ||
r49552 | msg = _(b"outstanding merge conflicts") | |||
hint = _(b"use 'hg resolve' to resolve") | ||||
raise error.StateError(msg, hint=hint) | ||||
Matt Mackall
|
r6375 | if branchmerge: | ||
r49553 | m_a = _(b"merging with a working directory ancestor has no effect") | |||
Mads Kiilerich
|
r21081 | if pas == [p2]: | ||
r49553 | raise error.Abort(m_a) | |||
Mads Kiilerich
|
r21081 | elif pas == [p1]: | ||
Mads Kiilerich
|
r31379 | if not mergeancestor and wc.branch() == p2.branch(): | ||
r49554 | msg = _(b"nothing to merge") | |||
hint = _(b"use 'hg update' or check 'hg heads'") | ||||
raise error.Abort(msg, hint=hint) | ||||
Matt Mackall
|
r6375 | if not force and (wc.files() or wc.deleted()): | ||
r49555 | msg = _(b"uncommitted changes") | |||
hint = _(b"use 'hg status' to list changes") | ||||
raise error.StateError(msg, hint=hint) | ||||
Phil Cohen
|
r35285 | if not wc.isinmemory(): | ||
for s in sorted(wc.substate): | ||||
wc.sub(s).bailifchanged() | ||||
Oleg Stepanov
|
r13437 | |||
Matt Mackall
|
r6375 | elif not overwrite: | ||
Augie Fackler
|
r43346 | if p1 == p2: # no-op update | ||
Siddharth Agarwal
|
r19929 | # call the hooks and exit early | ||
Augie Fackler
|
r43347 | repo.hook(b'preupdate', throw=True, parent1=xp2, parent2=b'') | ||
repo.hook(b'update', parent1=xp2, parent2=b'', error=0) | ||||
Gregory Szorc
|
r37125 | return updateresult(0, 0, 0, 0) | ||
Siddharth Agarwal
|
r19929 | |||
Augie Fackler
|
r43346 | if updatecheck == UPDATECHECK_LINEAR and pas not in ( | ||
[p1], | ||||
[p2], | ||||
): # nonlinear | ||||
Pierre-Yves David
|
r18985 | dirty = wc.dirty(missing=True) | ||
Martin von Zweigbergk
|
r30902 | if dirty: | ||
Pierre-Yves David
|
r18985 | # Branching is a bit strange to ensure we do the minimal | ||
r33147 | # amount of call to obsutil.foreground. | |||
foreground = obsutil.foreground(repo, [p1.node()]) | ||||
Pierre-Yves David
|
r18985 | # note: the <node> variable contains a random identifier | ||
if repo[node].node() in foreground: | ||||
Augie Fackler
|
r43346 | pass # allow updating to successors | ||
Martin von Zweigbergk
|
r30902 | else: | ||
Augie Fackler
|
r43347 | msg = _(b"uncommitted changes") | ||
hint = _(b"commit or update --clean to discard changes") | ||||
Martin von Zweigbergk
|
r30961 | raise error.UpdateAbort(msg, hint=hint) | ||
Pierre-Yves David
|
r18985 | else: | ||
# Allow jumping branches if clean and specific rev given | ||||
Martin von Zweigbergk
|
r30901 | pass | ||
if overwrite: | ||||
pas = [wc] | ||||
elif not branchmerge: | ||||
pas = [p1] | ||||
Matt Mackall
|
r2814 | |||
Matt Mackall
|
r25843 | # deprecated config: merge.followcopies | ||
Augie Fackler
|
r43347 | followcopies = repo.ui.configbool(b'merge', b'followcopies') | ||
Mads Kiilerich
|
r21080 | if overwrite: | ||
Gábor Stefanik
|
r30200 | followcopies = False | ||
elif not pas[0]: | ||||
followcopies = False | ||||
if not branchmerge and not wc.dirty(missing=True): | ||||
followcopies = False | ||||
Mads Kiilerich
|
r21080 | |||
Raphaël Gomès
|
r52953 | update_from_null = False | ||
update_from_null_fallback = False | ||||
if ( | ||||
MAYBE_USE_RUST_UPDATE | ||||
Raphaël Gomès
|
r52954 | and repo.ui.configbool(b"rust", b"update-from-null") | ||
Raphaël Gomès
|
r52953 | and rust_update_mod is not None | ||
and p1.rev() == nullrev | ||||
and not branchmerge | ||||
# TODO it's probably not too hard to pass down the transaction and | ||||
# respect the write patterns from Rust. But since it doesn't affect | ||||
# a simple update from null, then it doesn't matter yet. | ||||
and repo.currenttransaction() is None | ||||
and matcher is None | ||||
and not wc.mergestate().active() | ||||
and b'.hgsubstate' not in p2 | ||||
): | ||||
working_dir_iter = os.scandir(repo.root) | ||||
maybe_hg_folder = next(working_dir_iter) | ||||
assert maybe_hg_folder is not None | ||||
if maybe_hg_folder.name == b".hg": | ||||
try: | ||||
next(working_dir_iter) | ||||
except StopIteration: | ||||
update_from_null = True | ||||
if update_from_null: | ||||
# Check the narrowspec and sparsespec here to display warnings | ||||
# more easily. | ||||
# TODO figure out of a way of bubbling up warnings to Python | ||||
# while not polluting the Rust code (probably a channel) | ||||
repo.narrowmatch() | ||||
sparse.matcher(repo, [nullrev, p2.rev()]) | ||||
repo.hook(b'preupdate', throw=True, parent1=xp1, parent2=xp2) | ||||
# note that we're in the middle of an update | ||||
repo.vfs.write(b'updatestate', p2.hex()) | ||||
try: | ||||
updated_count = rust_update_mod.update_from_null( | ||||
repo.root, p2.rev() | ||||
) | ||||
except rust_update_mod.FallbackError: | ||||
update_from_null_fallback = True | ||||
else: | ||||
# We've changed the dirstate from Rust, we need to tell Python | ||||
repo.dirstate.invalidate() | ||||
# This includes setting the parents, since they are not read | ||||
# again on invalidation | ||||
with repo.dirstate.changing_parents(repo): | ||||
repo.dirstate.setparents(fp2) | ||||
repo.dirstate.setbranch(p2.branch(), repo.currenttransaction()) | ||||
sparse.prunetemporaryincludes(repo) | ||||
repo.hook(b'update', parent1=xp1, parent2=xp2, error=0) | ||||
# update completed, clear state | ||||
util.unlink(repo.vfs.join(b'updatestate')) | ||||
return updateresult(updated_count, 0, 0, 0) | ||||
Matt Mackall
|
r4915 | ### calculate phase | ||
Pulkit Goyal
|
r45831 | mresult = calculateupdates( | ||
Augie Fackler
|
r43346 | repo, | ||
wc, | ||||
p2, | ||||
pas, | ||||
branchmerge, | ||||
force, | ||||
mergeancestor, | ||||
followcopies, | ||||
matcher=matcher, | ||||
mergeforce=mergeforce, | ||||
) | ||||
Siddharth Agarwal
|
r27951 | |||
Augie Fackler
|
r43240 | if updatecheck == UPDATECHECK_NO_CONFLICT: | ||
Pulkit Goyal
|
r45835 | if mresult.hasconflicts(): | ||
msg = _(b"conflicting changes") | ||||
hint = _(b"commit or update --clean to discard changes") | ||||
Martin von Zweigbergk
|
r47146 | raise error.StateError(msg, hint=hint) | ||
Martin von Zweigbergk
|
r31168 | |||
Siddharth Agarwal
|
r27951 | # Prompt and create actions. Most of this is in the resolve phase | ||
# already, but we can't handle .hgsubstate in filemerge or | ||||
Yuya Nishihara
|
r36026 | # subrepoutil.submerge yet so we have to keep prompting for it. | ||
Pulkit Goyal
|
r45904 | vals = mresult.getfile(b'.hgsubstate') | ||
if vals: | ||||
Augie Fackler
|
r43347 | f = b'.hgsubstate' | ||
Pulkit Goyal
|
r45904 | m, args, msg = vals | ||
Simon Farnsworth
|
r29774 | prompts = filemerge.partextras(labels) | ||
Augie Fackler
|
r43347 | prompts[b'f'] = f | ||
Augie Fackler
|
r45383 | if m == mergestatemod.ACTION_CHANGED_DELETED: | ||
Siddharth Agarwal
|
r27951 | if repo.ui.promptchoice( | ||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b"local%(l)s changed %(f)s which other%(o)s deleted\n" | ||
b"use (c)hanged version or (d)elete?" | ||||
b"$$ &Changed $$ &Delete" | ||||
Augie Fackler
|
r43346 | ) | ||
% prompts, | ||||
0, | ||||
): | ||||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
Augie Fackler
|
r46554 | f, | ||
mergestatemod.ACTION_REMOVE, | ||||
None, | ||||
b'prompt delete', | ||||
Augie Fackler
|
r45383 | ) | ||
Siddharth Agarwal
|
r27951 | elif f in p1: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_ADD_MODIFIED, | ||
Augie Fackler
|
r43347 | None, | ||
b'prompt keep', | ||||
) | ||||
Siddharth Agarwal
|
r27951 | else: | ||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
Augie Fackler
|
r46554 | f, | ||
mergestatemod.ACTION_ADD, | ||||
None, | ||||
b'prompt keep', | ||||
Augie Fackler
|
r45383 | ) | ||
elif m == mergestatemod.ACTION_DELETED_CHANGED: | ||||
Siddharth Agarwal
|
r27951 | f1, f2, fa, move, anc = args | ||
flags = p2[f2].flags() | ||||
Augie Fackler
|
r43346 | if ( | ||
repo.ui.promptchoice( | ||||
_( | ||||
Augie Fackler
|
r43347 | b"other%(o)s changed %(f)s which local%(l)s deleted\n" | ||
b"use (c)hanged version or leave (d)eleted?" | ||||
b"$$ &Changed $$ &Deleted" | ||||
Augie Fackler
|
r43346 | ) | ||
% prompts, | ||||
0, | ||||
) | ||||
== 0 | ||||
): | ||||
Pulkit Goyal
|
r45839 | mresult.addfile( | ||
f, | ||||
Augie Fackler
|
r45383 | mergestatemod.ACTION_GET, | ||
Augie Fackler
|
r43346 | (flags, False), | ||
Augie Fackler
|
r43347 | b'prompt recreating', | ||
Augie Fackler
|
r43346 | ) | ||
Siddharth Agarwal
|
r27951 | else: | ||
Pulkit Goyal
|
r45840 | mresult.removefile(f) | ||
Siddharth Agarwal
|
r27951 | |||
Martin von Zweigbergk
|
r29889 | if not util.fscasesensitive(repo.path): | ||
Martin von Zweigbergk
|
r23544 | # check collision between files only in p2 for clean update | ||
Augie Fackler
|
r43346 | if not branchmerge and ( | ||
force or not wc.dirty(missing=True, branch=False) | ||||
): | ||||
Martin von Zweigbergk
|
r23544 | _checkcollision(repo, p2.manifest(), None) | ||
else: | ||||
Pulkit Goyal
|
r45893 | _checkcollision(repo, wc.manifest(), mresult) | ||
Martin von Zweigbergk
|
r23544 | |||
Martin von Zweigbergk
|
r23525 | # divergent renames | ||
Gregory Szorc
|
r49768 | for f, fl in sorted(mresult.diverge.items()): | ||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b"note: possible conflict - %s was renamed " | ||
b"multiple times to:\n" | ||||
Augie Fackler
|
r43346 | ) | ||
% f | ||||
) | ||||
Martin von Zweigbergk
|
r42279 | for nf in sorted(fl): | ||
Augie Fackler
|
r43347 | repo.ui.warn(b" %s\n" % nf) | ||
Martin von Zweigbergk
|
r23525 | |||
# rename and delete | ||||
Gregory Szorc
|
r49768 | for f, fl in sorted(mresult.renamedelete.items()): | ||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b"note: possible conflict - %s was deleted " | ||
b"and renamed to:\n" | ||||
Augie Fackler
|
r43346 | ) | ||
% f | ||||
) | ||||
Martin von Zweigbergk
|
r42279 | for nf in sorted(fl): | ||
Augie Fackler
|
r43347 | repo.ui.warn(b" %s\n" % nf) | ||
Martin von Zweigbergk
|
r23525 | |||
Martin von Zweigbergk
|
r26957 | ### apply phase | ||
Augie Fackler
|
r43346 | if not branchmerge: # just jump to the new rev | ||
Joerg Sonnenberger
|
r47771 | fp1, fp2, xp1, xp2 = fp2, repo.nullid, xp2, b'' | ||
Martin von Zweigbergk
|
r44625 | # If we're doing a partial update, we need to skip updating | ||
Martin von Zweigbergk
|
r44626 | # the dirstate. | ||
always = matcher is None or matcher.always() | ||||
Martin von Zweigbergk
|
r44744 | updatedirstate = updatedirstate and always and not wc.isinmemory() | ||
Raphaël Gomès
|
r52953 | # If we're in the fallback case, we've already done this | ||
if updatedirstate and not update_from_null_fallback: | ||||
Augie Fackler
|
r43347 | repo.hook(b'preupdate', throw=True, parent1=xp1, parent2=xp2) | ||
Martin von Zweigbergk
|
r26957 | # note that we're in the middle of an update | ||
Augie Fackler
|
r43347 | repo.vfs.write(b'updatestate', p2.hex()) | ||
Martin von Zweigbergk
|
r26957 | |||
Raphaël Gomès
|
r52953 | # TODO don't run if Rust is available | ||
Pulkit Goyal
|
r45727 | _advertisefsmonitor( | ||
Pulkit Goyal
|
r45898 | repo, mresult.len((mergestatemod.ACTION_GET,)), p1.node() | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r34886 | |||
Valentin Gatien-Baron
|
r42656 | wantfiledata = updatedirstate and not branchmerge | ||
Martin von Zweigbergk
|
r48163 | stats, getfiledata, extraactions = applyupdates( | ||
Augie Fackler
|
r46554 | repo, | ||
mresult, | ||||
wc, | ||||
p2, | ||||
overwrite, | ||||
wantfiledata, | ||||
labels=labels, | ||||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r26957 | |||
Valentin Gatien-Baron
|
r42656 | if updatedirstate: | ||
Martin von Zweigbergk
|
r48163 | if extraactions: | ||
Gregory Szorc
|
r49768 | for k, acts in extraactions.items(): | ||
Martin von Zweigbergk
|
r48163 | for a in acts: | ||
mresult.addfile(a[0], k, *a[1:]) | ||||
if k == mergestatemod.ACTION_GET and wantfiledata: | ||||
# no filedata until mergestate is updated to provide it | ||||
for a in acts: | ||||
getfiledata[a[0]] = None | ||||
assert len(getfiledata) == ( | ||||
mresult.len((mergestatemod.ACTION_GET,)) if wantfiledata else 0 | ||||
) | ||||
r50855 | with repo.dirstate.changing_parents(repo): | |||
r49203 | if getfiledata: | |||
Raphaël Gomès
|
r52951 | getfiledata = filter_ambiguous_files(repo, getfiledata) | ||
r49203 | ||||
Augie Fackler
|
r32351 | repo.setparents(fp1, fp2) | ||
Augie Fackler
|
r45383 | mergestatemod.recordupdates( | ||
Pulkit Goyal
|
r45894 | repo, mresult.actionsdict, branchmerge, getfiledata | ||
Augie Fackler
|
r45383 | ) | ||
Augie Fackler
|
r32351 | if not branchmerge: | ||
r51155 | repo.dirstate.setbranch( | |||
p2.branch(), repo.currenttransaction() | ||||
) | ||||
Sune Foldager
|
r10492 | |||
Raphaël Gomès
|
r52953 | # update completed, clear state | ||
util.unlink(repo.vfs.join(b'updatestate')) | ||||
r48409 | # If we're updating to a location, clean up any stale temporary includes | |||
# (ex: this happens during hg rebase --abort). | ||||
if not branchmerge: | ||||
sparse.prunetemporaryincludes(repo) | ||||
Gregory Szorc
|
r33321 | |||
Martin von Zweigbergk
|
r44611 | if updatedirstate: | ||
Augie Fackler
|
r43346 | repo.hook( | ||
Augie Fackler
|
r43347 | b'update', parent1=xp1, parent2=xp2, error=stats.unresolvedcount | ||
Augie Fackler
|
r43346 | ) | ||
Sune Foldager
|
r10492 | return stats | ||
Matt Mackall
|
r22902 | |||
Augie Fackler
|
r43346 | |||
Raphaël Gomès
|
r52951 | # filename -> (mode, size, timestamp) | ||
FileData = Dict[bytes, Optional[Tuple[int, int, Optional[timestamp.timestamp]]]] | ||||
def filter_ambiguous_files(repo, file_data: FileData) -> Optional[FileData]: | ||||
"""We've gathered "cache" information for the clean files while updating | ||||
them: their mtime, size and mode. | ||||
At the time this comment is written, there are various issues with how we | ||||
gather the `mode` and `mtime` information (see the comment in `batchget`). | ||||
We are going to smooth one of these issues here: mtime ambiguity. | ||||
i.e. even if the mtime gathered during `batchget` was correct[1] a change | ||||
happening right after it could change the content while keeping | ||||
the same mtime[2]. | ||||
When we reach the current code, the "on disk" part of the update operation | ||||
is finished. We still assume that no other process raced that "on disk" | ||||
part, but we want to at least prevent later file changes to alter the | ||||
contents of the file right after the update operation so quickly that the | ||||
same mtime is recorded for the operation. | ||||
Raphaël Gomès
|
r52952 | |||
To prevent such ambiguities from happenning, we will do (up to) two things: | ||||
- wait until the filesystem clock has ticked | ||||
- only keep the "file data" for files with mtimes that are strictly in | ||||
the past, i.e. whose mtime is strictly lower than the current time. | ||||
We only wait for the system clock to tick if using dirstate-v2, since v1 | ||||
only has second-level granularity and waiting for a whole second is | ||||
too much of a penalty in the general case. | ||||
Although we're assuming that people running dirstate-v2 on Linux | ||||
don't have a second-granularity FS (with the exclusion of NFS), users | ||||
can be surprising, and at some point in the future, dirstate-v2 will become | ||||
the default. To that end, we limit the wait time to 100ms and fall back | ||||
to the filtering method in case of a timeout. | ||||
+------------+------+--------------+ | ||||
| version | wait | filter level | | ||||
+------------+------+--------------+ | ||||
| V1 | No | Second | | ||||
| V2 | Yes | Nanosecond | | ||||
| V2-slow-fs | No | Second | | ||||
+------------+------+--------------+ | ||||
Raphaël Gomès
|
r52951 | |||
This protects us from race conditions from operations that could run right | ||||
after this one, especially other Mercurial operations that could be waiting | ||||
for the wlock to touch files contents and the dirstate. | ||||
In an ideal world, we could only get reliable information in `getfiledata` | ||||
Raphaël Gomès
|
r52952 | (from `getbatch`), however this filtering approach has been a successful | ||
compromise for many years. A patch series of the linux kernel might change | ||||
this in 6.12³. | ||||
Raphaël Gomès
|
r52951 | |||
At the time this comment is written, not using any "cache" file data at all | ||||
here would not be viable, as it would result is a very large amount of work | ||||
(equivalent to the previous `hg update` during the next status after an | ||||
update). | ||||
[1] the current code cannot grantee that the `mtime` and `mode` | ||||
are correct, but the result is "okay in practice". | ||||
(see the comment in `batchget`) | ||||
[2] using nano-second precision can greatly help here because it makes the | ||||
"different write with same mtime" issue virtually vanish. However, | ||||
dirstate v1 cannot store such precision and a bunch of python-runtime, | ||||
operating-system and filesystem parts do not provide us with such | ||||
Raphaël Gomès
|
r52952 | precision, so we have to operate as if it wasn't available. | ||
[3] https://lore.kernel.org/all/20241002-mgtime-v10-8-d1c4717f5284@kernel.org | ||||
""" | ||||
Raphaël Gomès
|
r52951 | ambiguous_mtime: FileData = {} | ||
Raphaël Gomès
|
r52952 | dirstate_v2 = repo.dirstate._use_dirstate_v2 | ||
fs_now_result = None | ||||
fast_enough_fs = True | ||||
if dirstate_v2: | ||||
fstype = util.getfstype(repo.vfs.base) | ||||
# Exclude NFS right off the bat | ||||
fast_enough_fs = fstype != b'nfs' | ||||
if fstype is not None and fast_enough_fs: | ||||
fs_now_result = timestamp.wait_until_fs_tick(repo.vfs) | ||||
if fs_now_result is None: | ||||
try: | ||||
now = timestamp.get_fs_now(repo.vfs) | ||||
fs_now_result = (now, False) | ||||
except OSError: | ||||
pass | ||||
Raphaël Gomès
|
r52951 | if fs_now_result is None: | ||
# we can't write to the FS, so we won't actually update | ||||
# the dirstate content anyway, no need to put cache | ||||
# information. | ||||
return None | ||||
else: | ||||
now, timed_out = fs_now_result | ||||
if timed_out: | ||||
fast_enough_fs = False | ||||
for f, m in file_data.items(): | ||||
Raphaël Gomès
|
r52952 | if m is not None: | ||
reliable = timestamp.make_mtime_reliable(m[2], now) | ||||
if reliable is None or ( | ||||
reliable[2] and (not dirstate_v2 or not fast_enough_fs) | ||||
): | ||||
# Either it's not reliable, or it's second ambiguous | ||||
# and we're in dirstate-v1 or in a slow fs, so discard | ||||
# the mtime. | ||||
ambiguous_mtime[f] = (m[0], m[1], None) | ||||
elif reliable[2]: | ||||
# We need to remember that this time is "second ambiguous" | ||||
# otherwise the next status might miss a subsecond change | ||||
# if its "stat" doesn't provide nanoseconds. | ||||
# | ||||
# TODO make osutil.c understand nanoseconds when possible | ||||
# (see timestamp.py for the same note) | ||||
ambiguous_mtime[f] = (m[0], m[1], reliable) | ||||
Raphaël Gomès
|
r52951 | for f, m in ambiguous_mtime.items(): | ||
file_data[f] = m | ||||
return file_data | ||||
Martin von Zweigbergk
|
r44883 | def merge(ctx, labels=None, force=False, wc=None): | ||
"""Merge another topological branch into the working copy. | ||||
force = whether the merge was run with 'merge --force' (deprecated) | ||||
""" | ||||
Martin von Zweigbergk
|
r46134 | return _update( | ||
Martin von Zweigbergk
|
r44883 | ctx.repo(), | ||
ctx.rev(), | ||||
labels=labels, | ||||
branchmerge=True, | ||||
force=force, | ||||
mergeforce=force, | ||||
wc=wc, | ||||
) | ||||
Martin von Zweigbergk
|
r46150 | def update(ctx, updatecheck=None, wc=None): | ||
"""Do a regular update to the given commit, aborting if there are conflicts. | ||||
The 'updatecheck' argument can be used to control what to do in case of | ||||
conflicts. | ||||
Note: This is a new, higher-level update() than the one that used to exist | ||||
in this module. That function is now called _update(). You can hopefully | ||||
replace your callers to use this new update(), or clean_update(), merge(), | ||||
revert_to(), or graft(). | ||||
""" | ||||
return _update( | ||||
ctx.repo(), | ||||
ctx.rev(), | ||||
branchmerge=False, | ||||
force=False, | ||||
Martin von Zweigbergk
|
r49436 | labels=[b'working copy', b'destination', b'working copy parent'], | ||
Martin von Zweigbergk
|
r46150 | updatecheck=updatecheck, | ||
wc=wc, | ||||
) | ||||
Martin von Zweigbergk
|
r44743 | def clean_update(ctx, wc=None): | ||
"""Do a clean update to the given commit. | ||||
This involves updating to the commit and discarding any changes in the | ||||
working copy. | ||||
""" | ||||
Martin von Zweigbergk
|
r46134 | return _update(ctx.repo(), ctx.rev(), branchmerge=False, force=True, wc=wc) | ||
Martin von Zweigbergk
|
r44743 | |||
Martin von Zweigbergk
|
r44744 | def revert_to(ctx, matcher=None, wc=None): | ||
"""Revert the working copy to the given commit. | ||||
The working copy will keep its current parent(s) but its content will | ||||
be the same as in the given commit. | ||||
""" | ||||
Martin von Zweigbergk
|
r46134 | return _update( | ||
Martin von Zweigbergk
|
r44744 | ctx.repo(), | ||
ctx.rev(), | ||||
branchmerge=False, | ||||
force=True, | ||||
updatedirstate=False, | ||||
matcher=matcher, | ||||
wc=wc, | ||||
) | ||||
Augie Fackler
|
r43346 | def graft( | ||
Martin von Zweigbergk
|
r44692 | repo, | ||
ctx, | ||||
Martin von Zweigbergk
|
r44693 | base=None, | ||
Martin von Zweigbergk
|
r44692 | labels=None, | ||
keepparent=False, | ||||
keepconflictparent=False, | ||||
wctx=None, | ||||
Augie Fackler
|
r43346 | ): | ||
Matt Mackall
|
r22902 | """Do a graft-like merge. | ||
This is a merge where the merge ancestor is chosen such that one | ||||
or more changesets are grafted onto the current changeset. In | ||||
addition to the merge, this fixes up the dirstate to include only | ||||
Andrew Halberstadt
|
r27267 | a single parent (if keepparent is False) and tries to duplicate any | ||
renames/copies appropriately. | ||||
Matt Mackall
|
r22902 | |||
ctx - changeset to rebase | ||||
Martin von Zweigbergk
|
r44693 | base - merge base, or ctx.p1() if not specified | ||
Matt Mackall
|
r22902 | labels - merge labels eg ['local', 'graft'] | ||
Andrew Halberstadt
|
r27267 | keepparent - keep second parent if any | ||
r42604 | keepconflictparent - if unresolved, keep parent used for the merge | |||
Matt Mackall
|
r22902 | |||
""" | ||||
Durham Goode
|
r24643 | # If we're grafting a descendant onto an ancestor, be sure to pass | ||
# mergeancestor=True to update. This does two things: 1) allows the merge if | ||||
# the destination is the same as the parent of the ctx (so we can use graft | ||||
# to copy commits), and 2) informs update that the incoming changes are | ||||
# newer than the destination so it doesn't prompt about "remote changed foo | ||||
# which local deleted". | ||||
Valentin Gatien-Baron
|
r44867 | # We also pass mergeancestor=True when base is the same revision as p1. 2) | ||
# doesn't matter as there can't possibly be conflicts, but 1) is necessary. | ||||
Martin von Zweigbergk
|
r44692 | wctx = wctx or repo[None] | ||
Martin von Zweigbergk
|
r44548 | pctx = wctx.p1() | ||
Martin von Zweigbergk
|
r44693 | base = base or ctx.p1() | ||
Valentin Gatien-Baron
|
r44867 | mergeancestor = ( | ||
repo.changelog.isancestor(pctx.node(), ctx.node()) | ||||
or pctx.rev() == base.rev() | ||||
) | ||||
Matt Mackall
|
r22902 | |||
Martin von Zweigbergk
|
r46134 | stats = _update( | ||
Augie Fackler
|
r43346 | repo, | ||
ctx.node(), | ||||
True, | ||||
True, | ||||
Martin von Zweigbergk
|
r44235 | base.node(), | ||
Augie Fackler
|
r43346 | mergeancestor=mergeancestor, | ||
labels=labels, | ||||
Martin von Zweigbergk
|
r44692 | wc=wctx, | ||
Augie Fackler
|
r43346 | ) | ||
Boris Feld
|
r38513 | |||
if keepconflictparent and stats.unresolvedcount: | ||||
pother = ctx.node() | ||||
else: | ||||
Joerg Sonnenberger
|
r47771 | pother = repo.nullid | ||
Boris Feld
|
r38513 | parents = ctx.parents() | ||
Martin von Zweigbergk
|
r44235 | if keepparent and len(parents) == 2 and base in parents: | ||
parents.remove(base) | ||||
Boris Feld
|
r38513 | pother = parents[0].node() | ||
Martin von Zweigbergk
|
r44237 | # Never set both parents equal to each other | ||
if pother == pctx.node(): | ||||
Joerg Sonnenberger
|
r47771 | pother = repo.nullid | ||
Andrew Halberstadt
|
r27267 | |||
Martin von Zweigbergk
|
r44692 | if wctx.isinmemory(): | ||
wctx.setparents(pctx.node(), pother) | ||||
Augie Fackler
|
r32351 | # fix up dirstate for copies and renames | ||
Martin von Zweigbergk
|
r44551 | copies.graftcopies(wctx, ctx, base) | ||
Martin von Zweigbergk
|
r44692 | else: | ||
r50855 | with repo.dirstate.changing_parents(repo): | |||
Martin von Zweigbergk
|
r44692 | repo.setparents(pctx.node(), pother) | ||
repo.dirstate.write(repo.currenttransaction()) | ||||
# fix up dirstate for copies and renames | ||||
copies.graftcopies(wctx, ctx, base) | ||||
Matt Mackall
|
r22902 | return stats | ||
Gregory Szorc
|
r39499 | |||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r46117 | def back_out(ctx, parent=None, wc=None): | ||
if parent is None: | ||||
if ctx.p2() is not None: | ||||
r49556 | msg = b"must specify parent of merge commit to back out" | |||
raise error.ProgrammingError(msg) | ||||
Martin von Zweigbergk
|
r46117 | parent = ctx.p1() | ||
Martin von Zweigbergk
|
r46134 | return _update( | ||
Martin von Zweigbergk
|
r46117 | ctx.repo(), | ||
parent, | ||||
branchmerge=True, | ||||
force=True, | ||||
ancestor=ctx.node(), | ||||
mergeancestor=False, | ||||
) | ||||
Augie Fackler
|
r43346 | def purge( | ||
repo, | ||||
matcher, | ||||
Valentin Gatien-Baron
|
r44771 | unknown=True, | ||
Augie Fackler
|
r43346 | ignored=False, | ||
removeemptydirs=True, | ||||
removefiles=True, | ||||
abortonerror=False, | ||||
noop=False, | ||||
r47078 | confirm=False, | |||
Augie Fackler
|
r43346 | ): | ||
Gregory Szorc
|
r39499 | """Purge the working directory of untracked files. | ||
``matcher`` is a matcher configured to scan the working directory - | ||||
potentially a subset. | ||||
Valentin Gatien-Baron
|
r44771 | ``unknown`` controls whether unknown files should be purged. | ||
``ignored`` controls whether ignored files should be purged. | ||||
Gregory Szorc
|
r39499 | |||
``removeemptydirs`` controls whether empty directories should be removed. | ||||
``removefiles`` controls whether files are removed. | ||||
``abortonerror`` causes an exception to be raised if an error occurs | ||||
deleting a file or directory. | ||||
``noop`` controls whether to actually remove files. If not defined, actions | ||||
will be taken. | ||||
r47078 | ``confirm`` ask confirmation before actually removing anything. | |||
Gregory Szorc
|
r39499 | Returns an iterable of relative paths in the working directory that were | ||
or would be removed. | ||||
""" | ||||
def remove(removefn, path): | ||||
try: | ||||
Gregory Szorc
|
r39500 | removefn(path) | ||
Gregory Szorc
|
r39499 | except OSError: | ||
Augie Fackler
|
r43347 | m = _(b'%s cannot be removed') % path | ||
Gregory Szorc
|
r39499 | if abortonerror: | ||
raise error.Abort(m) | ||||
else: | ||||
Augie Fackler
|
r43347 | repo.ui.warn(_(b'warning: %s\n') % m) | ||
Gregory Szorc
|
r39499 | |||
# There's no API to copy a matcher. So mutate the passed matcher and | ||||
# restore it when we're done. | ||||
oldtraversedir = matcher.traversedir | ||||
res = [] | ||||
try: | ||||
if removeemptydirs: | ||||
directories = [] | ||||
Martin von Zweigbergk
|
r44112 | matcher.traversedir = directories.append | ||
Gregory Szorc
|
r39499 | |||
Valentin Gatien-Baron
|
r44771 | status = repo.status(match=matcher, ignored=ignored, unknown=unknown) | ||
Gregory Szorc
|
r39499 | |||
r47078 | if confirm: | |||
r50239 | msg = None | |||
r47078 | nb_ignored = len(status.ignored) | |||
Raphaël Gomès
|
r49252 | nb_unknown = len(status.unknown) | ||
if nb_unknown and nb_ignored: | ||||
msg = _(b"permanently delete %d unknown and %d ignored files?") | ||||
msg %= (nb_unknown, nb_ignored) | ||||
elif nb_unknown: | ||||
msg = _(b"permanently delete %d unknown files?") | ||||
msg %= nb_unknown | ||||
r47078 | elif nb_ignored: | |||
msg = _(b"permanently delete %d ignored files?") | ||||
msg %= nb_ignored | ||||
r47079 | elif removeemptydirs: | |||
dir_count = 0 | ||||
for f in directories: | ||||
if matcher(f) and not repo.wvfs.listdir(f): | ||||
dir_count += 1 | ||||
if dir_count: | ||||
msg = _( | ||||
b"permanently delete at least %d empty directories?" | ||||
) | ||||
msg %= dir_count | ||||
r50239 | if msg is None: | |||
return res | ||||
else: | ||||
msg += b" (yN)$$ &Yes $$ &No" | ||||
if repo.ui.promptchoice(msg, default=1) == 1: | ||||
raise error.CanceledError(_(b'removal cancelled')) | ||||
r47078 | ||||
Gregory Szorc
|
r39499 | if removefiles: | ||
for f in sorted(status.unknown + status.ignored): | ||||
if not noop: | ||||
Augie Fackler
|
r43347 | repo.ui.note(_(b'removing file %s\n') % f) | ||
Gregory Szorc
|
r39500 | remove(repo.wvfs.unlink, f) | ||
Gregory Szorc
|
r39499 | res.append(f) | ||
if removeemptydirs: | ||||
for f in sorted(directories, reverse=True): | ||||
Gregory Szorc
|
r39500 | if matcher(f) and not repo.wvfs.listdir(f): | ||
Gregory Szorc
|
r39499 | if not noop: | ||
Augie Fackler
|
r43347 | repo.ui.note(_(b'removing directory %s\n') % f) | ||
Gregory Szorc
|
r39500 | remove(repo.wvfs.rmdir, f) | ||
Gregory Szorc
|
r39499 | res.append(f) | ||
return res | ||||
finally: | ||||
matcher.traversedir = oldtraversedir | ||||