##// END OF EJS Templates
templatekw: introduce showlatesttags() to handle {latesttag} keywords...
templatekw: introduce showlatesttags() to handle {latesttag} keywords The keywords {changes}, {distance} and {tag} will be available on a future template method that will allow pattern matching against tag names. For consistency, these should be available on the existing {latesttag} keyword as well. I debated whether or not to add {tag} instead of just continuing with the existing {latesttag}. But it seems clearer not to have the same name for two distinct things (a list in the LHS of %, and an individual tag value on the right). The value of latesttags[0] is the date of commit for the cset to which the tag is applied (i.e. not the date the tag was applied), and therefore isn't made visible because it doesn't seem interesting. It appears that this is merely an internal implementation detail for sorting csets in a stable manner when there are different branches.

File last commit:

r26224:a4da463d default
r26484:93c80e7e default
Show More
filemerge.py
545 lines | 18.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
def internaltool(name, trymerge, onfailure=None):
'''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
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 func.trymerge = trymerge
func.onfailure = onfailure
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
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 @internaltool('prompt', False)
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)
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 @internaltool('local', False)
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
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 @internaltool('other', False)
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
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 @internaltool('fail', False)
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
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."""
Mads Kiilerich
merge: make internal merge fail cleanly on symlinks...
r18256 tool, toolpath, binary, symlink = toolconf
if symlink:
Mads Kiilerich
filemerge: switch the default name for internal tools from internal:x to :x
r22707 repo.ui.warn(_('warning: internal :merge cannot merge symlinks '
Mads Kiilerich
merge: warn when internal:merge cannot merge symlinks...
r18325 'for %s\n') % fcd.path())
Mads Kiilerich
merge: make internal merge fail cleanly on symlinks...
r18256 return False, 1
Durham Goode
merge: define conflict marker labels in filemerge()...
r21273 r = _premerge(repo, toolconf, files, labels=labels)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 if r:
a, b, c, back = files
ui = repo.ui
Erik Huelsmann
filemerge: split internal merge into api entry point and internal helper...
r26070 r = simplemerge.simplemerge(ui, a, b, c, label=labels, mode=mode)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 return True, r
return False, 0
Erik Huelsmann
filemerge: add 'union' merge to internal merge tool...
r26071 @internaltool('union', True,
_("merging %s incomplete! "
"(edit conflicts, then use 'hg resolve --mark')\n"))
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')
Erik Huelsmann
filemerge: split internal merge into api entry point and internal helper...
r26070 @internaltool('merge', True,
_("merging %s incomplete! "
"(edit conflicts, then use 'hg resolve --mark')\n"))
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')
Pierre-Yves David
merge: add an internal:merge3 tool...
r22028 @internaltool('merge3', True,
_("merging %s incomplete! "
"(edit conflicts, then use 'hg resolve --mark')\n"))
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
@internaltool('merge-local', True)
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
@internaltool('merge-other', True)
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
Angel Ezquerra
filemerge: add internal:tagmerge merge tool...
r21922 @internaltool('tagmerge', True,
_("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)
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 @internaltool('dump', 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
same directory as ``a.txt``."""
Durham Goode
merge: define conflict marker labels in filemerge()...
r21273 r = _premerge(repo, toolconf, files, labels=labels)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 if r:
a, b, c, back = files
fd = fcd.path()
util.copyfile(a, a + ".local")
repo.wwrite(fd + ".other", fco.data(), fco.flags())
repo.wwrite(fd + ".base", fca.data(), fca.flags())
return False, r
Durham Goode
merge: define conflict marker labels in filemerge()...
r21273 def _xmerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
r = _premerge(repo, toolconf, files, labels=labels)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 if r:
tool, toolpath, binary, symlink = toolconf
a, b, c, back = files
out = ""
Augie Fackler
filemerge: move from dict() construction to {} literals...
r20676 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
ui = repo.ui
args = _toolstr(ui, tool, "args", '$local $base $other')
if "$output" in args:
out, a = a, back # read input from backup, write to original
Augie Fackler
filemerge: move from dict() construction to {} literals...
r20676 replace = {'local': a, 'base': b, 'other': c, 'output': out}
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 args = util.interpolate(r'\$', replace, args,
Keegan Carruthers-Smith
filemerge: use util.shellquote when calling merge (issue3581)
r17885 lambda s: util.shellquote(util.localpath(s)))
Mads Kiilerich
merge: better debug messages before/after invoking external merge tool
r24727 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)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 return True, r
return False, 0
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 """
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
Nicolas Dumazet
filectx: use cmp(self, fctx) instead of cmp(self, text)...
r11702 if not fco.cmp(fcd): # files identical?
Matt Mackall
filemerge: pull file-merging code into its own module
r6003 return None
Matt Mackall
merge: allow smarter tool configuration...
r6004 ui = repo.ui
Matt Mackall
merge: introduce mergestate
r6512 fd = fcd.path()
Laurens Holst
context: add isbinary function
r15738 binary = fcd.isbinary() or fco.isbinary() or fca.isbinary()
Matt Mackall
context: remove islink and isexec methods
r6744 symlink = 'l' in fcd.flags() + fco.flags()
Matt Mackall
merge: introduce mergestate
r6512 tool, toolpath = _picktool(repo, ui, fd, binary, symlink)
Martin Geisler
do not attempt to translate ui.debug output
r9467 ui.debug("picked tool '%s' for %s (binary %s symlink %s)\n" %
Matt Mackall
merge: introduce mergestate
r6512 (tool, fd, binary, symlink))
Matt Mackall
merge: allow smarter tool configuration...
r6004
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 if tool in internals:
func = internals[tool]
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 trymerge = func.trymerge
onfailure = func.onfailure
else:
func = _xmerge
trymerge = True
onfailure = _("merging %s failed!\n")
Matt Mackall
merge: allow smarter tool configuration...
r6004
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 toolconf = tool, toolpath, binary, symlink
if not trymerge:
return func(repo, mynode, orig, fcd, fco, fca, toolconf)
Matt Mackall
filemerge: pull file-merging code into its own module
r6003 a = repo.wjoin(fd)
b = temp("base", fca)
c = temp("other", fco)
Matt Mackall
merge: allow smarter tool configuration...
r6004 back = a + ".orig"
util.copyfile(a, back)
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Matt Mackall
merge: introduce mergestate
r6512 if orig != fco.path():
Martin Geisler
use ui instead of repo.ui when the former is in scope
r8615 ui.status(_("merging %s and %s to %s\n") % (orig, fco.path(), fd))
Matt Mackall
filemerge: pull file-merging code into its own module
r6003 else:
Martin Geisler
use ui instead of repo.ui when the former is in scope
r8615 ui.status(_("merging %s\n") % fd)
Matt Mackall
merge: introduce mergestate
r6512
Martin Geisler
do not attempt to translate ui.debug output
r9467 ui.debug("my %s other %s ancestor %s\n" % (fcd, fco, fca))
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
FUJIWARA Katsunori
filemerge: use 'basic' as the default of '[ui] mergemarkers' for safety...
r21918 markerstyle = ui.config('ui', 'mergemarkers', 'basic')
Pierre-Yves David
merge: refactor labels selection code...
r22021 if not labels:
labels = _defaultconflictlabels
if markerstyle != 'basic':
Pierre-Yves David
filemerge: allow the formatting of three labels instead of two...
r22026 labels = _formatlabels(repo, fcd, fco, fca, labels)
Durham Goode
merge: add conflict marker formatter (BC)...
r21519
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 needcheck, r = func(repo, mynode, orig, fcd, fco, fca, toolconf,
Pierre-Yves David
merge: refactor labels selection code...
r22021 (a, b, c, back), labels=labels)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 if not needcheck:
if r:
if onfailure:
ui.warn(onfailure % fd)
else:
Mads Kiilerich
filemerge: better handling of failing remove of temporary files...
r21100 util.unlink(back)
Matt Mackall
merge: allow smarter tool configuration...
r6004
Mads Kiilerich
filemerge: better handling of failing remove of temporary files...
r21100 util.unlink(b)
util.unlink(c)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 return r
Matt Mackall
merge: allow smarter tool configuration...
r6004
David Champion
merge: introduce tool.check parameter...
r11148 if not r and (_toolbool(ui, tool, "checkconflicts") or
'conflicts' in _toollist(ui, tool, "check")):
Thomas Arendsen Hein
Fix merge-tools.checkconflicts...
r12046 if re.search("^(<<<<<<< .*|=======|>>>>>>> .*)$", fcd.data(),
re.MULTILINE):
Matt Mackall
merge: allow smarter tool configuration...
r6004 r = 1
David Champion
merge: tool.check = prompt will force an interactive merge check...
r11149 checked = False
if 'prompt' in _toollist(ui, tool, "check"):
checked = True
Matt Mackall
ui: merge prompt text components into a singe string...
r19226 if ui.promptchoice(_("was merge of '%s' successful (yn)?"
"$$ &Yes $$ &No") % fd, 1):
David Champion
merge: tool.check = prompt will force an interactive merge check...
r11149 r = 1
if not r and not checked and (_toolbool(ui, tool, "checkchanged") or
'changed' in _toollist(ui, tool, "check")):
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 if filecmp.cmp(a, back):
Simon Heimberg
ui: extract choice from prompt...
r9048 if ui.promptchoice(_(" output file %s appears unchanged\n"
Matt Mackall
ui: merge prompt text components into a singe string...
r19226 "was merge successful (yn)?"
"$$ &Yes $$ &No") % fd, 1):
Steve Borho
filemerge: add 'checkchanged' merge tool property
r6075 r = 1
Matt Mackall
merge: add support for tool EOL fixups...
r6005 if _toolbool(ui, tool, "fixeol"):
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 _matcheol(a, back)
Matt Mackall
merge: add support for tool EOL fixups...
r6005
Matt Mackall
filemerge: pull file-merging code into its own module
r6003 if r:
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 if onfailure:
ui.warn(onfailure % fd)
Matt Mackall
merge: allow smarter tool configuration...
r6004 else:
Mads Kiilerich
filemerge: better handling of failing remove of temporary files...
r21100 util.unlink(back)
Matt Mackall
filemerge: pull file-merging code into its own module
r6003
Mads Kiilerich
filemerge: better handling of failing remove of temporary files...
r21100 util.unlink(b)
util.unlink(c)
Matt Mackall
filemerge: pull file-merging code into its own module
r6003 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()