##// END OF EJS Templates
simplemerge: use 3-way markers if mode=='merge3', ignoring number of labels...
simplemerge: use 3-way markers if mode=='merge3', ignoring number of labels Before this patch, we use a combination of `mode` and `labels` to decide which marker style to use. That's mostly my own fault because I was the one who added the `mergediff` as a "mode" instead of adding a separate argument for marker style. Still, let's make these arguments more orthogonal by ignoring the number of labels when deciding which marker style to use. This refactoring makes it a little clearer in the simplemerge *extension* that passing more than 2 `--label` arguments changes the marker style. Differential Revision: https://phab.mercurial-scm.org/D11972

File last commit:

r49398:60a06b0d default
r49404:c9141848 default
Show More
simplemerge.py
527 lines | 16.8 KiB | text/x-python | PythonLexer
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 # Copyright (C) 2004, 2005 Canonical Ltd
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
Martin Geisler
Remove FSF mailing address from GPL headers...
r15782 # along with this program; if not, see <http://www.gnu.org/licenses/>.
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
# mbp: "you know that thing where cvs gives you conflict markers?"
# s: "i hate that."
Gregory Szorc
simplemerge: use absolute_import
r25974 from __future__ import absolute_import
from .i18n import _
from . import (
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 error,
Gregory Szorc
simplemerge: use absolute_import
r25974 mdiff,
Pulkit Goyal
py3: convert kwargs' keys' to str using pycompat.strkwargs()...
r33017 pycompat,
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 )
Augie Fackler
formatting: blacken the codebase...
r43346 from .utils import stringutil
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
class CantReprocessAndShowBase(Exception):
pass
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 def intersect(ra, rb):
"""Given two ranges return the range where they intersect or None.
>>> intersect((0, 10), (0, 6))
(0, 6)
>>> intersect((0, 10), (5, 15))
(5, 10)
>>> intersect((0, 10), (10, 15))
>>> intersect((0, 9), (10, 15))
>>> intersect((0, 9), (7, 15))
(7, 9)
"""
assert ra[0] <= ra[1]
assert rb[0] <= rb[1]
sa = max(ra[0], rb[0])
sb = min(ra[1], rb[1])
if sa < sb:
return sa, sb
else:
return None
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 def compare_range(a, astart, aend, b, bstart, bend):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Compare a[astart:aend] == b[bstart:bend], without slicing."""
Matt Mackall
many, many trivial check-code fixups
r10282 if (aend - astart) != (bend - bstart):
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 return False
Augie Fackler
formatting: blacken the codebase...
r43346 for ia, ib in zip(
pycompat.xrange(astart, aend), pycompat.xrange(bstart, bend)
):
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 if a[ia] != b[ib]:
return False
else:
return True
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 class Merge3Text(object):
"""3-way merge of texts.
Given strings BASE, OTHER, THIS, tries to produce a combined text
incorporating the changes from both BASE->OTHER and BASE->THIS."""
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 def __init__(self, basetext, atext, btext, base=None, a=None, b=None):
self.basetext = basetext
self.atext = atext
self.btext = btext
if base is None:
base = mdiff.splitnewlines(basetext)
if a is None:
a = mdiff.splitnewlines(atext)
if b is None:
b = mdiff.splitnewlines(btext)
self.base = base
self.a = a
self.b = b
Augie Fackler
formatting: blacken the codebase...
r43346 def merge_lines(
self,
name_a=None,
name_b=None,
name_base=None,
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 start_marker=b'<<<<<<<',
mid_marker=b'=======',
end_marker=b'>>>>>>>',
Augie Fackler
formatting: blacken the codebase...
r43346 base_marker=None,
minimize=False,
):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Return merge in cvs-like form."""
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 self.conflicts = False
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 newline = b'\n'
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 if len(self.a) > 0:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if self.a[0].endswith(b'\r\n'):
newline = b'\r\n'
elif self.a[0].endswith(b'\r'):
newline = b'\r'
Erik Huelsmann
simplemerge: add 'mode' kwarg which - when 'union' - suppresses markers...
r26069 if name_a and start_marker:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 start_marker = start_marker + b' ' + name_a
Erik Huelsmann
simplemerge: add 'mode' kwarg which - when 'union' - suppresses markers...
r26069 if name_b and end_marker:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 end_marker = end_marker + b' ' + name_b
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 if name_base and base_marker:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 base_marker = base_marker + b' ' + name_base
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 merge_groups = self.merge_groups()
Ryan McElroy
merge: minimize conflicts when common base is not shown (issue4447)...
r28072 if minimize:
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 merge_groups = self.minimize(merge_groups)
for what, lines in merge_groups:
if what == b'conflict':
base_lines, a_lines, b_lines = lines
Martin von Zweigbergk
simplemerge: remove now-unused `localorother` argument from `merge_lines()`...
r49346 self.conflicts = True
if start_marker is not None:
yield start_marker + newline
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 for line in a_lines:
yield line
Martin von Zweigbergk
simplemerge: remove now-unused `localorother` argument from `merge_lines()`...
r49346 if base_marker is not None:
yield base_marker + newline
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 for line in base_lines:
yield line
Martin von Zweigbergk
simplemerge: remove now-unused `localorother` argument from `merge_lines()`...
r49346 if mid_marker is not None:
yield mid_marker + newline
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 for line in b_lines:
yield line
Martin von Zweigbergk
simplemerge: remove now-unused `localorother` argument from `merge_lines()`...
r49346 if end_marker is not None:
yield end_marker + newline
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 else:
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 for line in lines:
yield line
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
def merge_groups(self):
"""Yield sequence of line groups. Each one is a tuple:
'unchanged', lines
Lines unchanged from base
'a', lines
Lines taken from a
'same', lines
Lines taken from a (and equal to b)
'b', lines
Lines taken from b
Martin von Zweigbergk
simplemerge: make merge_groups() yield only 2-tuples...
r49379 'conflict', (base_lines, a_lines, b_lines)
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 Lines from base were changed to either a or b and conflict.
"""
for t in self.merge_regions():
what = t[0]
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if what == b'unchanged':
Augie Fackler
formatting: blacken the codebase...
r43346 yield what, self.base[t[1] : t[2]]
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif what == b'a' or what == b'same':
Augie Fackler
formatting: blacken the codebase...
r43346 yield what, self.a[t[1] : t[2]]
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif what == b'b':
Augie Fackler
formatting: blacken the codebase...
r43346 yield what, self.b[t[1] : t[2]]
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif what == b'conflict':
Augie Fackler
formatting: blacken the codebase...
r43346 yield (
what,
Martin von Zweigbergk
simplemerge: make merge_groups() yield only 2-tuples...
r49379 (
self.base[t[1] : t[2]],
self.a[t[3] : t[4]],
self.b[t[5] : t[6]],
),
Augie Fackler
formatting: blacken the codebase...
r43346 )
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 else:
raise ValueError(what)
def merge_regions(self):
"""Return sequences of matching and conflicting regions.
This returns tuples, where the first value says what kind we
have:
'unchanged', start, end
Take a region of base[start:end]
'same', astart, aend
b and a are different from base but give the same result
'a', start, end
Non-clashing insertion from a[start:end]
Ryan McElroy
merge: add some useful documentation
r28070 'conflict', zstart, zend, astart, aend, bstart, bend
Conflict between a and b, with z as common ancestor
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 Method is as follows:
The two sequences align only on regions which match the base
Matt Mackall
check-code: catch misspellings of descendant...
r14549 and both descendants. These are found by doing a two-way diff
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 of each one against the base, and then finding the
intersections between those regions. These "sync regions"
are by definition unchanged in both and easily dealt with.
The regions in between can be in any of three cases:
conflicted, or changed on only one side.
"""
# section a[0:ia] has been disposed of, etc
iz = ia = ib = 0
Brodie Rao
cleanup: eradicate long lines
r16683 for region in self.find_sync_regions():
zmatch, zend, amatch, aend, bmatch, bend = region
Augie Fackler
formatting: blacken the codebase...
r43346 # print 'match base [%d:%d]' % (zmatch, zend)
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
matchlen = zend - zmatch
assert matchlen >= 0
assert matchlen == (aend - amatch)
assert matchlen == (bend - bmatch)
len_a = amatch - ia
len_b = bmatch - ib
len_base = zmatch - iz
assert len_a >= 0
assert len_b >= 0
assert len_base >= 0
Augie Fackler
formatting: blacken the codebase...
r43346 # print 'unmatched a=%d, b=%d' % (len_a, len_b)
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
if len_a or len_b:
# try to avoid actually slicing the lists
Augie Fackler
formatting: blacken the codebase...
r43346 equal_a = compare_range(
self.a, ia, amatch, self.base, iz, zmatch
)
equal_b = compare_range(
self.b, ib, bmatch, self.base, iz, zmatch
)
same = compare_range(self.a, ia, amatch, self.b, ib, bmatch)
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
if same:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 yield b'same', ia, amatch
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 elif equal_a and not equal_b:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 yield b'b', ib, bmatch
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 elif equal_b and not equal_a:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 yield b'a', ia, amatch
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 elif not equal_a and not equal_b:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 yield b'conflict', iz, zmatch, ia, amatch, ib, bmatch
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 raise AssertionError(b"can't handle a=b=base but unmatched")
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
ia = amatch
ib = bmatch
iz = zmatch
# if the same part of the base was deleted on both sides
# that's OK, we can just skip it.
if matchlen > 0:
assert ia == amatch
assert ib == bmatch
assert iz == zmatch
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 yield b'unchanged', zmatch, zend
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 iz = zend
ia = aend
ib = bend
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 def minimize(self, merge_groups):
Ryan McElroy
merge: introduce method to minimize merge regions...
r28071 """Trim conflict regions of lines where A and B sides match.
Mads Kiilerich
spelling: fixes of non-dictionary words
r30332 Lines where both A and B have made the same changes at the beginning
Ryan McElroy
merge: introduce method to minimize merge regions...
r28071 or the end of each merge region are eliminated from the conflict
region and are instead considered the same.
"""
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 for what, lines in merge_groups:
if what != b"conflict":
yield what, lines
Ryan McElroy
merge: introduce method to minimize merge regions...
r28071 continue
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 base_lines, a_lines, b_lines = lines
alen = len(a_lines)
blen = len(b_lines)
Ryan McElroy
merge: introduce method to minimize merge regions...
r28071
# find matches at the front
ii = 0
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 while ii < alen and ii < blen and a_lines[ii] == b_lines[ii]:
Ryan McElroy
merge: introduce method to minimize merge regions...
r28071 ii += 1
startmatches = ii
# find matches at the end
ii = 0
Augie Fackler
formatting: blacken the codebase...
r43346 while (
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 ii < alen and ii < blen and a_lines[-ii - 1] == b_lines[-ii - 1]
Augie Fackler
formatting: blacken the codebase...
r43346 ):
Ryan McElroy
merge: introduce method to minimize merge regions...
r28071 ii += 1
endmatches = ii
if startmatches > 0:
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 yield b'same', a_lines[:startmatches]
Ryan McElroy
merge: introduce method to minimize merge regions...
r28071
Augie Fackler
formatting: blacken the codebase...
r43346 yield (
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'conflict',
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 (
base_lines,
a_lines[startmatches : alen - endmatches],
b_lines[startmatches : blen - endmatches],
),
Augie Fackler
formatting: blacken the codebase...
r43346 )
Ryan McElroy
merge: introduce method to minimize merge regions...
r28071
if endmatches > 0:
Martin von Zweigbergk
simplemerge: rewrite `merge_lines()` using `merge_groups()`...
r49380 yield b'same', a_lines[alen - endmatches :]
Ryan McElroy
merge: introduce method to minimize merge regions...
r28071
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 def find_sync_regions(self):
Matt Mackall
check-code: catch misspellings of descendant...
r14549 """Return a list of sync regions, where both descendants match the base.
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
Generates a list of (base1, base2, a1, a2, b1, b2). There is
always a zero-length sync region at the end of all the files.
"""
ia = ib = 0
amatches = mdiff.get_matching_blocks(self.basetext, self.atext)
bmatches = mdiff.get_matching_blocks(self.basetext, self.btext)
len_a = len(amatches)
len_b = len(bmatches)
sl = []
while ia < len_a and ib < len_b:
abase, amatch, alen = amatches[ia]
bbase, bmatch, blen = bmatches[ib]
# there is an unconflicted block at i; how long does it
# extend? until whichever one ends earlier.
Matt Mackall
many, many trivial check-code fixups
r10282 i = intersect((abase, abase + alen), (bbase, bbase + blen))
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 if i:
intbase = i[0]
intend = i[1]
intlen = intend - intbase
# found a match of base[i[0], i[1]]; this may be less than
# the region that matches in either one
assert intlen <= alen
assert intlen <= blen
assert abase <= intbase
assert bbase <= intbase
asub = amatch + (intbase - abase)
bsub = bmatch + (intbase - bbase)
aend = asub + intlen
bend = bsub + intlen
Augie Fackler
cleanup: use () to wrap long lines instead of \...
r41925 assert self.base[intbase:intend] == self.a[asub:aend], (
Augie Fackler
formatting: blacken the codebase...
r43346 self.base[intbase:intend],
self.a[asub:aend],
)
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
assert self.base[intbase:intend] == self.b[bsub:bend]
Augie Fackler
formatting: blacken the codebase...
r43346 sl.append((intbase, intend, asub, aend, bsub, bend))
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
# advance whichever one ends first in the base text
if (abase + alen) < (bbase + blen):
ia += 1
else:
ib += 1
intbase = len(self.base)
abase = len(self.a)
bbase = len(self.b)
sl.append((intbase, intbase, abase, abase, bbase, bbase))
return sl
Augie Fackler
formatting: blacken the codebase...
r43346
Phil Cohen
simplemerge: extract verifytext as a helper function...
r33824 def _verifytext(text, path, ui, opts):
"""verifies that text is non-binary (unless opts[text] is passed,
then we just warn)"""
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 if stringutil.binary(text):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 msg = _(b"%s looks like a binary file.") % path
Martin von Zweigbergk
simplemerge: work with opts as native strings instead of bytes...
r46714 if not opts.get('quiet'):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ui.warn(_(b'warning: %s\n') % msg)
Martin von Zweigbergk
simplemerge: work with opts as native strings instead of bytes...
r46714 if not opts.get('text'):
Phil Cohen
simplemerge: extract verifytext as a helper function...
r33824 raise error.Abort(msg)
return text
Augie Fackler
formatting: blacken the codebase...
r43346
Phil Cohen
filemerge: extract `_picklabels` as a helper function...
r33829 def _picklabels(defaults, overrides):
if len(overrides) > 3:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 raise error.Abort(_(b"can only specify three labels."))
Phil Cohen
simplemerge: refactor _picklabels to be more compact...
r33934 result = defaults[:]
for i, override in enumerate(overrides):
result[i] = override
return result
Phil Cohen
filemerge: extract `_picklabels` as a helper function...
r33829
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
mergetools: add new conflict marker format with diffs in...
r46724 def _mergediff(m3, name_a, name_b, name_base):
lines = []
conflicts = False
Martin von Zweigbergk
simplemerge: make merge_groups() yield only 2-tuples...
r49379 for what, group_lines in m3.merge_groups():
if what == b'conflict':
base_lines, a_lines, b_lines = group_lines
Martin von Zweigbergk
mergetools: add new conflict marker format with diffs in...
r46724 base_text = b''.join(base_lines)
b_blocks = list(
mdiff.allblocks(
base_text,
b''.join(b_lines),
lines1=base_lines,
lines2=b_lines,
)
)
a_blocks = list(
mdiff.allblocks(
base_text,
b''.join(a_lines),
lines1=base_lines,
lines2=b_lines,
)
)
def matching_lines(blocks):
return sum(
block[1] - block[0]
for block, kind in blocks
if kind == b'='
)
def diff_lines(blocks, lines1, lines2):
for block, kind in blocks:
if kind == b'=':
for line in lines1[block[0] : block[1]]:
yield b' ' + line
else:
for line in lines1[block[0] : block[1]]:
yield b'-' + line
for line in lines2[block[2] : block[3]]:
yield b'+' + line
lines.append(b"<<<<<<<\n")
if matching_lines(a_blocks) < matching_lines(b_blocks):
lines.append(b"======= %s\n" % name_a)
lines.extend(a_lines)
lines.append(b"------- %s\n" % name_base)
lines.append(b"+++++++ %s\n" % name_b)
lines.extend(diff_lines(b_blocks, base_lines, b_lines))
else:
lines.append(b"------- %s\n" % name_base)
lines.append(b"+++++++ %s\n" % name_a)
lines.extend(diff_lines(a_blocks, base_lines, a_lines))
lines.append(b"======= %s\n" % name_b)
lines.extend(b_lines)
lines.append(b">>>>>>>\n")
conflicts = True
else:
Martin von Zweigbergk
simplemerge: make merge_groups() yield only 2-tuples...
r49379 lines.extend(group_lines)
Martin von Zweigbergk
mergetools: add new conflict marker format with diffs in...
r46724 return lines, conflicts
Martin von Zweigbergk
simplemerge: add a specialized function for "union", "local", "other"...
r49345 def _resolve(m3, sides):
lines = []
Martin von Zweigbergk
simplemerge: make merge_groups() yield only 2-tuples...
r49379 for what, group_lines in m3.merge_groups():
if what == b'conflict':
Martin von Zweigbergk
simplemerge: add a specialized function for "union", "local", "other"...
r49345 for side in sides:
Martin von Zweigbergk
simplemerge: make merge_groups() yield only 2-tuples...
r49379 lines.extend(group_lines[side])
Martin von Zweigbergk
simplemerge: add a specialized function for "union", "local", "other"...
r49345 else:
Martin von Zweigbergk
simplemerge: make merge_groups() yield only 2-tuples...
r49379 lines.extend(group_lines)
Martin von Zweigbergk
simplemerge: add a specialized function for "union", "local", "other"...
r49345 return lines
Phil Cohen
simplemerge: remove unused `repo` parameter...
r34051 def simplemerge(ui, localctx, basectx, otherctx, **opts):
Phil Cohen
simplemerge: use contexts to read file data from, if passed...
r33827 """Performs the simplemerge algorithm.
Phil Cohen
simplemerge: make context parameters non-optional...
r33908 The merged result is written into `localctx`.
Phil Cohen
simplemerge: stop reading from, and writing to, files...
r33906 """
Pulkit Goyal
py3: handle keyword arguments correctly in simplemerge.py...
r35369
Phil Cohen
simplemerge: use contexts to read file data from, if passed...
r33827 def readctx(ctx):
Phil Cohen
simplemerge: use `ctx.decoddeddata()` instead of `repo.wreaddata`...
r33903 # Merges were always run in the working copy before, which means
# they used decoded data, if the user defined any repository
# filters.
#
# Maintain that behavior today for BC, though perhaps in the future
# it'd be worth considering whether merging encoded data (what the
# repository usually sees) might be more useful.
return _verifytext(ctx.decodeddata(), ctx.path(), ui, opts)
Phil Cohen
simplemerge: use contexts to read file data from, if passed...
r33827
Steve Borho
simplemerge: do not allow binary files to abort an entire merge...
r14328 try:
Phil Cohen
simplemerge: stop reading from, and writing to, files...
r33906 localtext = readctx(localctx)
basetext = readctx(basectx)
othertext = readctx(otherctx)
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 except error.Abort:
Steve Borho
simplemerge: do not allow binary files to abort an entire merge...
r14328 return 1
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
m3 = Merge3Text(basetext, localtext, othertext)
Martin von Zweigbergk
simplemerge: add a specialized function for "union", "local", "other"...
r49345 conflicts = False
Martin von Zweigbergk
simplemerge: don't calculate conflict labels when resolving automatically...
r49347 mode = opts.get('mode', b'merge')
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if mode == b'union':
Martin von Zweigbergk
simplemerge: add a specialized function for "union", "local", "other"...
r49345 lines = _resolve(m3, (1, 2))
Martin von Zweigbergk
simplemerge: make `localorother` a "mode" instead of a separate thing...
r49340 elif mode == b'local':
Martin von Zweigbergk
simplemerge: add a specialized function for "union", "local", "other"...
r49345 lines = _resolve(m3, (1,))
Martin von Zweigbergk
simplemerge: make `localorother` a "mode" instead of a separate thing...
r49340 elif mode == b'other':
Martin von Zweigbergk
simplemerge: add a specialized function for "union", "local", "other"...
r49345 lines = _resolve(m3, (2,))
Martin von Zweigbergk
mergetools: add new conflict marker format with diffs in...
r46724 else:
Martin von Zweigbergk
simplemerge: don't calculate conflict labels when resolving automatically...
r49347 name_a, name_b, name_base = _picklabels(
[localctx.path(), otherctx.path(), None], opts.get('label', [])
Martin von Zweigbergk
mergetools: add new conflict marker format with diffs in...
r46724 )
Martin von Zweigbergk
simplemerge: don't calculate conflict labels when resolving automatically...
r49347 if mode == b'mergediff':
lines, conflicts = _mergediff(m3, name_a, name_b, name_base)
else:
extrakwargs = {
'minimize': True,
}
Martin von Zweigbergk
simplemerge: use 3-way markers if mode=='merge3', ignoring number of labels...
r49404 if mode == b'merge3':
Martin von Zweigbergk
simplemerge: don't calculate conflict labels when resolving automatically...
r49347 extrakwargs['base_marker'] = b'|||||||'
extrakwargs['name_base'] = name_base
extrakwargs['minimize'] = False
lines = list(
m3.merge_lines(name_a=name_a, name_b=name_b, **extrakwargs)
)
conflicts = m3.conflicts
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
Martin von Zweigbergk
simplemerge: write output only once it's complete...
r46716 mergedtext = b''.join(lines)
if opts.get('print'):
ui.fout.write(mergedtext)
else:
Martin von Zweigbergk
simplemerge: stop merging file flags...
r49338 # localctx.flags() already has the merged flags (done in
# mergestate.resolve())
localctx.write(mergedtext, localctx.flags())
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002
Martin von Zweigbergk
simplemerge: set `conflicts` earlier for :union tool...
r49254 if conflicts:
Matt Mackall
merge: move the bulk of simplemerge into core...
r6002 return 1