##// END OF EJS Templates
tests: perform grep manually in test-doctest.py...
tests: perform grep manually in test-doctest.py This test has been failing on Windows since 0af56d3ee24c introduced the `hg files` invocation. Specifically, Windows seems to be choking on special characters in the fileset pattern. I believe at least \n and > were causing issues. I attempted various incantations to make the Windows command line parser accept the fileset but couldn't get anything working. I declared bankruptcy and just reimplemented the grepping code in Python. After this change, the test now passes on Windows again. Differential Revision: https://phab.mercurial-scm.org/D8343

File last commit:

r44305:d5ce99a6 default
r45144:15aef805 default
Show More
filemerge.py
1264 lines | 40.5 KiB | text/x-python | PythonLexer
Matt Mackall
filemerge: pull file-merging code into its own module
r6003 # filemerge.py - file-level merge handling for Mercurial
#
# Copyright 2006, 2007, 2008 Matt Mackall <mpm@selenic.com>
#
Martin Geisler
updated license to be explicit about GPL version 2
r8225 # This software may be used and distributed according to the terms of the
Matt Mackall
Update license to GPLv2+
r10263 # GNU General Public License version 2 or any later version.
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Gregory Szorc
filemerge: use absolute_import
r25949 from __future__ import absolute_import
Kyle Lippincott
filemerge: move temp file unlinks to _maketempfiles...
r37016 import contextlib
Gregory Szorc
filemerge: use absolute_import
r25949 import os
import re
Kyle Lippincott
filemerge: use a single temp dir instead of temp files...
r37017 import shutil
Gregory Szorc
filemerge: use absolute_import
r25949
from .i18n import _
Kyle Lippincott
merge-tools: when calling external merge tool, describe the resolve inputs...
r40512 from .node import (
hex,
nullid,
short,
)
Gregory Szorc
py3: manually import getattr where it is needed...
r43359 from .pycompat import (
getattr,
open,
)
Gregory Szorc
filemerge: use absolute_import
r25949
from . import (
Pulkit Goyal
py3: replace os.environ with encoding.environ (part 3 of 5)
r30636 encoding,
Gregory Szorc
filemerge: use absolute_import
r25949 error,
Yuya Nishihara
templater: factor out function that creates templater from string template...
r28955 formatter,
Gregory Szorc
filemerge: use absolute_import
r25949 match,
Pulkit Goyal
py3: make format strings unicodes and not bytes...
r30073 pycompat,
FUJIWARA Katsunori
filemerge: move decorator definition for internal merge tools to registrar...
r33663 registrar,
Siddharth Agarwal
origpath: move from cmdutil to scmutil...
r27651 scmutil,
Gregory Szorc
filemerge: use absolute_import
r25949 simplemerge,
tagmerge,
templatekw,
templater,
Kyle Lippincott
merge-tools: when calling external merge tool, describe the resolve inputs...
r40512 templateutil,
Gregory Szorc
filemerge: use absolute_import
r25949 util,
)
Matt Mackall
merge: allow smarter tool configuration...
r6004
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 from .utils import (
Yuya Nishihara
procutil: bulk-replace function calls to point to new module
r37138 procutil,
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 stringutil,
)
Augie Fackler
formatting: blacken the codebase...
r43346
Boris Feld
configitems: register the full 'merge-tools' config and sub-options...
r34827 def _toolstr(ui, tool, part, *args):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return ui.config(b"merge-tools", tool + b"." + part, *args)
Matt Mackall
merge: allow smarter tool configuration...
r6004
Augie Fackler
formatting: blacken the codebase...
r43346
def _toolbool(ui, tool, part, *args):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return ui.configbool(b"merge-tools", tool + b"." + part, *args)
Matt Mackall
merge: allow smarter tool configuration...
r6004
Augie Fackler
formatting: blacken the codebase...
r43346
Boris Feld
configitems: register the full 'merge-tools' config and sub-options...
r34827 def _toollist(ui, tool, part):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return ui.configlist(b"merge-tools", tool + b"." + part)
David Champion
merge: introduce tool.check parameter...
r11148
Augie Fackler
formatting: blacken the codebase...
r43346
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 internals = {}
Gregory Szorc
help.merge-tools: do not double document merge tools...
r24099 # Merge tools to document.
internalsdoc = {}
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
FUJIWARA Katsunori
filemerge: move decorator definition for internal merge tools to registrar...
r33663 internaltool = registrar.internalmerge()
Siddharth Agarwal
filemerge: add some merge types...
r26525 # internal tool merge types
FUJIWARA Katsunori
filemerge: move decorator definition for internal merge tools to registrar...
r33663 nomerge = internaltool.nomerge
Augie Fackler
formatting: blacken the codebase...
r43346 mergeonly = internaltool.mergeonly # just the full merge, no premerge
fullmerge = internaltool.fullmerge # both premerge and merge
Siddharth Agarwal
filemerge: add some merge types...
r26525
Kyle Lippincott
filemerge: make last line of prompts <40 english chars (issue6158)...
r42765 # IMPORTANT: keep the last line of this prompt very short ("What do you want to
# do?") because of issue6158, ideally to <40 English characters (to allow other
# languages that may take more columns to still have a chance to fit in an
# 80-column screen).
Stanislau Hlebik
filemerge: store error messages in module variables...
r32318 _localchangedotherdeletedmsg = _(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"file '%(fd)s' was deleted in other%(o)s but was modified in local%(l)s.\n"
b"You can use (c)hanged version, (d)elete, or leave (u)nresolved.\n"
b"What do you want to do?"
b"$$ &Changed $$ &Delete $$ &Unresolved"
Augie Fackler
formatting: blacken the codebase...
r43346 )
Stanislau Hlebik
filemerge: store error messages in module variables...
r32318
_otherchangedlocaldeletedmsg = _(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"file '%(fd)s' was deleted in local%(l)s but was modified in other%(o)s.\n"
b"You can use (c)hanged version, leave (d)eleted, or leave (u)nresolved.\n"
b"What do you want to do?"
b"$$ &Changed $$ &Deleted $$ &Unresolved"
Augie Fackler
formatting: blacken the codebase...
r43346 )
Stanislau Hlebik
filemerge: store error messages in module variables...
r32318
Siddharth Agarwal
filemerge: introduce class whose objects represent files not in a context...
r26979 class absentfilectx(object):
"""Represents a file that's ostensibly in a context but is actually not
present in it.
This is here because it's very specific to the filemerge code for now --
other code is likely going to break with the values this returns."""
Augie Fackler
formatting: blacken the codebase...
r43346
Siddharth Agarwal
filemerge: introduce class whose objects represent files not in a context...
r26979 def __init__(self, ctx, f):
self._ctx = ctx
self._f = f
def path(self):
return self._f
def size(self):
return None
def data(self):
return None
def filenode(self):
return nullid
_customcmp = True
Augie Fackler
formatting: blacken the codebase...
r43346
Siddharth Agarwal
filemerge: introduce class whose objects represent files not in a context...
r26979 def cmp(self, fctx):
"""compare with other file context
returns True if different from fctx.
"""
Augie Fackler
formatting: blacken the codebase...
r43346 return not (
fctx.isabsent()
Matt Harbison
filemerge: fix a missing attribute usage...
r44305 and fctx.changectx() == self.changectx()
Augie Fackler
formatting: blacken the codebase...
r43346 and fctx.path() == self.path()
)
Siddharth Agarwal
filemerge: introduce class whose objects represent files not in a context...
r26979
def flags(self):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b''
Siddharth Agarwal
filemerge: introduce class whose objects represent files not in a context...
r26979
def changectx(self):
return self._ctx
def isbinary(self):
return False
def isabsent(self):
return True
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Mackall
merge: allow smarter tool configuration...
r6004 def _findtool(ui, tool):
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 if tool in internals:
Dov Feldstern
use internal merge tool when specified for a merge-pattern in hgrc...
r6522 return tool
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 cmd = _toolstr(ui, tool, b"executable", tool)
if cmd.startswith(b'python:'):
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 return cmd
Matt Harbison
filemerge: split the logic for finding an external tool to its own function...
r23148 return findexternaltool(ui, tool)
Augie Fackler
formatting: blacken the codebase...
r43346
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 def _quotetoolpath(cmd):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if cmd.startswith(b'python:'):
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 return cmd
return procutil.shellquote(cmd)
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Harbison
filemerge: split the logic for finding an external tool to its own function...
r23148 def findexternaltool(ui, tool):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 for kn in (b"regkey", b"regkeyalt"):
Steve Borho
filemerge: introduce a 'regkeyalt' merge tool variable...
r13565 k = _toolstr(ui, tool, kn)
if not k:
continue
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 p = util.lookupreg(k, _toolstr(ui, tool, b"regname"))
Matt Mackall
merge: add registry look up bits to tool search
r6006 if p:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 p = procutil.findexe(p + _toolstr(ui, tool, b"regappend", b""))
Matt Mackall
merge: add registry look up bits to tool search
r6006 if p:
return p
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 exe = _toolstr(ui, tool, b"executable", tool)
Yuya Nishihara
procutil: bulk-replace function calls to point to new module
r37138 return procutil.findexe(util.expandpath(exe))
Matt Mackall
merge: allow smarter tool configuration...
r6004
Augie Fackler
formatting: blacken the codebase...
r43346
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 def _picktool(repo, ui, path, binary, symlink, changedelete):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 strictcheck = ui.configbool(b'merge', b'strict-capability-check')
FUJIWARA Katsunori
filemerge: add config knob to check capabilities of internal merge tools...
r39161
FUJIWARA Katsunori
filemerge: add the function to examine a capability of a internal tool...
r39159 def hascapability(tool, capability, strict=False):
FUJIWARA Katsunori
filemerge: make capability check for internal tools ignore merge-tools section...
r39302 if tool in internals:
return strict and internals[tool].capabilities.get(capability)
FUJIWARA Katsunori
filemerge: add the function to examine a capability of a internal tool...
r39159 return _toolbool(ui, tool, capability)
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 def supportscd(tool):
return tool in internals and internals[tool].mergetype == nomerge
def check(tool, pat, symlink, binary, changedelete):
Matt Mackall
merge: allow smarter tool configuration...
r6004 tmsg = tool
if pat:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 tmsg = _(b"%s (for pattern %s)") % (tool, pat)
Mads Kiilerich
More verbose logging when filemerge searches for merge-tool...
r7397 if not _findtool(ui, tool):
Augie Fackler
formatting: blacken the codebase...
r43346 if pat: # explicitly requested tool deserves a warning
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ui.warn(_(b"couldn't find merge tool %s\n") % tmsg)
Augie Fackler
formatting: blacken the codebase...
r43346 else: # configured but non-existing tools are more silent
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ui.note(_(b"couldn't find merge tool %s\n") % tmsg)
elif symlink and not hascapability(tool, b"symlink", strictcheck):
ui.warn(_(b"tool %s can't handle symlinks\n") % tmsg)
elif binary and not hascapability(tool, b"binary", strictcheck):
ui.warn(_(b"tool %s can't handle binary\n") % tmsg)
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 elif changedelete and not supportscd(tool):
# the nomerge tools are the only tools that support change/delete
# conflicts
pass
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif not procutil.gui() and _toolbool(ui, tool, b"gui"):
ui.warn(_(b"tool %s requires a GUI\n") % tmsg)
Matt Mackall
merge: allow smarter tool configuration...
r6004 else:
return True
return False
Matt Mackall
filemerge: mark internal-only config option
r25835 # internal config: ui.forcemerge
Steve Borho
merge: implement --tool arguments using new ui.forcemerge configurable...
r12788 # forcemerge comes from command line arguments, highest priority
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 force = ui.config(b'ui', b'forcemerge')
Steve Borho
merge: implement --tool arguments using new ui.forcemerge configurable...
r12788 if force:
toolpath = _findtool(ui, force)
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 if changedelete and not supportscd(toolpath):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b":prompt", None
Steve Borho
merge: implement --tool arguments using new ui.forcemerge configurable...
r12788 else:
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 if toolpath:
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 return (force, _quotetoolpath(toolpath))
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 else:
# mimic HGMERGE if given tool not found
return (force, force)
Steve Borho
merge: implement --tool arguments using new ui.forcemerge configurable...
r12788
# HGMERGE takes next precedence
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 hgmerge = encoding.environ.get(b"HGMERGE")
Steve Borho
filemerge: wrap quotes around tool path
r6025 if hgmerge:
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 if changedelete and not supportscd(hgmerge):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b":prompt", None
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 else:
return (hgmerge, hgmerge)
Matt Mackall
merge: allow smarter tool configuration...
r6004
# then patterns
FUJIWARA Katsunori
filemerge: add config knob to check capabilities of internal merge tools...
r39161
# whether binary capability should be checked strictly
binarycap = binary and strictcheck
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 for pat, tool in ui.configitems(b"merge-patterns"):
mf = match.match(repo.root, b'', [pat])
FUJIWARA Katsunori
filemerge: add config knob to check capabilities of internal merge tools...
r39161 if mf(path) and check(tool, pat, symlink, binarycap, changedelete):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if binary and not hascapability(tool, b"binary", strict=True):
Augie Fackler
formatting: blacken the codebase...
r43346 ui.warn(
_(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"warning: check merge-patterns configurations,"
b" if %r for binary file %r is unintentional\n"
b"(see 'hg help merge-tools'"
b" for binary files capability)\n"
Augie Fackler
formatting: blacken the codebase...
r43346 )
% (pycompat.bytestr(tool), pycompat.bytestr(path))
)
Benoit Boissinot
fix spaces/identation issues
r10339 toolpath = _findtool(ui, tool)
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 return (tool, _quotetoolpath(toolpath))
Matt Mackall
merge: allow smarter tool configuration...
r6004
# then merge tools
tools = {}
Augie Fackler
merge-tools: allow marking a mergetool as completely disabled...
r26730 disabled = set()
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 for k, v in ui.configitems(b"merge-tools"):
t = k.split(b'.')[0]
Matt Mackall
merge: allow smarter tool configuration...
r6004 if t not in tools:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 tools[t] = int(_toolstr(ui, t, b"priority"))
if _toolbool(ui, t, b"disabled"):
Augie Fackler
merge-tools: allow marking a mergetool as completely disabled...
r26730 disabled.add(t)
Steve Borho
filemerge: more backwards compatible behavior for ui.merge...
r6076 names = tools.keys()
Augie Fackler
formatting: blacken the codebase...
r43346 tools = sorted(
[(-p, tool) for tool, p in tools.items() if tool not in disabled]
)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 uimerge = ui.config(b"ui", b"merge")
Steve Borho
filemerge: more backwards compatible behavior for ui.merge...
r6076 if uimerge:
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 # external tools defined in uimerge won't be able to handle
# change/delete conflicts
Martin von Zweigbergk
mergetool: warn if ui.merge points to nonexistent tool...
r38987 if check(uimerge, path, symlink, binary, changedelete):
if uimerge not in names and not changedelete:
return (uimerge, uimerge)
Augie Fackler
formatting: blacken the codebase...
r43346 tools.insert(0, (None, uimerge)) # highest priority
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 tools.append((None, b"hgmerge")) # the old default, if found
Matt Mackall
many, many trivial check-code fixups
r10282 for p, t in tools:
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 if check(t, None, symlink, binary, changedelete):
Mads Kiilerich
More verbose logging when filemerge searches for merge-tool...
r7397 toolpath = _findtool(ui, t)
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 return (t, _quotetoolpath(toolpath))
Matt Mackall
filemerge: restore default prompt for binary/symlink lost in 83925d3a4559...
r16254
# internal merge or prompt as last resort
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 if symlink or binary or changedelete:
FUJIWARA Katsunori
filemerge: show warning about choice of :prompt only at an actual fallback...
r32253 if not changedelete and len(tools):
# any tool is rejected by capability for symlink or binary
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ui.warn(_(b"no tool found to merge %s\n") % path)
return b":prompt", None
return b":merge", None
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Mackall
merge: add support for tool EOL fixups...
r6005 def _eoltype(data):
Matt Harbison
cleanup: fix docstring formatting...
r44226 """Guess the EOL type of a file"""
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if b'\0' in data: # binary
Matt Mackall
merge: add support for tool EOL fixups...
r6005 return None
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if b'\r\n' in data: # Windows
return b'\r\n'
if b'\r' in data: # Old Mac
return b'\r'
if b'\n' in data: # UNIX
return b'\n'
Augie Fackler
formatting: blacken the codebase...
r43346 return None # unknown
Matt Mackall
merge: add support for tool EOL fixups...
r6005
Phil Cohen
filemerge: use arbitraryfilectx for backups...
r34783 def _matcheol(file, back):
Matt Harbison
cleanup: fix docstring formatting...
r44226 """Convert EOL markers in a file to match origfile"""
Augie Fackler
formatting: blacken the codebase...
r43346 tostyle = _eoltype(back.data()) # No repo.wread filters?
Matt Mackall
merge: add support for tool EOL fixups...
r6005 if tostyle:
Dan Villiom Podlaski Christiansen
prevent transient leaks of file handle by using new helper functions...
r14168 data = util.readfile(file)
Matt Mackall
merge: add support for tool EOL fixups...
r6005 style = _eoltype(data)
if style:
newdata = data.replace(style, tostyle)
if newdata != data:
Dan Villiom Podlaski Christiansen
prevent transient leaks of file handle by using new helper functions...
r14168 util.writefile(file, newdata)
Matt Mackall
merge: add support for tool EOL fixups...
r6005
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 @internaltool(b'prompt', nomerge)
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 def _iprompt(repo, mynode, orig, fcd, fco, fca, toolconf, labels=None):
timeless
filemerge: use revset notation for p1/p2 of local/other descriptions
r28640 """Asks the user which of the local `p1()` or the other `p2()` version to
keep as the merged version."""
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 ui = repo.ui
fd = fcd.path()
Martin von Zweigbergk
merge: respect ui.relative-paths...
r41651 uipathfn = scmutil.getuipathfn(repo)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Phil Cohen
filemerge: raise InMemoryMergeConflictsError if we hit merge conflicts in IMM...
r35283 # Avoid prompting during an in-memory merge since it doesn't support merge
# conflicts.
if fcd.changectx().isinmemory():
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.InMemoryMergeConflictsError(
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 b'in-memory merge does not support file conflicts'
Augie Fackler
formatting: blacken the codebase...
r43346 )
Phil Cohen
filemerge: raise InMemoryMergeConflictsError if we hit merge conflicts in IMM...
r35283
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 prompts = partextras(labels)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 prompts[b'fd'] = uipathfn(fd)
Siddharth Agarwal
filemerge: treat EOF at prompt as fail, not abort...
r26898 try:
Siddharth Agarwal
filemerge: add support for change/delete conflicts to the ':prompt' tool...
r27038 if fco.isabsent():
Augie Fackler
formatting: blacken the codebase...
r43346 index = ui.promptchoice(_localchangedotherdeletedmsg % prompts, 2)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 choice = [b'local', b'other', b'unresolved'][index]
Siddharth Agarwal
filemerge: add support for change/delete conflicts to the ':prompt' tool...
r27038 elif fcd.isabsent():
Augie Fackler
formatting: blacken the codebase...
r43346 index = ui.promptchoice(_otherchangedlocaldeletedmsg % prompts, 2)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 choice = [b'other', b'local', b'unresolved'][index]
Siddharth Agarwal
filemerge: add support for change/delete conflicts to the ':prompt' tool...
r27038 else:
Kyle Lippincott
filemerge: make last line of prompts <40 english chars (issue6158)...
r42765 # IMPORTANT: keep the last line of this prompt ("What do you want to
# do?") very short, see comment next to _localchangedotherdeletedmsg
# at the top of the file for details.
Siddharth Agarwal
filemerge: add a 'leave unresolved' option to regular prompts...
r27162 index = ui.promptchoice(
Augie Fackler
formatting: blacken the codebase...
r43346 _(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"file '%(fd)s' needs to be resolved.\n"
b"You can keep (l)ocal%(l)s, take (o)ther%(o)s, or leave "
b"(u)nresolved.\n"
b"What do you want to do?"
b"$$ &Local $$ &Other $$ &Unresolved"
Augie Fackler
formatting: blacken the codebase...
r43346 )
% prompts,
2,
)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 choice = [b'local', b'other', b'unresolved'][index]
Siddharth Agarwal
filemerge.prompt: separate out choice selection and action...
r26851
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if choice == b'other':
Augie Fackler
formatting: blacken the codebase...
r43346 return _iother(repo, mynode, orig, fcd, fco, fca, toolconf, labels)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif choice == b'local':
Augie Fackler
formatting: blacken the codebase...
r43346 return _ilocal(repo, mynode, orig, fcd, fco, fca, toolconf, labels)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif choice == b'unresolved':
Augie Fackler
formatting: blacken the codebase...
r43346 return _ifail(repo, mynode, orig, fcd, fco, fca, toolconf, labels)
Siddharth Agarwal
filemerge: treat EOF at prompt as fail, not abort...
r26898 except error.ResponseExpected:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ui.write(b"\n")
Augie Fackler
formatting: blacken the codebase...
r43346 return _ifail(repo, mynode, orig, fcd, fco, fca, toolconf, labels)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 @internaltool(b'local', nomerge)
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 def _ilocal(repo, mynode, orig, fcd, fco, fca, toolconf, labels=None):
timeless
filemerge: use revset notation for p1/p2 of local/other descriptions
r28640 """Uses the local `p1()` version of files as the merged version."""
Siddharth Agarwal
filemerge: add support for change/delete conflicts to the ':local' merge tool...
r27036 return 0, fcd.isabsent()
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 @internaltool(b'other', nomerge)
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 def _iother(repo, mynode, orig, fcd, fco, fca, toolconf, labels=None):
timeless
filemerge: use revset notation for p1/p2 of local/other descriptions
r28640 """Uses the other `p2()` version of files as the merged version."""
Siddharth Agarwal
filemerge: add support for change/delete conflicts to the ':other' merge tool...
r27037 if fco.isabsent():
# local changed, remote deleted -- 'deleted' picked
Phil Cohen
filemerge: convert a couple of wvfs calls in internal mergetools to contexts...
r33152 _underlyingfctxifabsent(fcd).remove()
Siddharth Agarwal
filemerge: add support for change/delete conflicts to the ':other' merge tool...
r27037 deleted = True
else:
Phil Cohen
filemerge: convert a couple of wvfs calls in internal mergetools to contexts...
r33152 _underlyingfctxifabsent(fcd).write(fco.data(), fco.flags())
Siddharth Agarwal
filemerge: add support for change/delete conflicts to the ':other' merge tool...
r27037 deleted = False
return 0, deleted
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 @internaltool(b'fail', nomerge)
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 def _ifail(repo, mynode, orig, fcd, fco, fca, toolconf, labels=None):
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 """
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 Rather than attempting to merge files that were modified on both
branches, it marks them as unresolved. The resolve command must be
used to resolve these conflicts."""
Siddharth Agarwal
filemerge: in ':fail' tool, write out other side if local side is deleted...
r27123 # for change/delete conflicts write out the changed version, then fail
if fcd.isabsent():
Phil Cohen
filemerge: convert a couple of wvfs calls in internal mergetools to contexts...
r33152 _underlyingfctxifabsent(fcd).write(fco.data(), fco.flags())
Siddharth Agarwal
filemerge: return whether the file is deleted for nomerge internal tools...
r27032 return 1, False
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Augie Fackler
formatting: blacken the codebase...
r43346
Phil Cohen
filemerge: convert a couple of wvfs calls in internal mergetools to contexts...
r33152 def _underlyingfctxifabsent(filectx):
"""Sometimes when resolving, our fcd is actually an absentfilectx, but
we want to write to it (to do the resolve). This helper returns the
underyling workingfilectx in that case.
"""
if filectx.isabsent():
return filectx.changectx()[filectx.path()]
else:
return filectx
Augie Fackler
formatting: blacken the codebase...
r43346
Siddharth Agarwal
filemerge: don't attempt to premerge change/delete conflicts...
r27041 def _premerge(repo, fcd, fco, fca, toolconf, files, labels=None):
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 tool, toolpath, binary, symlink, scriptfn = toolconf
Siddharth Agarwal
filemerge: don't attempt to premerge change/delete conflicts...
r27041 if symlink or fcd.isabsent() or fco.isabsent():
Mads Kiilerich
merge: never do premerge on symlinks...
r18257 return 1
Phil Cohen
filemerge: eliminate most uses of tempfiles...
r34034 unused, unused, unused, back = files
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
ui = repo.ui
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 validkeep = [b'keep', b'keep-merge3']
Pierre-Yves David
merge-tools: make premerge valid values extensible...
r22031
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 # do we attempt to simplemerge first?
try:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 premerge = _toolbool(ui, tool, b"premerge", not binary)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 except error.ConfigError:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 premerge = _toolstr(ui, tool, b"premerge", b"").lower()
Pierre-Yves David
merge-tools: make premerge valid values extensible...
r22031 if premerge not in validkeep:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _valid = b', '.join([b"'" + v + b"'" for v in validkeep])
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.ConfigError(
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 _(b"%s.premerge not valid ('%s' is neither boolean nor %s)")
Augie Fackler
formatting: blacken the codebase...
r43346 % (tool, premerge, _valid)
)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
if premerge:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if premerge == b'keep-merge3':
Pierre-Yves David
merge-tools: add a `premerge=keep-merge3` config option...
r22032 if not labels:
labels = _defaultconflictlabels
if len(labels) < 3:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 labels.append(b'base')
Phil Cohen
simplemerge: remove unused `repo` parameter...
r34051 r = simplemerge.simplemerge(ui, fcd, fca, fco, quiet=True, label=labels)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 if not r:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ui.debug(b" premerge successful\n")
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 return 0
Pierre-Yves David
merge-tools: make premerge valid values extensible...
r22031 if premerge not in validkeep:
Phil Cohen
filemerge: eliminate most uses of tempfiles...
r34034 # restore from backup and try again
Phil Cohen
filemerge: add _restorebackup...
r34038 _restorebackup(fcd, back)
Augie Fackler
formatting: blacken the codebase...
r43346 return 1 # continue merging
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Siddharth Agarwal
filemerge: rename _symlinkcheck to _mergecheck...
r26948 def _mergecheck(repo, mynode, orig, fcd, fco, fca, toolconf):
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 tool, toolpath, binary, symlink, scriptfn = toolconf
Martin von Zweigbergk
merge: respect ui.relative-paths...
r41651 uipathfn = scmutil.getuipathfn(repo)
Siddharth Agarwal
filemerge: add a precheck for symlinks...
r26515 if symlink:
Augie Fackler
formatting: blacken the codebase...
r43346 repo.ui.warn(
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 _(b'warning: internal %s cannot merge symlinks for %s\n')
Augie Fackler
formatting: blacken the codebase...
r43346 % (tool, uipathfn(fcd.path()))
)
Siddharth Agarwal
filemerge: add a precheck for symlinks...
r26515 return False
Siddharth Agarwal
filemerge._mergecheck: add check for change/delete conflicts...
r27040 if fcd.isabsent() or fco.isabsent():
Augie Fackler
formatting: blacken the codebase...
r43346 repo.ui.warn(
_(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'warning: internal %s cannot merge change/delete '
b'conflict for %s\n'
Augie Fackler
formatting: blacken the codebase...
r43346 )
% (tool, uipathfn(fcd.path()))
)
Siddharth Agarwal
filemerge._mergecheck: add check for change/delete conflicts...
r27040 return False
Siddharth Agarwal
filemerge: add a precheck for symlinks...
r26515 return True
Augie Fackler
formatting: blacken the codebase...
r43346
Erik Huelsmann
filemerge: split internal merge into api entry point and internal helper...
r26070 def _merge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels, mode):
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 """
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 Uses the internal non-interactive simple merge algorithm for merging
files. It will fail if there are any conflicts and leave markers in
Pierre-Yves David
internal:merge: update documentation...
r22027 the partially merged file. Markers will have two sections, one for each side
Erik Huelsmann
filemerge: split internal merge into api entry point and internal helper...
r26070 of merge, unless mode equals 'union' which suppresses the markers."""
Siddharth Agarwal
filemerge._merge: drop no longer necessary 'if r:' check...
r26572 ui = repo.ui
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Phil Cohen
simplemerge: remove unused `repo` parameter...
r34051 r = simplemerge.simplemerge(ui, fcd, fca, fco, label=labels, mode=mode)
Siddharth Agarwal
filemerge: return whether the file is deleted from all other merge tools...
r27033 return True, r, False
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Augie Fackler
formatting: blacken the codebase...
r43346
@internaltool(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'union',
Augie Fackler
formatting: blacken the codebase...
r43346 fullmerge,
_(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"warning: conflicts while merging %s! "
b"(edit, then use 'hg resolve --mark')\n"
Augie Fackler
formatting: blacken the codebase...
r43346 ),
precheck=_mergecheck,
)
Erik Huelsmann
filemerge: add 'union' merge to internal merge tool...
r26071 def _iunion(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
"""
Uses the internal non-interactive simple merge algorithm for merging
files. It will use both left and right sides for conflict regions.
No markers are inserted."""
Augie Fackler
formatting: blacken the codebase...
r43346 return _merge(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 repo, mynode, orig, fcd, fco, fca, toolconf, files, labels, b'union'
Augie Fackler
formatting: blacken the codebase...
r43346 )
Erik Huelsmann
filemerge: add 'union' merge to internal merge tool...
r26071
Augie Fackler
formatting: blacken the codebase...
r43346 @internaltool(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'merge',
Augie Fackler
formatting: blacken the codebase...
r43346 fullmerge,
_(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"warning: conflicts while merging %s! "
b"(edit, then use 'hg resolve --mark')\n"
Augie Fackler
formatting: blacken the codebase...
r43346 ),
precheck=_mergecheck,
)
Erik Huelsmann
filemerge: split internal merge into api entry point and internal helper...
r26070 def _imerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
"""
Uses the internal non-interactive simple merge algorithm for merging
files. It will fail if there are any conflicts and leave markers in
the partially merged file. Markers will have two sections, one for each side
of merge."""
Augie Fackler
formatting: blacken the codebase...
r43346 return _merge(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 repo, mynode, orig, fcd, fco, fca, toolconf, files, labels, b'merge'
Augie Fackler
formatting: blacken the codebase...
r43346 )
Erik Huelsmann
filemerge: split internal merge into api entry point and internal helper...
r26070
Augie Fackler
formatting: blacken the codebase...
r43346 @internaltool(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'merge3',
Augie Fackler
formatting: blacken the codebase...
r43346 fullmerge,
_(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"warning: conflicts while merging %s! "
b"(edit, then use 'hg resolve --mark')\n"
Augie Fackler
formatting: blacken the codebase...
r43346 ),
precheck=_mergecheck,
)
Pierre-Yves David
merge: add an internal:merge3 tool...
r22028 def _imerge3(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
"""
Uses the internal non-interactive simple merge algorithm for merging
files. It will fail if there are any conflicts and leave markers in
the partially merged file. Marker will have three sections, one from each
side of the merge and one for the base content."""
if not labels:
labels = _defaultconflictlabels
if len(labels) < 3:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 labels.append(b'base')
Pierre-Yves David
merge: add an internal:merge3 tool...
r22028 return _imerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels)
Augie Fackler
formatting: blacken the codebase...
r43346
def _imergeauto(
repo,
mynode,
orig,
fcd,
fco,
fca,
toolconf,
files,
labels=None,
localorother=None,
):
Jordi Gutiérrez Hermoso
filemerge: add non-interactive :merge-local and :merge-other...
r26224 """
Generic driver for _imergelocal and _imergeother
"""
assert localorother is not None
Augie Fackler
formatting: blacken the codebase...
r43346 r = simplemerge.simplemerge(
repo.ui, fcd, fca, fco, label=labels, localorother=localorother
)
Jordi Gutiérrez Hermoso
filemerge: add non-interactive :merge-local and :merge-other...
r26224 return True, r
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 @internaltool(b'merge-local', mergeonly, precheck=_mergecheck)
Jordi Gutiérrez Hermoso
filemerge: add non-interactive :merge-local and :merge-other...
r26224 def _imergelocal(*args, **kwargs):
"""
Like :merge, but resolve all conflicts non-interactively in favor
timeless
filemerge: use revset notation for p1/p2 of local/other descriptions
r28640 of the local `p1()` changes."""
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 success, status = _imergeauto(localorother=b'local', *args, **kwargs)
Siddharth Agarwal
filemerge: return whether the file is deleted from all other merge tools...
r27033 return success, status, False
Jordi Gutiérrez Hermoso
filemerge: add non-interactive :merge-local and :merge-other...
r26224
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 @internaltool(b'merge-other', mergeonly, precheck=_mergecheck)
Jordi Gutiérrez Hermoso
filemerge: add non-interactive :merge-local and :merge-other...
r26224 def _imergeother(*args, **kwargs):
"""
Like :merge, but resolve all conflicts non-interactively in favor
timeless
filemerge: use revset notation for p1/p2 of local/other descriptions
r28640 of the other `p2()` changes."""
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 success, status = _imergeauto(localorother=b'other', *args, **kwargs)
Siddharth Agarwal
filemerge: return whether the file is deleted from all other merge tools...
r27033 return success, status, False
Jordi Gutiérrez Hermoso
filemerge: add non-interactive :merge-local and :merge-other...
r26224
Augie Fackler
formatting: blacken the codebase...
r43346
@internaltool(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'tagmerge',
Augie Fackler
formatting: blacken the codebase...
r43346 mergeonly,
_(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"automatic tag merging of %s failed! "
b"(use 'hg resolve --tool :merge' or another merge "
b"tool of your choice)\n"
Augie Fackler
formatting: blacken the codebase...
r43346 ),
)
Angel Ezquerra
filemerge: add internal:tagmerge merge tool...
r21922 def _itagmerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
"""
Uses the internal tag merge algorithm (experimental).
"""
Siddharth Agarwal
filemerge: return whether the file is deleted from all other merge tools...
r27033 success, status = tagmerge.merge(repo, fcd, fco, fca)
return success, status, False
Angel Ezquerra
filemerge: add internal:tagmerge merge tool...
r21922
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 @internaltool(b'dump', fullmerge, binary=True, symlink=True)
Durham Goode
merge: define conflict marker labels in filemerge()...
r21273 def _idump(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 """
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 Creates three versions of the files to merge, containing the
contents of local, other and base. These files can then be used to
perform a merge manually. If the file to be merged is named
``a.txt``, these files will accordingly be named ``a.txt.local``,
``a.txt.other`` and ``a.txt.base`` and they will be placed in the
FUJIWARA Katsunori
filemerge: add internal merge tool to dump files forcibly...
r32255 same directory as ``a.txt``.
Joe Blaylock
help: fix typo in hg merge documentation
r34916 This implies premerge. Therefore, files aren't dumped, if premerge
FUJIWARA Katsunori
filemerge: add internal merge tool to dump files forcibly...
r32255 runs successfully. Use :forcedump to forcibly write files out.
"""
Phil Cohen
filemerge: add `_workingpath`...
r34036 a = _workingpath(repo, fcd)
Siddharth Agarwal
filemerge._idump: drop no longer necessary 'if r:' check...
r26573 fd = fcd.path()
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Phil Cohen
filemerge: add a missing flushall()...
r34786 from . import context
Augie Fackler
formatting: blacken the codebase...
r43346
Phil Cohen
filemerge: add a missing flushall()...
r34786 if isinstance(fcd, context.overlayworkingfilectx):
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.InMemoryMergeConflictsError(
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 b'in-memory merge does not support the :dump tool.'
Augie Fackler
formatting: blacken the codebase...
r43346 )
Phil Cohen
filemerge: add a missing flushall()...
r34786
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 util.writefile(a + b".local", fcd.decodeddata())
repo.wwrite(fd + b".other", fco.data(), fco.flags())
repo.wwrite(fd + b".base", fca.data(), fca.flags())
Siddharth Agarwal
filemerge: return whether the file is deleted from all other merge tools...
r27033 return False, 1, False
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 @internaltool(b'forcedump', mergeonly, binary=True, symlink=True)
Augie Fackler
formatting: blacken the codebase...
r43346 def _forcedump(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
FUJIWARA Katsunori
filemerge: add internal merge tool to dump files forcibly...
r32255 """
Creates three versions of the files as same as :dump, but omits premerge.
"""
Augie Fackler
formatting: blacken the codebase...
r43346 return _idump(
repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=labels
)
FUJIWARA Katsunori
filemerge: add internal merge tool to dump files forcibly...
r32255
Phil Cohen
filemerge: only raise InMemoryMergeConflictsError when running _xmerge...
r35479 def _xmergeimm(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
# In-memory merge simply raises an exception on all external merge tools,
# for now.
#
# It would be possible to run most tools with temporary files, but this
# raises the question of what to do if the user only partially resolves the
# file -- we can't leave a merge state. (Copy to somewhere in the .hg/
# directory and tell the user how to get it is my best idea, but it's
# clunky.)
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.InMemoryMergeConflictsError(
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 b'in-memory merge does not support external merge tools'
Augie Fackler
formatting: blacken the codebase...
r43346 )
Phil Cohen
filemerge: only raise InMemoryMergeConflictsError when running _xmerge...
r35479
Kyle Lippincott
merge-tools: when calling external merge tool, describe the resolve inputs...
r40512 def _describemerge(ui, repo, mynode, fcl, fcb, fco, env, toolpath, args):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 tmpl = ui.config(b'ui', b'pre-merge-tool-output-template')
Kyle Lippincott
merge-tools: when calling external merge tool, describe the resolve inputs...
r40512 if not tmpl:
return
mappingdict = templateutil.mappingdict
Augie Fackler
formatting: blacken the codebase...
r43346 props = {
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'ctx': fcl.changectx(),
b'node': hex(mynode),
b'path': fcl.path(),
b'local': mappingdict(
Augie Fackler
formatting: blacken the codebase...
r43346 {
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'ctx': fcl.changectx(),
b'fctx': fcl,
b'node': hex(mynode),
b'name': _(b'local'),
b'islink': b'l' in fcl.flags(),
b'label': env[b'HG_MY_LABEL'],
Augie Fackler
formatting: blacken the codebase...
r43346 }
),
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'base': mappingdict(
Augie Fackler
formatting: blacken the codebase...
r43346 {
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'ctx': fcb.changectx(),
b'fctx': fcb,
b'name': _(b'base'),
b'islink': b'l' in fcb.flags(),
b'label': env[b'HG_BASE_LABEL'],
Augie Fackler
formatting: blacken the codebase...
r43346 }
),
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'other': mappingdict(
Augie Fackler
formatting: blacken the codebase...
r43346 {
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'ctx': fco.changectx(),
b'fctx': fco,
b'name': _(b'other'),
b'islink': b'l' in fco.flags(),
b'label': env[b'HG_OTHER_LABEL'],
Augie Fackler
formatting: blacken the codebase...
r43346 }
),
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'toolpath': toolpath,
b'toolargs': args,
Augie Fackler
formatting: blacken the codebase...
r43346 }
Kyle Lippincott
merge-tools: when calling external merge tool, describe the resolve inputs...
r40512
# TODO: make all of this something that can be specified on a per-tool basis
tmpl = templater.unquotestring(tmpl)
# Not using cmdutil.rendertemplate here since it causes errors importing
# things for us to import cmdutil.
tres = formatter.templateresources(ui, repo)
Augie Fackler
formatting: blacken the codebase...
r43346 t = formatter.maketemplater(
ui, tmpl, defaults=templatekw.keywords, resources=tres
)
Kyle Lippincott
merge-tools: when calling external merge tool, describe the resolve inputs...
r40512 ui.status(t.renderdefault(props))
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Harbison
filemerge: drop a default argument to appease pytype...
r44304 def _xmerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels):
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 tool, toolpath, binary, symlink, scriptfn = toolconf
Martin von Zweigbergk
merge: respect ui.relative-paths...
r41651 uipathfn = scmutil.getuipathfn(repo)
Siddharth Agarwal
filemerge: don't try using external tools on change/delete conflicts...
r27042 if fcd.isabsent() or fco.isabsent():
Augie Fackler
formatting: blacken the codebase...
r43346 repo.ui.warn(
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 _(b'warning: %s cannot merge change/delete conflict for %s\n')
Augie Fackler
formatting: blacken the codebase...
r43346 % (tool, uipathfn(fcd.path()))
)
Siddharth Agarwal
filemerge: don't try using external tools on change/delete conflicts...
r27042 return False, 1, None
Phil Cohen
filemerge: reduce creation of tempfiles until needed...
r34037 unused, unused, unused, back = files
Kyle Lippincott
filemerge: give some variables in _xmerge more descriptive names...
r36994 localpath = _workingpath(repo, fcd)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 args = _toolstr(repo.ui, tool, b"args")
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095
Augie Fackler
formatting: blacken the codebase...
r43346 with _maketempfiles(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 repo, fco, fca, repo.wvfs.join(back.path()), b"$output" in args
Augie Fackler
formatting: blacken the codebase...
r43346 ) as temppaths:
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 basepath, otherpath, localoutputpath = temppaths
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 outpath = b""
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 mylabel, otherlabel = labels[:2]
if len(labels) >= 3:
baselabel = labels[2]
else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 baselabel = b'base'
Augie Fackler
formatting: blacken the codebase...
r43346 env = {
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'HG_FILE': fcd.path(),
b'HG_MY_NODE': short(mynode),
b'HG_OTHER_NODE': short(fco.changectx().node()),
b'HG_BASE_NODE': short(fca.changectx().node()),
b'HG_MY_ISLINK': b'l' in fcd.flags(),
b'HG_OTHER_ISLINK': b'l' in fco.flags(),
b'HG_BASE_ISLINK': b'l' in fca.flags(),
b'HG_MY_LABEL': mylabel,
b'HG_OTHER_LABEL': otherlabel,
b'HG_BASE_LABEL': baselabel,
Augie Fackler
formatting: blacken the codebase...
r43346 }
Phil Cohen
filemerge: reduce creation of tempfiles until needed...
r34037 ui = repo.ui
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if b"$output" in args:
Phil Cohen
filemerge: use arbitraryfilectx for backups...
r34783 # read input from backup, write to original
Kyle Lippincott
filemerge: give some variables in _xmerge more descriptive names...
r36994 outpath = localpath
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 localpath = localoutputpath
Augie Fackler
formatting: blacken the codebase...
r43346 replace = {
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'local': localpath,
b'base': basepath,
b'other': otherpath,
b'output': outpath,
b'labellocal': mylabel,
b'labelother': otherlabel,
b'labelbase': baselabel,
Augie Fackler
formatting: blacken the codebase...
r43346 }
Yuya Nishihara
procutil: bulk-replace function calls to point to new module
r37138 args = util.interpolate(
Augie Fackler
formatting: blacken the codebase...
r43346 br'\$',
replace,
args,
lambda s: procutil.shellquote(util.localpath(s)),
)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if _toolbool(ui, tool, b"gui"):
Augie Fackler
formatting: blacken the codebase...
r43346 repo.ui.status(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b'running merge tool %s for file %s\n')
Augie Fackler
formatting: blacken the codebase...
r43346 % (tool, uipathfn(fcd.path()))
)
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 if scriptfn is None:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 cmd = toolpath + b' ' + args
repo.ui.debug(b'launching merge tool: %s\n' % cmd)
Kyle Lippincott
merge-tools: when calling external merge tool, describe the resolve inputs...
r40512 _describemerge(ui, repo, mynode, fcd, fca, fco, env, toolpath, args)
Augie Fackler
formatting: blacken the codebase...
r43346 r = ui.system(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 cmd, cwd=repo.root, environ=env, blockedtag=b'mergetool'
Augie Fackler
formatting: blacken the codebase...
r43346 )
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 else:
Augie Fackler
formatting: blacken the codebase...
r43346 repo.ui.debug(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'launching python merge script: %s:%s\n' % (toolpath, scriptfn)
Augie Fackler
formatting: blacken the codebase...
r43346 )
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 r = 0
try:
# avoid cycle cmdutil->merge->filemerge->extensions->cmdutil
from . import extensions
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 mod = extensions.loadpath(toolpath, b'hgmerge.%s' % tool)
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 except Exception:
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.Abort(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b"loading python merge script failed: %s") % toolpath
Augie Fackler
formatting: blacken the codebase...
r43346 )
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 mergefn = getattr(mod, scriptfn, None)
if mergefn is None:
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.Abort(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b"%s does not have function: %s") % (toolpath, scriptfn)
Augie Fackler
formatting: blacken the codebase...
r43346 )
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 argslist = procutil.shellsplit(args)
# avoid cycle cmdutil->merge->filemerge->hook->extensions->cmdutil
from . import hook
Augie Fackler
formatting: blacken the codebase...
r43346
ret, raised = hook.pythonhook(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ui, repo, b"merge", toolpath, mergefn, {b'args': argslist}, True
Augie Fackler
formatting: blacken the codebase...
r43346 )
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 if raised:
r = 1
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 repo.ui.debug(b'merge tool returned: %d\n' % r)
Phil Cohen
filemerge: reduce creation of tempfiles until needed...
r34037 return True, r, False
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Augie Fackler
formatting: blacken the codebase...
r43346
Yuya Nishihara
templater: drop unneeded resources from conflict-marker data...
r35498 def _formatconflictmarker(ctx, template, label, pad):
Durham Goode
merge: add conflict marker formatter (BC)...
r21519 """Applies the given template to the ctx, prefixed by the label.
Pad is the minimum width of the label prefix, so that multiple markers
can have aligned templated parts.
"""
if ctx.node() is None:
ctx = ctx.p1()
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 props = {b'ctx': ctx}
Yuya Nishihara
templater: rename .render(mapping) to .renderdefault(mapping) (API)...
r37003 templateresult = template.renderdefault(props)
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 label = (b'%s:' % label).ljust(pad + 1)
mark = b'%s %s' % (label, templateresult)
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
FUJIWARA Katsunori
filemerge: use only the first line of the generated conflict marker for safety...
r21864 if mark:
Augie Fackler
formatting: blacken the codebase...
r43346 mark = mark.splitlines()[0] # split for safety
FUJIWARA Katsunori
filemerge: use only the first line of the generated conflict marker for safety...
r21864
FUJIWARA Katsunori
filemerge: use 'util.ellipsis' to trim custom conflict markers correctly...
r21865 # 8 for the prefix of conflict marker lines (e.g. '<<<<<<< ')
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 return stringutil.ellipsis(mark, 80 - 8)
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _defaultconflictlabels = [b'local', b'other']
Durham Goode
merge: add labels parameter from merge.update to filemerge...
r21524
Augie Fackler
formatting: blacken the codebase...
r43346
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 def _formatlabels(repo, fcd, fco, fca, labels, tool=None):
Durham Goode
merge: add conflict marker formatter (BC)...
r21519 """Formats the given labels using the conflict marker template.
Returns a list of formatted labels.
"""
cd = fcd.changectx()
co = fco.changectx()
Pierre-Yves David
filemerge: allow the formatting of three labels instead of two...
r22026 ca = fca.changectx()
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
ui = repo.ui
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 template = ui.config(b'ui', b'mergemarkertemplate')
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 if tool is not None:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 template = _toolstr(ui, tool, b'mergemarkertemplate', template)
Yuya Nishihara
filemerge: optionally strip quotes from merge marker template (BC)...
r32047 template = templater.unquotestring(template)
Yuya Nishihara
templater: move repo, ui and cache to per-engine resources
r35485 tres = formatter.templateresources(ui, repo)
Augie Fackler
formatting: blacken the codebase...
r43346 tmpl = formatter.maketemplater(
ui, template, defaults=templatekw.keywords, resources=tres
)
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
Pierre-Yves David
filemerge: allow the formatting of three labels instead of two...
r22026 pad = max(len(l) for l in labels)
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
Augie Fackler
formatting: blacken the codebase...
r43346 newlabels = [
_formatconflictmarker(cd, tmpl, labels[0], pad),
_formatconflictmarker(co, tmpl, labels[1], pad),
]
Pierre-Yves David
filemerge: allow the formatting of three labels instead of two...
r22026 if len(labels) > 2:
Yuya Nishihara
templater: drop unneeded resources from conflict-marker data...
r35498 newlabels.append(_formatconflictmarker(ca, tmpl, labels[2], pad))
Pierre-Yves David
filemerge: allow the formatting of three labels instead of two...
r22026 return newlabels
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
Augie Fackler
formatting: blacken the codebase...
r43346
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 def partextras(labels):
"""Return a dictionary of extra labels for use in prompts to the user
Intended use is in strings of the form "(l)ocal%(l)s".
"""
if labels is None:
return {
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"l": b"",
b"o": b"",
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 }
return {
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"l": b" [%s]" % labels[0],
b"o": b" [%s]" % labels[1],
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 }
Augie Fackler
formatting: blacken the codebase...
r43346
Phil Cohen
filemerge: add _restorebackup...
r34038 def _restorebackup(fcd, back):
# TODO: Add a workingfilectx.write(otherfilectx) path so we can use
# util.copy here instead.
Phil Cohen
filemerge: use arbitraryfilectx for backups...
r34783 fcd.write(back.data(), fcd.flags())
Phil Cohen
filemerge: add _restorebackup...
r34038
Augie Fackler
formatting: blacken the codebase...
r43346
Phil Cohen
filemerge: use arbitraryfilectx for backups...
r34783 def _makebackup(repo, ui, wctx, fcd, premerge):
"""Makes and returns a filectx-like object for ``fcd``'s backup file.
Phil Cohen
filemerge: extract _maketemp and _makebackup...
r34033
In addition to preserving the user's pre-existing modifications to `fcd`
(if any), the backup is used to undo certain premerges, confirm whether a
merge changed anything, and determine what line endings the new file should
have.
Phil Cohen
filemerge: fix backing up an in-memory file to a custom location...
r35720
Backups only need to be written once (right before the premerge) since their
content doesn't change afterwards.
Phil Cohen
filemerge: extract _maketemp and _makebackup...
r34033 """
if fcd.isabsent():
return None
Phil Cohen
filemerge: use arbitraryfilectx for backups...
r34783 # TODO: Break this import cycle somehow. (filectx -> ctx -> fileset ->
# merge -> filemerge). (I suspect the fileset import is the weakest link)
from . import context
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
filemerge: migrate to scmutil.backuppath()...
r41749 back = scmutil.backuppath(ui, repo, fcd.path())
Augie Fackler
formatting: blacken the codebase...
r43346 inworkingdir = back.startswith(repo.wvfs.base) and not back.startswith(
repo.vfs.base
)
Phil Cohen
filemerge: store backups in the overlayworkingctx if using imm...
r34785 if isinstance(fcd, context.overlayworkingfilectx) and inworkingdir:
# If the backup file is to be in the working directory, and we're
# merging in-memory, we must redirect the backup to the memory context
# so we don't disturb the working directory.
Augie Fackler
formatting: blacken the codebase...
r43346 relpath = back[len(repo.wvfs.base) + 1 :]
Phil Cohen
filemerge: only write in-memory backup during premerge...
r35721 if premerge:
wctx[relpath].write(fcd.data(), fcd.flags())
Phil Cohen
filemerge: store backups in the overlayworkingctx if using imm...
r34785 return wctx[relpath]
else:
Phil Cohen
filemerge: fix backing up an in-memory file to a custom location...
r35720 if premerge:
# Otherwise, write to wherever path the user specified the backups
# should go. We still need to switch based on whether the source is
# in-memory so we can use the fast path of ``util.copy`` if both are
# on disk.
if isinstance(fcd, context.overlayworkingfilectx):
util.writefile(back, fcd.data())
else:
Martin von Zweigbergk
filemerge: migrate to scmutil.backuppath()...
r41749 a = _workingpath(repo, fcd)
Phil Cohen
filemerge: fix backing up an in-memory file to a custom location...
r35720 util.copyfile(a, back)
Phil Cohen
filemerge: store backups in the overlayworkingctx if using imm...
r34785 # A arbitraryfilectx is returned, so we can run the same functions on
# the backup context regardless of where it lives.
return context.arbitraryfilectx(back, repo=repo)
Phil Cohen
filemerge: extract _maketemp and _makebackup...
r34033
Augie Fackler
formatting: blacken the codebase...
r43346
Kyle Lippincott
filemerge: move temp file unlinks to _maketempfiles...
r37016 @contextlib.contextmanager
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 def _maketempfiles(repo, fco, fca, localpath, uselocalpath):
"""Writes out `fco` and `fca` as temporary files, and (if uselocalpath)
copies `localpath` to another temporary file, so an external merge tool may
use them.
Phil Cohen
filemerge: extract _maketemp and _makebackup...
r34033 """
Kyle Lippincott
filemerge: use a single temp dir instead of temp files...
r37017 tmproot = None
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 tmprootprefix = repo.ui.config(b'experimental', b'mergetempdirprefix')
Kyle Lippincott
filemerge: use a single temp dir instead of temp files...
r37017 if tmprootprefix:
Yuya Nishihara
py3: wrap tempfile.mkdtemp() to use bytes path...
r38183 tmproot = pycompat.mkdtemp(prefix=tmprootprefix)
Kyle Lippincott
filemerge: use a single temp dir instead of temp files...
r37017
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 def maketempfrompath(prefix, path):
fullbase, ext = os.path.splitext(path)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 pre = b"%s~%s" % (os.path.basename(fullbase), prefix)
Kyle Lippincott
filemerge: use a single temp dir instead of temp files...
r37017 if tmproot:
name = os.path.join(tmproot, pre)
if ext:
name += ext
Augie Fackler
cleanup: remove pointless r-prefixes on double-quoted strings...
r43809 f = open(name, "wb")
Kyle Lippincott
filemerge: use a single temp dir instead of temp files...
r37017 else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 fd, name = pycompat.mkstemp(prefix=pre + b'.', suffix=ext)
Augie Fackler
cleanup: remove pointless r-prefixes on double-quoted strings...
r43809 f = os.fdopen(fd, "wb")
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 return f, name
def tempfromcontext(prefix, ctx):
f, name = maketempfrompath(prefix, ctx.path())
Phil Cohen
filemerge: extract _maketemp and _makebackup...
r34033 data = repo.wwritedata(ctx.path(), ctx.data())
f.write(data)
f.close()
return name
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b = tempfromcontext(b"base", fca)
c = tempfromcontext(b"other", fco)
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 d = localpath
if uselocalpath:
# We start off with this being the backup filename, so remove the .orig
# to make syntax-highlighting more likely.
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if d.endswith(b'.orig'):
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 d, _ = os.path.splitext(d)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 f, d = maketempfrompath(b"local", d)
with open(localpath, b'rb') as src:
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 f.write(src.read())
f.close()
Kyle Lippincott
filemerge: move temp file unlinks to _maketempfiles...
r37016 try:
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 yield b, c, d
Kyle Lippincott
filemerge: move temp file unlinks to _maketempfiles...
r37016 finally:
Kyle Lippincott
filemerge: use a single temp dir instead of temp files...
r37017 if tmproot:
shutil.rmtree(tmproot)
else:
util.unlink(b)
util.unlink(c)
Kyle Lippincott
filemerge: make the 'local' path match the format that 'base' and 'other' use...
r37095 # if not uselocalpath, d is the 'orig'/backup file which we
# shouldn't delete.
if d and uselocalpath:
util.unlink(d)
Phil Cohen
filemerge: extract _maketemp and _makebackup...
r34033
Augie Fackler
formatting: blacken the codebase...
r43346
Phil Cohen
merge: pass wctx to premerge, filemerge...
r34124 def _filemerge(premerge, repo, wctx, mynode, orig, fcd, fco, fca, labels=None):
Matt Mackall
filemerge: pull file-merging code into its own module
r6003 """perform a 3-way merge in the working directory
Siddharth Agarwal
filemerge: introduce a premerge flag and function...
r26607 premerge = whether this is a premerge
Matt Mackall
merge: introduce mergestate
r6512 mynode = parent node before merge
orig = original local filename before merge
fco = other file context
fca = ancestor file context
fcd = local file context for current/destination file
Siddharth Agarwal
filemerge: also return whether the merge is complete...
r26606
Siddharth Agarwal
filemerge: return whether the file was deleted...
r27034 Returns whether the merge is complete, the return value of the merge, and
a boolean indicating whether the file was deleted from disk."""
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Augie Fackler
formatting: blacken the codebase...
r43346 if not fco.cmp(fcd): # files identical?
Siddharth Agarwal
filemerge: return whether the file was deleted...
r27034 return True, None, False
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 ui = repo.ui
fd = fcd.path()
Martin von Zweigbergk
merge: respect ui.relative-paths...
r41651 uipathfn = scmutil.getuipathfn(repo)
fduipath = uipathfn(fd)
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 binary = fcd.isbinary() or fco.isbinary() or fca.isbinary()
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 symlink = b'l' in fcd.flags() + fco.flags()
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 changedelete = fcd.isabsent() or fco.isabsent()
tool, toolpath = _picktool(repo, ui, fd, binary, symlink, changedelete)
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 scriptfn = None
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if tool in internals and tool.startswith(b'internal:'):
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 # normalize to new-style names (':merge' etc)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 tool = tool[len(b'internal') :]
if toolpath and toolpath.startswith(b'python:'):
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 invalidsyntax = False
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if toolpath.count(b':') >= 2:
script, scriptfn = toolpath[7:].rsplit(b':', 1)
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 if not scriptfn:
invalidsyntax = True
# missing :callable can lead to spliting on windows drive letter
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if b'\\' in scriptfn or b'/' in scriptfn:
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 invalidsyntax = True
else:
invalidsyntax = True
if invalidsyntax:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 raise error.Abort(_(b"invalid 'python:' syntax: %s") % toolpath)
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 toolpath = script
Augie Fackler
formatting: blacken the codebase...
r43346 ui.debug(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b"picked tool '%s' for %s (binary %s symlink %s changedelete %s)\n"
Augie Fackler
formatting: blacken the codebase...
r43346 % (
tool,
fduipath,
pycompat.bytestr(binary),
pycompat.bytestr(symlink),
pycompat.bytestr(changedelete),
)
)
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 if tool in internals:
func = internals[tool]
mergetype = func.mergetype
onfailure = func.onfailure
precheck = func.precheck
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 isexternal = False
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 else:
Phil Cohen
filemerge: only raise InMemoryMergeConflictsError when running _xmerge...
r35479 if wctx.isinmemory():
func = _xmergeimm
else:
func = _xmerge
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 mergetype = fullmerge
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 onfailure = _(b"merging %s failed!\n")
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 precheck = None
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 isexternal = True
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512
hindlemail
filemerge: support specifying a python function to custom merge-tools...
r38052 toolconf = tool, toolpath, binary, symlink, scriptfn
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 if mergetype == nomerge:
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 r, deleted = func(repo, mynode, orig, fcd, fco, fca, toolconf, labels)
Siddharth Agarwal
filemerge: return whether the file was deleted...
r27034 return True, r, deleted
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512
Siddharth Agarwal
filemerge: only print out "merging f" output at premerge step...
r26609 if premerge:
if orig != fco.path():
Augie Fackler
formatting: blacken the codebase...
r43346 ui.status(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b"merging %s and %s to %s\n")
Augie Fackler
formatting: blacken the codebase...
r43346 % (uipathfn(orig), uipathfn(fco.path()), fduipath)
)
Siddharth Agarwal
filemerge: only print out "merging f" output at premerge step...
r26609 else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ui.status(_(b"merging %s\n") % fduipath)
Siddharth Agarwal
filemerge: move 'merging' output to before file creation...
r26528
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ui.debug(b"my %s other %s ancestor %s\n" % (fcd, fco, fca))
Siddharth Agarwal
filemerge: move 'merging' output to before file creation...
r26528
Augie Fackler
formatting: blacken the codebase...
r43346 if precheck and not precheck(repo, mynode, orig, fcd, fco, fca, toolconf):
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 if onfailure:
Phil Cohen
filemerge: raise InMemoryMergeConflictsError if we hit merge conflicts in IMM...
r35283 if wctx.isinmemory():
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.InMemoryMergeConflictsError(
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 b'in-memory merge does not support merge conflicts'
Augie Fackler
formatting: blacken the codebase...
r43346 )
Martin von Zweigbergk
merge: respect ui.relative-paths...
r41651 ui.warn(onfailure % fduipath)
Siddharth Agarwal
filemerge: return whether the file was deleted...
r27034 return True, 1, False
Siddharth Agarwal
filemerge: move precheck to before files are written out...
r26529
Phil Cohen
filemerge: use arbitraryfilectx for backups...
r34783 back = _makebackup(repo, ui, wctx, fcd, premerge)
Phil Cohen
filemerge: reduce creation of tempfiles until needed...
r34037 files = (None, None, None, back)
Siddharth Agarwal
filemerge: clean up temp files in a finally block...
r26589 r = 1
try:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 internalmarkerstyle = ui.config(b'ui', b'mergemarkers')
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 if isexternal:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 markerstyle = _toolstr(ui, tool, b'mergemarkers')
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 else:
markerstyle = internalmarkerstyle
Siddharth Agarwal
filemerge: move precheck to before files are written out...
r26529 if not labels:
labels = _defaultconflictlabels
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 formattedlabels = labels
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if markerstyle != b'basic':
Augie Fackler
formatting: blacken the codebase...
r43346 formattedlabels = _formatlabels(
repo, fcd, fco, fca, labels, tool=tool
)
Matt Mackall
merge: allow smarter tool configuration...
r6004
Siddharth Agarwal
filemerge: introduce a premerge flag and function...
r26607 if premerge and mergetype == fullmerge:
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 # conflict markers generated by premerge will use 'detailed'
# settings if either ui.mergemarkers or the tool's mergemarkers
# setting is 'detailed'. This way tools can have basic labels in
# space-constrained areas of the UI, but still get full information
# in conflict markers if premerge is 'keep' or 'keep-merge3'.
premergelabels = labels
labeltool = None
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if markerstyle != b'basic':
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925 # respect 'tool's mergemarkertemplate (which defaults to
# ui.mergemarkertemplate)
labeltool = tool
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if internalmarkerstyle != b'basic' or markerstyle != b'basic':
Augie Fackler
formatting: blacken the codebase...
r43346 premergelabels = _formatlabels(
repo, fcd, fco, fca, premergelabels, tool=labeltool
)
Kyle Lippincott
filemerge: support passing labels to external merge tools...
r35925
Augie Fackler
formatting: blacken the codebase...
r43346 r = _premerge(
repo, fcd, fco, fca, toolconf, files, labels=premergelabels
)
Siddharth Agarwal
filemerge: break overall filemerge into separate premerge and merge steps...
r26611 # complete if premerge successful (r is 0)
Siddharth Agarwal
filemerge: return whether the file was deleted...
r27034 return not r, r, False
Siddharth Agarwal
filemerge: call premerge directly from main merge function...
r26567
Augie Fackler
formatting: blacken the codebase...
r43346 needcheck, r, deleted = func(
repo,
mynode,
orig,
fcd,
fco,
fca,
toolconf,
files,
labels=formattedlabels,
)
Siddharth Agarwal
filemerge: return whether the file is deleted from all other merge tools...
r27033
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 if needcheck:
Phil Cohen
filemerge: add `_workingpath`...
r34036 r = _check(repo, r, ui, tool, fcd, files)
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 if r:
if onfailure:
Phil Cohen
filemerge: raise InMemoryMergeConflictsError if we hit merge conflicts in IMM...
r35283 if wctx.isinmemory():
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.InMemoryMergeConflictsError(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'in-memory merge '
b'does not support '
b'merge conflicts'
Augie Fackler
formatting: blacken the codebase...
r43346 )
Martin von Zweigbergk
merge: respect ui.relative-paths...
r41651 ui.warn(onfailure % fduipath)
Ryan McElroy
merge: allow user to halt merge on merge-tool failures...
r34798 _onfilemergefailure(ui)
Siddharth Agarwal
filemerge: clean up temp files in a finally block...
r26589
Siddharth Agarwal
filemerge: return whether the file was deleted...
r27034 return True, r, deleted
Siddharth Agarwal
filemerge: clean up temp files in a finally block...
r26589 finally:
Siddharth Agarwal
filemerge: don't try to copy files known to be absent...
r27047 if not r and back is not None:
Phil Cohen
filemerge: use arbitraryfilectx for backups...
r34783 back.remove()
Matt Mackall
merge: allow smarter tool configuration...
r6004
Augie Fackler
formatting: blacken the codebase...
r43346
Ryan McElroy
filemerge: introduce functions to halt merge flow...
r34797 def _haltmerge():
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 msg = _(b'merge halted after failed merge (see hg resolve)')
Ryan McElroy
filemerge: introduce functions to halt merge flow...
r34797 raise error.InterventionRequired(msg)
Augie Fackler
formatting: blacken the codebase...
r43346
Ryan McElroy
filemerge: introduce functions to halt merge flow...
r34797 def _onfilemergefailure(ui):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 action = ui.config(b'merge', b'on-failure')
if action == b'prompt':
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 msg = _(b'continue merge operation (yn)?$$ &Yes $$ &No')
Ryan McElroy
filemerge: introduce functions to halt merge flow...
r34797 if ui.promptchoice(msg, 0) == 1:
_haltmerge()
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if action == b'halt':
Ryan McElroy
filemerge: introduce functions to halt merge flow...
r34797 _haltmerge()
# default action is 'continue', in which case we neither prompt nor halt
Augie Fackler
formatting: blacken the codebase...
r43346
Kyle Lippincott
resolve: add option to warn/abort on -m with unresolved conflict markers...
r38829 def hasconflictmarkers(data):
Augie Fackler
formatting: blacken the codebase...
r43346 return bool(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 re.search(b"^(<<<<<<< .*|=======|>>>>>>> .*)$", data, re.MULTILINE)
Augie Fackler
formatting: blacken the codebase...
r43346 )
Kyle Lippincott
resolve: add option to warn/abort on -m with unresolved conflict markers...
r38829
Phil Cohen
filemerge: add `_workingpath`...
r34036 def _check(repo, r, ui, tool, fcd, files):
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 fd = fcd.path()
Martin von Zweigbergk
merge: respect ui.relative-paths...
r41651 uipathfn = scmutil.getuipathfn(repo)
Phil Cohen
filemerge: add `_workingpath`...
r34036 unused, unused, unused, back = files
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575
Augie Fackler
formatting: blacken the codebase...
r43346 if not r and (
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _toolbool(ui, tool, b"checkconflicts")
or b'conflicts' in _toollist(ui, tool, b"check")
Augie Fackler
formatting: blacken the codebase...
r43346 ):
Kyle Lippincott
resolve: add option to warn/abort on -m with unresolved conflict markers...
r38829 if hasconflictmarkers(fcd.data()):
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 r = 1
checked = False
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if b'prompt' in _toollist(ui, tool, b"check"):
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 checked = True
Augie Fackler
formatting: blacken the codebase...
r43346 if ui.promptchoice(
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 _(b"was merge of '%s' successful (yn)?$$ &Yes $$ &No")
Augie Fackler
formatting: blacken the codebase...
r43346 % uipathfn(fd),
1,
):
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 r = 1
Augie Fackler
formatting: blacken the codebase...
r43346 if (
not r
and not checked
and (
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _toolbool(ui, tool, b"checkchanged")
or b'changed' in _toollist(ui, tool, b"check")
Augie Fackler
formatting: blacken the codebase...
r43346 )
):
Phil Cohen
filemerge: use arbitraryfilectx for backups...
r34783 if back is not None and not fcd.cmp(back):
Augie Fackler
formatting: blacken the codebase...
r43346 if ui.promptchoice(
_(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b" output file %s appears unchanged\n"
b"was merge successful (yn)?"
b"$$ &Yes $$ &No"
Augie Fackler
formatting: blacken the codebase...
r43346 )
% uipathfn(fd),
1,
):
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 r = 1
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if back is not None and _toolbool(ui, tool, b"fixeol"):
Phil Cohen
filemerge: add `_workingpath`...
r34036 _matcheol(_workingpath(repo, fcd), back)
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575
return r
Augie Fackler
formatting: blacken the codebase...
r43346
Phil Cohen
filemerge: add `_workingpath`...
r34036 def _workingpath(repo, ctx):
return repo.wjoin(ctx.path())
Augie Fackler
formatting: blacken the codebase...
r43346
Phil Cohen
merge: pass wctx to premerge, filemerge...
r34124 def premerge(repo, wctx, mynode, orig, fcd, fco, fca, labels=None):
Augie Fackler
formatting: blacken the codebase...
r43346 return _filemerge(
True, repo, wctx, mynode, orig, fcd, fco, fca, labels=labels
)
Siddharth Agarwal
filemerge: introduce a premerge flag and function...
r26607
Phil Cohen
merge: pass wctx to premerge, filemerge...
r34124 def filemerge(repo, wctx, mynode, orig, fcd, fco, fca, labels=None):
Augie Fackler
formatting: blacken the codebase...
r43346 return _filemerge(
False, repo, wctx, mynode, orig, fcd, fco, fca, labels=labels
)
Siddharth Agarwal
filemerge: add a wrapper around the filemerge function...
r26605
FUJIWARA Katsunori
filemerge: move decorator definition for internal merge tools to registrar...
r33663 def loadinternalmerge(ui, extname, registrarobj):
"""Load internal merge tool from specified registrarobj
"""
Gregory Szorc
py3: finish porting iteritems() to pycompat and remove source transformer...
r43376 for name, func in pycompat.iteritems(registrarobj._table):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 fullname = b':' + name
FUJIWARA Katsunori
filemerge: move decorator definition for internal merge tools to registrar...
r33663 internals[fullname] = func
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 internals[b'internal:' + name] = func
FUJIWARA Katsunori
filemerge: move decorator definition for internal merge tools to registrar...
r33663 internalsdoc[fullname] = func
FUJIWARA Katsunori
filemerge: show actual capabilities of internal merge tools...
r39162 capabilities = sorted([k for k, v in func.capabilities.items() if v])
if capabilities:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 capdesc = b" (actual capabilities: %s)" % b', '.join(
capabilities
)
func.__doc__ = func.__doc__ + pycompat.sysstr(b"\n\n%s" % capdesc)
FUJIWARA Katsunori
filemerge: avoid putting translated text into docstring...
r39303
# to put i18n comments into hg.pot for automatically generated texts
Matt Harbison
filemerge: fix an i18n comment typo
r39395 # i18n: "binary" and "symlink" are keywords
FUJIWARA Katsunori
filemerge: avoid putting translated text into docstring...
r39303 # i18n: this text is added automatically
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b" (actual capabilities: binary, symlink)")
FUJIWARA Katsunori
filemerge: avoid putting translated text into docstring...
r39303 # i18n: "binary" is keyword
# i18n: this text is added automatically
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b" (actual capabilities: binary)")
FUJIWARA Katsunori
filemerge: avoid putting translated text into docstring...
r39303 # i18n: "symlink" is keyword
# i18n: this text is added automatically
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b" (actual capabilities: symlink)")
FUJIWARA Katsunori
filemerge: show actual capabilities of internal merge tools...
r39162
Augie Fackler
formatting: blacken the codebase...
r43346
FUJIWARA Katsunori
filemerge: move decorator definition for internal merge tools to registrar...
r33663 # load built-in merge tools explicitly to setup internalsdoc
loadinternalmerge(None, None, internaltool)
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 # tell hggettext to extract docstrings from these functions:
i18nfunctions = internals.values()