##// END OF EJS Templates
filemerge: clean up temp files in a finally block...
filemerge: clean up temp files in a finally block This isn't really a big deal because the temp files are created in $TMPDIR, but it makes some upcoming work simpler.

File last commit:

r26589:fb388aa2 default
r26589:fb388aa2 default
Show More
filemerge.py
571 lines | 19.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
import filecmp
import os
import re
import tempfile
from .i18n import _
from .node import short
from . import (
error,
match,
simplemerge,
tagmerge,
templatekw,
templater,
util,
)
Matt Mackall
merge: allow smarter tool configuration...
r6004
Matt Mackall
filemerge: handle missing regappend
r6013 def _toolstr(ui, tool, part, default=""):
Matt Mackall
merge: allow smarter tool configuration...
r6004 return ui.config("merge-tools", tool + "." + part, default)
def _toolbool(ui, tool, part, default=False):
return ui.configbool("merge-tools", tool + "." + part, default)
David Champion
merge: introduce tool.check parameter...
r11148 def _toollist(ui, tool, part, default=[]):
return ui.configlist("merge-tools", tool + "." + part, default)
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
Siddharth Agarwal
filemerge: add some merge types...
r26525 # internal tool merge types
nomerge = None
mergeonly = 'mergeonly' # just the full merge, no premerge
fullmerge = 'fullmerge' # both premerge and merge
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 def internaltool(name, mergetype, onfailure=None, precheck=None):
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 '''return a decorator for populating internal merge tool table'''
def decorator(func):
Mads Kiilerich
filemerge: switch the default name for internal tools from internal:x to :x
r22707 fullname = ':' + name
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 func.__doc__ = "``%s``\n" % fullname + func.__doc__.strip()
internals[fullname] = func
Mads Kiilerich
filemerge: switch the default name for internal tools from internal:x to :x
r22707 internals['internal:' + name] = func
Gregory Szorc
help.merge-tools: do not double document merge tools...
r24099 internalsdoc[fullname] = func
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 func.mergetype = mergetype
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 func.onfailure = onfailure
Siddharth Agarwal
filemerge: add a before-merge callback to internal merge tools...
r26513 func.precheck = precheck
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 return func
return decorator
Matt Mackall
filemerge: add internal:prompt target
r8830
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
Matt Harbison
filemerge: split the logic for finding an external tool to its own function...
r23148 return findexternaltool(ui, tool)
def findexternaltool(ui, tool):
Steve Borho
filemerge: introduce a 'regkeyalt' merge tool variable...
r13565 for kn in ("regkey", "regkeyalt"):
k = _toolstr(ui, tool, kn)
if not k:
continue
Adrian Buehlmann
rename util.lookup_reg to lookupreg
r14230 p = util.lookupreg(k, _toolstr(ui, tool, "regname"))
Matt Mackall
merge: add registry look up bits to tool search
r6006 if p:
Adrian Buehlmann
rename util.find_exe to findexe
r14271 p = util.findexe(p + _toolstr(ui, tool, "regappend"))
Matt Mackall
merge: add registry look up bits to tool search
r6006 if p:
return p
Greg Ward
merge: expand environment variables and ~/ in tool.executable...
r15264 exe = _toolstr(ui, tool, "executable", tool)
return util.findexe(util.expandpath(exe))
Matt Mackall
merge: allow smarter tool configuration...
r6004
def _picktool(repo, ui, path, binary, symlink):
def check(tool, pat, symlink, binary):
tmsg = tool
if pat:
tmsg += " specified for " + pat
Mads Kiilerich
More verbose logging when filemerge searches for merge-tool...
r7397 if not _findtool(ui, tool):
if pat: # explicitly requested tool deserves a warning
ui.warn(_("couldn't find merge tool %s\n") % tmsg)
else: # configured but non-existing tools are more silent
ui.note(_("couldn't find merge tool %s\n") % tmsg)
Matt Mackall
merge: allow smarter tool configuration...
r6004 elif symlink and not _toolbool(ui, tool, "symlink"):
ui.warn(_("tool %s can't handle symlinks\n") % tmsg)
elif binary and not _toolbool(ui, tool, "binary"):
ui.warn(_("tool %s can't handle binary\n") % tmsg)
Matt Mackall
filemerge: add config item for GUI tools...
r6007 elif not util.gui() and _toolbool(ui, tool, "gui"):
ui.warn(_("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
force = ui.config('ui', 'forcemerge')
if force:
toolpath = _findtool(ui, force)
if toolpath:
Keegan Carruthers-Smith
filemerge: use util.shellquote when calling merge (issue3581)
r17885 return (force, util.shellquote(toolpath))
Steve Borho
merge: implement --tool arguments using new ui.forcemerge configurable...
r12788 else:
# mimic HGMERGE if given tool not found
return (force, force)
# HGMERGE takes next precedence
Steve Borho
filemerge: wrap quotes around tool path
r6025 hgmerge = os.environ.get("HGMERGE")
if hgmerge:
return (hgmerge, hgmerge)
Matt Mackall
merge: allow smarter tool configuration...
r6004
# then patterns
dhruva
filemerge: fix pattern matching
r6016 for pat, tool in ui.configitems("merge-patterns"):
Matt Mackall
match: add some default args
r8567 mf = match.match(repo.root, '', [pat])
Matt Mackall
merge: allow smarter tool configuration...
r6004 if mf(path) and check(tool, pat, symlink, False):
Benoit Boissinot
fix spaces/identation issues
r10339 toolpath = _findtool(ui, tool)
Keegan Carruthers-Smith
filemerge: use util.shellquote when calling merge (issue3581)
r17885 return (tool, util.shellquote(toolpath))
Matt Mackall
merge: allow smarter tool configuration...
r6004
# then merge tools
tools = {}
Matt Mackall
many, many trivial check-code fixups
r10282 for k, v in ui.configitems("merge-tools"):
Matt Mackall
merge: allow smarter tool configuration...
r6004 t = k.split('.')[0]
if t not in tools:
tools[t] = int(_toolstr(ui, t, "priority", "0"))
Steve Borho
filemerge: more backwards compatible behavior for ui.merge...
r6076 names = tools.keys()
Matt Mackall
many, many trivial check-code fixups
r10282 tools = sorted([(-p, t) for t, p in tools.items()])
Steve Borho
filemerge: more backwards compatible behavior for ui.merge...
r6076 uimerge = ui.config("ui", "merge")
if uimerge:
if uimerge not in names:
return (uimerge, uimerge)
tools.insert(0, (None, uimerge)) # highest priority
Matt Mackall
merge: allow smarter tool configuration...
r6004 tools.append((None, "hgmerge")) # the old default, if found
Matt Mackall
many, many trivial check-code fixups
r10282 for p, t in tools:
Mads Kiilerich
More verbose logging when filemerge searches for merge-tool...
r7397 if check(t, None, symlink, binary):
toolpath = _findtool(ui, t)
Keegan Carruthers-Smith
filemerge: use util.shellquote when calling merge (issue3581)
r17885 return (t, util.shellquote(toolpath))
Matt Mackall
filemerge: restore default prompt for binary/symlink lost in 83925d3a4559...
r16254
# internal merge or prompt as last resort
if symlink or binary:
Mads Kiilerich
filemerge: switch the default name for internal tools from internal:x to :x
r22707 return ":prompt", None
return ":merge", None
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Matt Mackall
merge: add support for tool EOL fixups...
r6005 def _eoltype(data):
"Guess the EOL type of a file"
if '\0' in data: # binary
return None
if '\r\n' in data: # Windows
return '\r\n'
if '\r' in data: # Old Mac
return '\r'
if '\n' in data: # UNIX
return '\n'
return None # unknown
def _matcheol(file, origfile):
"Convert EOL markers in a file to match origfile"
Dan Villiom Podlaski Christiansen
prevent transient leaks of file handle by using new helper functions...
r14168 tostyle = _eoltype(util.readfile(origfile))
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
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('prompt', nomerge)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 def _iprompt(repo, mynode, orig, fcd, fco, fca, toolconf):
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 """Asks the user which of the local or the other version to keep as
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 the merged version."""
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 ui = repo.ui
fd = fcd.path()
if ui.promptchoice(_(" no tool found to merge %s\n"
Matt Mackall
ui: merge prompt text components into a singe string...
r19226 "keep (l)ocal or take (o)ther?"
"$$ &Local $$ &Other") % fd, 0):
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 return _iother(repo, mynode, orig, fcd, fco, fca, toolconf)
else:
return _ilocal(repo, mynode, orig, fcd, fco, fca, toolconf)
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('local', nomerge)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 def _ilocal(repo, mynode, orig, fcd, fco, fca, toolconf):
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 """Uses the local version of files as the merged version."""
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 return 0
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('other', nomerge)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 def _iother(repo, mynode, orig, fcd, fco, fca, toolconf):
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 """Uses the other version of files as the merged version."""
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 repo.wwrite(fcd.path(), fco.data(), fco.flags())
return 0
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('fail', nomerge)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 def _ifail(repo, mynode, orig, fcd, fco, fca, toolconf):
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."""
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 return 1
Durham Goode
merge: define conflict marker labels in filemerge()...
r21273 def _premerge(repo, toolconf, files, labels=None):
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 tool, toolpath, binary, symlink = toolconf
Mads Kiilerich
merge: never do premerge on symlinks...
r18257 if symlink:
return 1
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 a, b, c, back = files
ui = repo.ui
Pierre-Yves David
merge-tools: add a `premerge=keep-merge3` config option...
r22032 validkeep = ['keep', '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:
Mads Kiilerich
merge: never do premerge on symlinks...
r18257 premerge = _toolbool(ui, tool, "premerge", not binary)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 except error.ConfigError:
premerge = _toolstr(ui, tool, "premerge").lower()
Pierre-Yves David
merge-tools: make premerge valid values extensible...
r22031 if premerge not in validkeep:
_valid = ', '.join(["'" + v + "'" for v in validkeep])
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 raise error.ConfigError(_("%s.premerge not valid "
"('%s' is neither boolean nor %s)") %
(tool, premerge, _valid))
if premerge:
Pierre-Yves David
merge-tools: add a `premerge=keep-merge3` config option...
r22032 if premerge == 'keep-merge3':
if not labels:
labels = _defaultconflictlabels
if len(labels) < 3:
labels.append('base')
Pierre-Yves David
simplemerge: burn "minimal" feature to the ground...
r22023 r = simplemerge.simplemerge(ui, a, b, c, quiet=True, label=labels)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 if not r:
ui.debug(" premerge successful\n")
return 0
Pierre-Yves David
merge-tools: make premerge valid values extensible...
r22031 if premerge not in validkeep:
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 util.copyfile(back, a) # restore from backup and try again
return 1 # continue merging
Siddharth Agarwal
filemerge: add a precheck for symlinks...
r26515 def _symlinkcheck(repo, mynode, orig, fcd, fco, fca, toolconf):
tool, toolpath, binary, symlink = toolconf
if symlink:
Siddharth Agarwal
filemerge: print correct name of tool for symlink checks...
r26518 repo.ui.warn(_('warning: internal %s cannot merge symlinks '
'for %s\n') % (tool, fcd.path()))
Siddharth Agarwal
filemerge: add a precheck for symlinks...
r26515 return False
return True
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 a, b, c, back = files
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Siddharth Agarwal
filemerge._merge: drop no longer necessary 'if r:' check...
r26572 ui = repo.ui
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Siddharth Agarwal
filemerge._merge: drop no longer necessary 'if r:' check...
r26572 r = simplemerge.simplemerge(ui, a, b, c, label=labels, mode=mode)
return True, r
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('union', fullmerge,
Erik Huelsmann
filemerge: add 'union' merge to internal merge tool...
r26071 _("merging %s incomplete! "
Siddharth Agarwal
filemerge: use symlinkcheck for :merge and :union...
r26516 "(edit conflicts, then use 'hg resolve --mark')\n"),
precheck=_symlinkcheck)
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."""
return _merge(repo, mynode, orig, fcd, fco, fca, toolconf,
files, labels, 'union')
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('merge', fullmerge,
Erik Huelsmann
filemerge: split internal merge into api entry point and internal helper...
r26070 _("merging %s incomplete! "
Siddharth Agarwal
filemerge: use symlinkcheck for :merge and :union...
r26516 "(edit conflicts, then use 'hg resolve --mark')\n"),
precheck=_symlinkcheck)
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."""
return _merge(repo, mynode, orig, fcd, fco, fca, toolconf,
files, labels, 'merge')
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('merge3', fullmerge,
Pierre-Yves David
merge: add an internal:merge3 tool...
r22028 _("merging %s incomplete! "
Siddharth Agarwal
filemerge: run symlink check for :merge3...
r26519 "(edit conflicts, then use 'hg resolve --mark')\n"),
precheck=_symlinkcheck)
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:
labels.append('base')
return _imerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels)
Jordi Gutiérrez Hermoso
filemerge: add non-interactive :merge-local and :merge-other...
r26224 def _imergeauto(repo, mynode, orig, fcd, fco, fca, toolconf, files,
labels=None, localorother=None):
"""
Generic driver for _imergelocal and _imergeother
"""
assert localorother is not None
tool, toolpath, binary, symlink = toolconf
if symlink:
repo.ui.warn(_('warning: :merge-%s cannot merge symlinks '
'for %s\n') % (localorother, fcd.path()))
return False, 1
a, b, c, back = files
r = simplemerge.simplemerge(repo.ui, a, b, c, label=labels,
localorother=localorother)
return True, r
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('merge-local', mergeonly)
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
of the local changes."""
success, status = _imergeauto(localorother='local', *args, **kwargs)
return success, status
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('merge-other', mergeonly)
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
of the other changes."""
success, status = _imergeauto(localorother='other', *args, **kwargs)
return success, status
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('tagmerge', mergeonly,
Angel Ezquerra
filemerge: add internal:tagmerge merge tool...
r21922 _("automatic tag merging of %s failed! "
Mads Kiilerich
filemerge: switch the default name for internal tools from internal:x to :x
r22707 "(use 'hg resolve --tool :merge' or another merge "
Angel Ezquerra
filemerge: add internal:tagmerge merge tool...
r21922 "tool of your choice)\n"))
def _itagmerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
"""
Uses the internal tag merge algorithm (experimental).
"""
return tagmerge.merge(repo, fcd, fco, fca)
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('dump', fullmerge)
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
same directory as ``a.txt``."""
Siddharth Agarwal
filemerge._idump: drop no longer necessary 'if r:' check...
r26573 a, b, c, back = files
fd = fcd.path()
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Siddharth Agarwal
filemerge._idump: drop no longer necessary 'if r:' check...
r26573 util.copyfile(a, a + ".local")
repo.wwrite(fd + ".other", fco.data(), fco.flags())
repo.wwrite(fd + ".base", fca.data(), fca.flags())
return False, 1
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Durham Goode
merge: define conflict marker labels in filemerge()...
r21273 def _xmerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
Siddharth Agarwal
filemerge._xmerge: drop no longer necessary 'if r:' check...
r26574 tool, toolpath, binary, symlink = toolconf
a, b, c, back = files
out = ""
env = {'HG_FILE': fcd.path(),
'HG_MY_NODE': short(mynode),
'HG_OTHER_NODE': str(fco.changectx()),
'HG_BASE_NODE': str(fca.changectx()),
'HG_MY_ISLINK': 'l' in fcd.flags(),
'HG_OTHER_ISLINK': 'l' in fco.flags(),
'HG_BASE_ISLINK': 'l' in fca.flags(),
}
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Siddharth Agarwal
filemerge._xmerge: drop no longer necessary 'if r:' check...
r26574 ui = repo.ui
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Siddharth Agarwal
filemerge._xmerge: drop no longer necessary 'if r:' check...
r26574 args = _toolstr(ui, tool, "args", '$local $base $other')
if "$output" in args:
out, a = a, back # read input from backup, write to original
replace = {'local': a, 'base': b, 'other': c, 'output': out}
args = util.interpolate(r'\$', replace, args,
lambda s: util.shellquote(util.localpath(s)))
cmd = toolpath + ' ' + args
repo.ui.debug('launching merge tool: %s\n' % cmd)
r = ui.system(cmd, cwd=repo.root, environ=env)
repo.ui.debug('merge tool returned: %s\n' % r)
return True, r
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Durham Goode
merge: add conflict marker formatter (BC)...
r21519 def _formatconflictmarker(repo, ctx, template, label, pad):
"""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()
props = templatekw.keywords.copy()
props['templ'] = template
props['ctx'] = ctx
props['repo'] = repo
templateresult = template('conflictmarker', **props)
label = ('%s:' % label).ljust(pad + 1)
mark = '%s %s' % (label, templater.stringify(templateresult))
FUJIWARA Katsunori
filemerge: use only the first line of the generated conflict marker for safety...
r21864 if mark:
mark = mark.splitlines()[0] # split for safety
FUJIWARA Katsunori
filemerge: use 'util.ellipsis' to trim custom conflict markers correctly...
r21865 # 8 for the prefix of conflict marker lines (e.g. '<<<<<<< ')
return util.ellipsis(mark, 80 - 8)
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
_defaultconflictmarker = ('{node|short} ' +
'{ifeq(tags, "tip", "", "{tags} ")}' +
'{if(bookmarks, "{bookmarks} ")}' +
'{ifeq(branch, "default", "", "{branch} ")}' +
Pierre-Yves David
merge: drop the quotes around commit description...
r21693 '- {author|user}: {desc|firstline}')
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
Durham Goode
merge: add labels parameter from merge.update to filemerge...
r21524 _defaultconflictlabels = ['local', 'other']
Pierre-Yves David
filemerge: allow the formatting of three labels instead of two...
r22026 def _formatlabels(repo, fcd, fco, fca, labels):
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
template = ui.config('ui', 'mergemarkertemplate', _defaultconflictmarker)
Pierre-Yves David
filemerge: drop extra white space...
r22025 tmpl = templater.templater(None, cache={'conflictmarker': template})
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
Pierre-Yves David
filemerge: allow the formatting of three labels instead of two...
r22026 newlabels = [_formatconflictmarker(repo, cd, tmpl, labels[0], pad),
_formatconflictmarker(repo, co, tmpl, labels[1], pad)]
if len(labels) > 2:
newlabels.append(_formatconflictmarker(repo, ca, tmpl, labels[2], pad))
return newlabels
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
Durham Goode
merge: add labels parameter from merge.update to filemerge...
r21524 def filemerge(repo, 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
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
Matt Mackall
filemerge: pull file-merging code into its own module
r6003 """
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512 if True:
def temp(prefix, ctx):
pre = "%s~%s." % (os.path.basename(ctx.path()), prefix)
(fd, name) = tempfile.mkstemp(prefix=pre)
data = repo.wwritedata(ctx.path(), ctx.data())
f = os.fdopen(fd, "wb")
f.write(data)
f.close()
return name
if not fco.cmp(fcd): # files identical?
return None
ui = repo.ui
fd = fcd.path()
binary = fcd.isbinary() or fco.isbinary() or fca.isbinary()
symlink = 'l' in fcd.flags() + fco.flags()
tool, toolpath = _picktool(repo, ui, fd, binary, symlink)
Siddharth Agarwal
filemerge: normalize 'internal:foo' names to ':foo'...
r26517 if tool in internals and tool.startswith('internal:'):
# normalize to new-style names (':merge' etc)
tool = tool[len('internal'):]
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512 ui.debug("picked tool '%s' for %s (binary %s symlink %s)\n" %
(tool, fd, binary, symlink))
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512 if tool in internals:
func = internals[tool]
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 mergetype = func.mergetype
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512 onfailure = func.onfailure
Siddharth Agarwal
filemerge: call precheck if available...
r26514 precheck = func.precheck
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512 else:
func = _xmerge
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 mergetype = fullmerge
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512 onfailure = _("merging %s failed!\n")
Siddharth Agarwal
filemerge: call precheck if available...
r26514 precheck = None
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512
toolconf = tool, toolpath, binary, symlink
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 if mergetype == nomerge:
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512 return func(repo, mynode, orig, fcd, fco, fca, toolconf)
Siddharth Agarwal
filemerge: move 'merging' output to before file creation...
r26528 if orig != fco.path():
ui.status(_("merging %s and %s to %s\n") % (orig, fco.path(), fd))
else:
ui.status(_("merging %s\n") % fd)
ui.debug("my %s other %s ancestor %s\n" % (fcd, fco, fca))
Siddharth Agarwal
filemerge: move precheck to before files are written out...
r26529 if precheck and not precheck(repo, mynode, orig, fcd, fco, fca,
toolconf):
if onfailure:
ui.warn(onfailure % fd)
return 1
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512 a = repo.wjoin(fd)
b = temp("base", fca)
c = temp("other", fco)
back = a + ".orig"
util.copyfile(a, back)
Siddharth Agarwal
filemerge.filemerge: make a tuple containing merge paths on disk...
r26527 files = (a, b, c, back)
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512
Siddharth Agarwal
filemerge: clean up temp files in a finally block...
r26589 r = 1
try:
Siddharth Agarwal
filemerge: move precheck to before files are written out...
r26529 markerstyle = ui.config('ui', 'mergemarkers', 'basic')
if not labels:
labels = _defaultconflictlabels
if markerstyle != 'basic':
labels = _formatlabels(repo, fcd, fco, fca, labels)
Matt Mackall
merge: allow smarter tool configuration...
r6004
Siddharth Agarwal
filemerge: call premerge directly from main merge function...
r26567 if mergetype == fullmerge:
r = _premerge(repo, toolconf, files, labels=labels)
if not r: # premerge successfully merged the file
needcheck = False
else:
needcheck, r = func(repo, mynode, orig, fcd, fco, fca, toolconf,
files, labels=labels)
Siddharth Agarwal
filemerge: call precheck if available...
r26514
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 if needcheck:
r = _check(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:
ui.warn(onfailure % fd)
Siddharth Agarwal
filemerge: clean up temp files in a finally block...
r26589
return r
finally:
if not r:
Mads Kiilerich
filemerge: better handling of failing remove of temporary files...
r21100 util.unlink(back)
util.unlink(b)
util.unlink(c)
Matt Mackall
merge: allow smarter tool configuration...
r6004
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 def _check(r, ui, tool, fcd, files):
fd = fcd.path()
a, b, c, back = files
if not r and (_toolbool(ui, tool, "checkconflicts") or
'conflicts' in _toollist(ui, tool, "check")):
if re.search("^(<<<<<<< .*|=======|>>>>>>> .*)$", fcd.data(),
re.MULTILINE):
r = 1
checked = False
if 'prompt' in _toollist(ui, tool, "check"):
checked = True
if ui.promptchoice(_("was merge of '%s' successful (yn)?"
"$$ &Yes $$ &No") % fd, 1):
r = 1
if not r and not checked and (_toolbool(ui, tool, "checkchanged") or
'changed' in
_toollist(ui, tool, "check")):
if filecmp.cmp(a, back):
if ui.promptchoice(_(" output file %s appears unchanged\n"
"was merge successful (yn)?"
"$$ &Yes $$ &No") % fd, 1):
r = 1
if _toolbool(ui, tool, "fixeol"):
_matcheol(a, back)
return r
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 # tell hggettext to extract docstrings from these functions:
i18nfunctions = internals.values()