##// END OF EJS Templates
merge with i18n
merge with i18n

File last commit:

r30924:48dea083 default
r31137:25703b62 merge 4.1.1 stable
Show More
filemerge.py
715 lines | 24.8 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 _
Siddharth Agarwal
filemerge: introduce class whose objects represent files not in a context...
r26979 from .node import nullid, short
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,
Siddharth Agarwal
origpath: move from cmdutil to scmutil...
r27651 scmutil,
Gregory Szorc
filemerge: use absolute_import
r25949 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: 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."""
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
def cmp(self, fctx):
"""compare with other file context
returns True if different from fctx.
"""
return not (fctx.isabsent() and
fctx.ctx() == self.ctx() and
fctx.path() == self.path())
def flags(self):
return ''
def changectx(self):
return self._ctx
def isbinary(self):
return False
def isabsent(self):
return True
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
Pulkit Goyal
py3: make format strings unicodes and not bytes...
r30073 func.__doc__ = (pycompat.sysstr("``%s``\n" % fullname)
+ func.__doc__.strip())
Matt Mackall
filemerge: remove some redundancy in decorators/docstrings
r16127 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
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 def _picktool(repo, ui, path, binary, symlink, changedelete):
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:
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)
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
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)
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 if changedelete and not supportscd(toolpath):
return ":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:
return (force, util.shellquote(toolpath))
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
Pulkit Goyal
py3: replace os.environ with encoding.environ (part 3 of 5)
r30636 hgmerge = encoding.environ.get("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):
return ":prompt", None
else:
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])
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 if mf(path) and check(tool, pat, symlink, False, changedelete):
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 = {}
Augie Fackler
merge-tools: allow marking a mergetool as completely disabled...
r26730 disabled = set()
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"))
Augie Fackler
merge-tools: allow marking a mergetool as completely disabled...
r26730 if _toolbool(ui, t, "disabled", False):
disabled.add(t)
Steve Borho
filemerge: more backwards compatible behavior for ui.merge...
r6076 names = tools.keys()
Augie Fackler
filemerge: avoid shadowing a variable in a list comprehension
r30388 tools = sorted([(-p, tool) for tool, p in tools.items()
if tool not in disabled])
Steve Borho
filemerge: more backwards compatible behavior for ui.merge...
r6076 uimerge = ui.config("ui", "merge")
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
if uimerge not in names and not changedelete:
Steve Borho
filemerge: more backwards compatible behavior for ui.merge...
r6076 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:
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)
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
Siddharth Agarwal
filemerge._picktool: only pick from nomerge tools for change/delete conflicts...
r27039 if symlink or binary or changedelete:
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)
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()
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 prompts = partextras(labels)
prompts['fd'] = 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():
index = ui.promptchoice(
Simon Farnsworth
merge: always use other, not remote, in user prompts...
r29775 _("local%(l)s changed %(fd)s which other%(o)s deleted\n"
Siddharth Agarwal
filemerge: add a 'leave unresolved' option to change/delete prompts...
r27163 "use (c)hanged version, (d)elete, or leave (u)nresolved?"
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 "$$ &Changed $$ &Delete $$ &Unresolved") % prompts, 2)
Siddharth Agarwal
filemerge: add a 'leave unresolved' option to change/delete prompts...
r27163 choice = ['local', 'other', 'unresolved'][index]
Siddharth Agarwal
filemerge: add support for change/delete conflicts to the ':prompt' tool...
r27038 elif fcd.isabsent():
index = ui.promptchoice(
Simon Farnsworth
merge: always use other, not remote, in user prompts...
r29775 _("other%(o)s changed %(fd)s which local%(l)s deleted\n"
Siddharth Agarwal
filemerge: add a 'leave unresolved' option to change/delete prompts...
r27163 "use (c)hanged version, leave (d)eleted, or "
"leave (u)nresolved?"
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 "$$ &Changed $$ &Deleted $$ &Unresolved") % prompts, 2)
Siddharth Agarwal
filemerge: add a 'leave unresolved' option to change/delete prompts...
r27163 choice = ['other', 'local', 'unresolved'][index]
Siddharth Agarwal
filemerge: add support for change/delete conflicts to the ':prompt' tool...
r27038 else:
Siddharth Agarwal
filemerge: add a 'leave unresolved' option to regular prompts...
r27162 index = ui.promptchoice(
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 _("no tool found to merge %(fd)s\n"
"keep (l)ocal%(l)s, take (o)ther%(o)s, or leave (u)nresolved?"
"$$ &Local $$ &Other $$ &Unresolved") % prompts, 2)
Siddharth Agarwal
filemerge: add a 'leave unresolved' option to regular prompts...
r27162 choice = ['local', 'other', 'unresolved'][index]
Siddharth Agarwal
filemerge.prompt: separate out choice selection and action...
r26851
Siddharth Agarwal
filemerge: treat EOF at prompt as fail, not abort...
r26898 if choice == 'other':
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 return _iother(repo, mynode, orig, fcd, fco, fca, toolconf,
labels)
Siddharth Agarwal
filemerge: add a 'leave unresolved' option to regular prompts...
r27162 elif choice == 'local':
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 return _ilocal(repo, mynode, orig, fcd, fco, fca, toolconf,
labels)
Siddharth Agarwal
filemerge: add a 'leave unresolved' option to regular prompts...
r27162 elif choice == 'unresolved':
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 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:
ui.write("\n")
Simon Farnsworth
merge: use labels in prompts to the user...
r29774 return _ifail(repo, mynode, orig, fcd, fco, fca, toolconf,
labels)
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('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
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('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
repo.wvfs.unlinkpath(fcd.path())
deleted = True
else:
repo.wwrite(fcd.path(), fco.data(), fco.flags())
deleted = False
return 0, deleted
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('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():
repo.wwrite(fcd.path(), 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
Siddharth Agarwal
filemerge: don't attempt to premerge change/delete conflicts...
r27041 def _premerge(repo, fcd, fco, fca, toolconf, files, labels=None):
FUJIWARA Katsunori
filemerge: refactoring of 'filemerge()'...
r16125 tool, toolpath, binary, symlink = 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
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: rename _symlinkcheck to _mergecheck...
r26948 def _mergecheck(repo, mynode, orig, fcd, fco, fca, toolconf):
Siddharth Agarwal
filemerge: add a precheck for symlinks...
r26515 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
Siddharth Agarwal
filemerge._mergecheck: add check for change/delete conflicts...
r27040 if fcd.isabsent() or fco.isabsent():
repo.ui.warn(_('warning: internal %s cannot merge change/delete '
'conflict for %s\n') % (tool, fcd.path()))
return False
Siddharth Agarwal
filemerge: add a precheck for symlinks...
r26515 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)
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
Siddharth Agarwal
filemerge: switch trymerge boolean to mergetype enum...
r26526 @internaltool('union', fullmerge,
Siddharth Agarwal
simplemerge: move conflict warning message to filemerge...
r26614 _("warning: conflicts while merging %s! "
"(edit, then use 'hg resolve --mark')\n"),
Siddharth Agarwal
filemerge: rename _symlinkcheck to _mergecheck...
r26948 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."""
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,
Siddharth Agarwal
simplemerge: move conflict warning message to filemerge...
r26614 _("warning: conflicts while merging %s! "
"(edit, then use 'hg resolve --mark')\n"),
Siddharth Agarwal
filemerge: rename _symlinkcheck to _mergecheck...
r26948 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."""
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,
Siddharth Agarwal
simplemerge: move conflict warning message to filemerge...
r26614 _("warning: conflicts while merging %s! "
"(edit, then use 'hg resolve --mark')\n"),
Siddharth Agarwal
filemerge: rename _symlinkcheck to _mergecheck...
r26948 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:
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
a, b, c, back = files
r = simplemerge.simplemerge(repo.ui, a, b, c, label=labels,
localorother=localorother)
return True, r
Siddharth Agarwal
filemerge: rename _symlinkcheck to _mergecheck...
r26948 @internaltool('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."""
Jordi Gutiérrez Hermoso
filemerge: add non-interactive :merge-local and :merge-other...
r26224 success, status = _imergeauto(localorother='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
Siddharth Agarwal
filemerge: rename _symlinkcheck to _mergecheck...
r26948 @internaltool('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."""
Jordi Gutiérrez Hermoso
filemerge: add non-interactive :merge-local and :merge-other...
r26224 success, status = _imergeauto(localorother='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
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).
"""
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
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())
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
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
Siddharth Agarwal
filemerge: don't try using external tools on change/delete conflicts...
r27042 if fcd.isabsent() or fco.isabsent():
repo.ui.warn(_('warning: %s cannot merge change/delete conflict '
'for %s\n') % (tool, fcd.path()))
return False, 1, None
Siddharth Agarwal
filemerge._xmerge: drop no longer necessary 'if r:' check...
r26574 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)
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
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
Yuya Nishihara
merge: concatenate default conflict marker at parsing phase of .py...
r29660 _defaultconflictmarker = ('{node|short} '
Kostia Balytskyi
conflicts: make spacing consistent in conflict markers...
r30460 '{ifeq(tags, "tip", "", '
'ifeq(tags, "", "", "{tags} "))}'
Yuya Nishihara
merge: concatenate default conflict marker at parsing phase of .py...
r29660 '{if(bookmarks, "{bookmarks} ")}'
'{ifeq(branch, "default", "", "{branch} ")}'
'- {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)
Yuya Nishihara
templater: factor out function that creates templater from string template...
r28955 tmpl = formatter.maketemplater(ui, '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
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 {
"l": "",
"o": "",
}
return {
"l": " [%s]" % labels[0],
"o": " [%s]" % labels[1],
}
Siddharth Agarwal
filemerge: introduce a premerge flag and function...
r26607 def _filemerge(premerge, 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
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
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 def temp(prefix, ctx):
Mads Kiilerich
merge: use original file extension for temporary files...
r30538 fullbase, ext = os.path.splitext(ctx.path())
pre = "%s~%s." % (os.path.basename(fullbase), prefix)
(fd, name) = tempfile.mkstemp(prefix=pre, suffix=ext)
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 data = repo.wwritedata(ctx.path(), ctx.data())
f = os.fdopen(fd, "wb")
f.write(data)
f.close()
return name
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 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()
binary = fcd.isbinary() or fco.isbinary() or fca.isbinary()
symlink = '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)
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 if tool in internals and tool.startswith('internal:'):
# normalize to new-style names (':merge' etc)
tool = tool[len('internal'):]
Siddharth Agarwal
filemerge: add debug output for whether this is a change/delete conflict...
r27161 ui.debug("picked tool '%s' for %s (binary %s symlink %s changedelete %s)\n"
% (tool, fd, binary, symlink, 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
else:
func = _xmerge
mergetype = fullmerge
onfailure = _("merging %s failed!\n")
precheck = None
Siddharth Agarwal
filemerge: indent filemerge.filemerge...
r26512
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 toolconf = tool, toolpath, binary, symlink
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():
ui.status(_("merging %s and %s to %s\n") % (orig, fco.path(), fd))
else:
ui.status(_("merging %s\n") % fd)
Siddharth Agarwal
filemerge: move 'merging' output to before file creation...
r26528
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 ui.debug("my %s other %s ancestor %s\n" % (fcd, fco, fca))
Siddharth Agarwal
filemerge: move 'merging' output to before file creation...
r26528
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 if precheck and not precheck(repo, mynode, orig, fcd, fco, fca,
toolconf):
if onfailure:
ui.warn(onfailure % fd)
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
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 a = repo.wjoin(fd)
b = temp("base", fca)
c = temp("other", fco)
Siddharth Agarwal
filemerge: don't try to copy files known to be absent...
r27047 if not fcd.isabsent():
Siddharth Agarwal
origpath: move from cmdutil to scmutil...
r27651 back = scmutil.origpath(ui, repo, a)
Siddharth Agarwal
filemerge: don't try to copy files known to be absent...
r27047 if premerge:
util.copyfile(a, back)
else:
back = None
Siddharth Agarwal
filemerge: deindent the parts of filemerge outside the try block...
r26608 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: introduce a premerge flag and function...
r26607 if premerge and mergetype == fullmerge:
Siddharth Agarwal
filemerge: don't attempt to premerge change/delete conflicts...
r27041 r = _premerge(repo, fcd, fco, fca, toolconf, files, labels=labels)
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
Siddharth Agarwal
filemerge: return whether the file is deleted from all other merge tools...
r27033 needcheck, r, deleted = func(repo, mynode, orig, fcd, fco, fca,
toolconf, files, labels=labels)
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
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:
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")):
Siddharth Agarwal
filemerge: don't try to copy files known to be absent...
r27047 if back is not None and filecmp.cmp(a, back):
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 if ui.promptchoice(_(" output file %s appears unchanged\n"
"was merge successful (yn)?"
"$$ &Yes $$ &No") % fd, 1):
r = 1
Siddharth Agarwal
filemerge: don't try to copy files known to be absent...
r27047 if back is not None and _toolbool(ui, tool, "fixeol"):
Siddharth Agarwal
filemerge: move post-merge checks into a separate function...
r26575 _matcheol(a, back)
return r
Siddharth Agarwal
filemerge: introduce a premerge flag and function...
r26607 def premerge(repo, mynode, orig, fcd, fco, fca, labels=None):
return _filemerge(True, repo, mynode, orig, fcd, fco, fca, labels=labels)
Siddharth Agarwal
filemerge: add a wrapper around the filemerge function...
r26605 def filemerge(repo, mynode, orig, fcd, fco, fca, labels=None):
Siddharth Agarwal
filemerge: break overall filemerge into separate premerge and merge steps...
r26611 return _filemerge(False, repo, mynode, orig, fcd, fco, fca, labels=labels)
Siddharth Agarwal
filemerge: add a wrapper around the filemerge function...
r26605
FUJIWARA Katsunori
filemerge: create detail of internal merge tools from documentation string...
r16126 # tell hggettext to extract docstrings from these functions:
i18nfunctions = internals.values()