|
|
|
#!/bin/sh
|
|
|
|
|
|
|
|
test_description='RCS merge replacement: merge-file'
|
|
|
|
. ./test-lib.sh
|
|
|
|
|
|
|
|
cat > orig.txt << EOF
|
|
|
|
Dominus regit me,
|
|
|
|
et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
EOF
|
|
|
|
|
|
|
|
cat > new1.txt << EOF
|
|
|
|
Dominus regit me,
|
|
|
|
et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
Nam et si ambulavero in medio umbrae mortis,
|
|
|
|
non timebo mala, quoniam tu mecum es:
|
|
|
|
virga tua et baculus tuus ipsa me consolata sunt.
|
|
|
|
EOF
|
|
|
|
|
|
|
|
cat > new2.txt << EOF
|
|
|
|
Dominus regit me, et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
EOF
|
|
|
|
|
|
|
|
cat > new3.txt << EOF
|
|
|
|
DOMINUS regit me,
|
|
|
|
et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
EOF
|
|
|
|
|
|
|
|
cat > new4.txt << EOF
|
|
|
|
Dominus regit me, et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
EOF
|
|
|
|
printf "propter nomen suum." >> new4.txt
|
|
|
|
|
|
|
|
test_expect_success 'merge with no changes' '
|
|
|
|
cp orig.txt test.txt &&
|
|
|
|
git merge-file test.txt orig.txt orig.txt &&
|
|
|
|
test_cmp test.txt orig.txt
|
|
|
|
'
|
|
|
|
|
|
|
|
cp new1.txt test.txt
|
|
|
|
test_expect_success "merge without conflict" \
|
|
|
|
"git merge-file test.txt orig.txt new2.txt"
|
|
|
|
|
|
|
|
test_expect_success 'works in subdirectory' '
|
|
|
|
mkdir dir &&
|
|
|
|
cp new1.txt dir/a.txt &&
|
|
|
|
cp orig.txt dir/o.txt &&
|
|
|
|
cp new2.txt dir/b.txt &&
|
|
|
|
( cd dir && git merge-file a.txt o.txt b.txt ) &&
|
|
|
|
test_path_is_missing a.txt
|
|
|
|
'
|
|
|
|
|
|
|
|
cp new1.txt test.txt
|
|
|
|
test_expect_success "merge without conflict (--quiet)" \
|
|
|
|
"git merge-file --quiet test.txt orig.txt new2.txt"
|
|
|
|
|
|
|
|
cp new1.txt test2.txt
|
|
|
|
test_expect_failure "merge without conflict (missing LF at EOF)" \
|
|
|
|
"git merge-file test2.txt orig.txt new4.txt"
|
|
|
|
|
|
|
|
test_expect_failure "merge result added missing LF" \
|
|
|
|
"test_cmp test.txt test2.txt"
|
|
|
|
|
|
|
|
cp new4.txt test3.txt
|
|
|
|
test_expect_success "merge without conflict (missing LF at EOF, away from change in the other file)" \
|
|
|
|
"git merge-file --quiet test3.txt new2.txt new3.txt"
|
|
|
|
|
|
|
|
cat > expect.txt << EOF
|
|
|
|
DOMINUS regit me,
|
|
|
|
et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
EOF
|
|
|
|
printf "propter nomen suum." >> expect.txt
|
|
|
|
|
|
|
|
test_expect_success "merge does not add LF away of change" \
|
|
|
|
"test_cmp test3.txt expect.txt"
|
|
|
|
|
|
|
|
cp test.txt backup.txt
|
|
|
|
test_expect_success "merge with conflicts" \
|
|
|
|
"test_must_fail git merge-file test.txt orig.txt new3.txt"
|
|
|
|
|
|
|
|
cat > expect.txt << EOF
|
|
|
|
<<<<<<< test.txt
|
|
|
|
Dominus regit me, et nihil mihi deerit.
|
|
|
|
=======
|
|
|
|
DOMINUS regit me,
|
|
|
|
et nihil mihi deerit.
|
|
|
|
>>>>>>> new3.txt
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
Nam et si ambulavero in medio umbrae mortis,
|
|
|
|
non timebo mala, quoniam tu mecum es:
|
|
|
|
virga tua et baculus tuus ipsa me consolata sunt.
|
|
|
|
EOF
|
|
|
|
|
|
|
|
test_expect_success "expected conflict markers" "test_cmp test.txt expect.txt"
|
|
|
|
|
|
|
|
cp backup.txt test.txt
|
|
|
|
|
|
|
|
cat > expect.txt << EOF
|
|
|
|
Dominus regit me, et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
Nam et si ambulavero in medio umbrae mortis,
|
|
|
|
non timebo mala, quoniam tu mecum es:
|
|
|
|
virga tua et baculus tuus ipsa me consolata sunt.
|
|
|
|
EOF
|
|
|
|
test_expect_success "merge conflicting with --ours" \
|
|
|
|
"git merge-file --ours test.txt orig.txt new3.txt && test_cmp test.txt expect.txt"
|
|
|
|
cp backup.txt test.txt
|
|
|
|
|
|
|
|
cat > expect.txt << EOF
|
|
|
|
DOMINUS regit me,
|
|
|
|
et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
Nam et si ambulavero in medio umbrae mortis,
|
|
|
|
non timebo mala, quoniam tu mecum es:
|
|
|
|
virga tua et baculus tuus ipsa me consolata sunt.
|
|
|
|
EOF
|
|
|
|
test_expect_success "merge conflicting with --theirs" \
|
|
|
|
"git merge-file --theirs test.txt orig.txt new3.txt && test_cmp test.txt expect.txt"
|
|
|
|
cp backup.txt test.txt
|
|
|
|
|
|
|
|
cat > expect.txt << EOF
|
|
|
|
Dominus regit me, et nihil mihi deerit.
|
|
|
|
DOMINUS regit me,
|
|
|
|
et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
Nam et si ambulavero in medio umbrae mortis,
|
|
|
|
non timebo mala, quoniam tu mecum es:
|
|
|
|
virga tua et baculus tuus ipsa me consolata sunt.
|
|
|
|
EOF
|
|
|
|
test_expect_success "merge conflicting with --union" \
|
|
|
|
"git merge-file --union test.txt orig.txt new3.txt && test_cmp test.txt expect.txt"
|
|
|
|
cp backup.txt test.txt
|
|
|
|
|
|
|
|
test_expect_success "merge with conflicts, using -L" \
|
|
|
|
"test_must_fail git merge-file -L 1 -L 2 test.txt orig.txt new3.txt"
|
|
|
|
|
|
|
|
cat > expect.txt << EOF
|
|
|
|
<<<<<<< 1
|
|
|
|
Dominus regit me, et nihil mihi deerit.
|
|
|
|
=======
|
|
|
|
DOMINUS regit me,
|
|
|
|
et nihil mihi deerit.
|
|
|
|
>>>>>>> new3.txt
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
Nam et si ambulavero in medio umbrae mortis,
|
|
|
|
non timebo mala, quoniam tu mecum es:
|
|
|
|
virga tua et baculus tuus ipsa me consolata sunt.
|
|
|
|
EOF
|
|
|
|
|
|
|
|
test_expect_success "expected conflict markers, with -L" \
|
|
|
|
"test_cmp test.txt expect.txt"
|
|
|
|
|
|
|
|
sed "s/ tu / TU /" < new1.txt > new5.txt
|
|
|
|
test_expect_success "conflict in removed tail" \
|
|
|
|
"test_must_fail git merge-file -p orig.txt new1.txt new5.txt > out"
|
|
|
|
|
|
|
|
cat > expect << EOF
|
|
|
|
Dominus regit me,
|
|
|
|
et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
<<<<<<< orig.txt
|
|
|
|
=======
|
|
|
|
Nam et si ambulavero in medio umbrae mortis,
|
|
|
|
non timebo mala, quoniam TU mecum es:
|
|
|
|
virga tua et baculus tuus ipsa me consolata sunt.
|
|
|
|
>>>>>>> new5.txt
|
|
|
|
EOF
|
|
|
|
|
|
|
|
test_expect_success "expected conflict markers" "test_cmp expect out"
|
|
|
|
|
|
|
|
test_expect_success 'binary files cannot be merged' '
|
|
|
|
test_must_fail git merge-file -p \
|
|
|
|
orig.txt "$TEST_DIRECTORY"/test-binary-1.png new1.txt 2> merge.err &&
|
|
|
|
grep "Cannot merge binary files" merge.err
|
|
|
|
'
|
|
|
|
|
|
|
|
sed -e "s/deerit.\$/deerit;/" -e "s/me;\$/me./" < new5.txt > new6.txt
|
|
|
|
sed -e "s/deerit.\$/deerit,/" -e "s/me;\$/me,/" < new5.txt > new7.txt
|
|
|
|
|
|
|
|
test_expect_success 'MERGE_ZEALOUS simplifies non-conflicts' '
|
|
|
|
|
|
|
|
test_must_fail git merge-file -p new6.txt new5.txt new7.txt > output &&
|
|
|
|
test 1 = $(grep ======= < output | wc -l)
|
|
|
|
|
|
|
|
'
|
|
|
|
|
|
|
|
sed -e 's/deerit./&%%%%/' -e "s/locavit,/locavit;/"< new6.txt | tr '%' '\012' > new8.txt
|
|
|
|
sed -e 's/deerit./&%%%%/' -e "s/locavit,/locavit --/" < new7.txt | tr '%' '\012' > new9.txt
|
|
|
|
|
|
|
|
test_expect_success 'ZEALOUS_ALNUM' '
|
|
|
|
|
|
|
|
test_must_fail git merge-file -p \
|
|
|
|
new8.txt new5.txt new9.txt > merge.out &&
|
|
|
|
test 1 = $(grep ======= < merge.out | wc -l)
|
|
|
|
|
|
|
|
'
|
|
|
|
|
|
|
|
cat >expect <<\EOF
|
|
|
|
Dominus regit me,
|
|
|
|
<<<<<<< new8.txt
|
|
|
|
et nihil mihi deerit;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
In loco pascuae ibi me collocavit;
|
|
|
|
super aquam refectionis educavit me.
|
|
|
|
||||||| new5.txt
|
|
|
|
et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
=======
|
|
|
|
et nihil mihi deerit,
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
In loco pascuae ibi me collocavit --
|
|
|
|
super aquam refectionis educavit me,
|
|
|
|
>>>>>>> new9.txt
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
Nam et si ambulavero in medio umbrae mortis,
|
|
|
|
non timebo mala, quoniam TU mecum es:
|
|
|
|
virga tua et baculus tuus ipsa me consolata sunt.
|
|
|
|
EOF
|
|
|
|
|
|
|
|
test_expect_success '"diff3 -m" style output (1)' '
|
|
|
|
test_must_fail git merge-file -p --diff3 \
|
|
|
|
new8.txt new5.txt new9.txt >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success '"diff3 -m" style output (2)' '
|
|
|
|
git config merge.conflictstyle diff3 &&
|
|
|
|
test_must_fail git merge-file -p \
|
|
|
|
new8.txt new5.txt new9.txt >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
|
|
|
cat >expect <<\EOF
|
|
|
|
Dominus regit me,
|
|
|
|
<<<<<<<<<< new8.txt
|
|
|
|
et nihil mihi deerit;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
In loco pascuae ibi me collocavit;
|
|
|
|
super aquam refectionis educavit me.
|
|
|
|
|||||||||| new5.txt
|
|
|
|
et nihil mihi deerit.
|
|
|
|
In loco pascuae ibi me collocavit,
|
|
|
|
super aquam refectionis educavit me;
|
|
|
|
==========
|
|
|
|
et nihil mihi deerit,
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
In loco pascuae ibi me collocavit --
|
|
|
|
super aquam refectionis educavit me,
|
|
|
|
>>>>>>>>>> new9.txt
|
|
|
|
animam meam convertit,
|
|
|
|
deduxit me super semitas jusitiae,
|
|
|
|
propter nomen suum.
|
|
|
|
Nam et si ambulavero in medio umbrae mortis,
|
|
|
|
non timebo mala, quoniam TU mecum es:
|
|
|
|
virga tua et baculus tuus ipsa me consolata sunt.
|
|
|
|
EOF
|
|
|
|
|
|
|
|
test_expect_success 'marker size' '
|
|
|
|
test_must_fail git merge-file -p --marker-size=10 \
|
|
|
|
new8.txt new5.txt new9.txt >actual &&
|
|
|
|
test_cmp expect actual
|
|
|
|
'
|
|
|
|
|
|
|
|
printf "line1\nline2\nline3" >nolf-orig.txt
|
|
|
|
printf "line1\nline2\nline3x" >nolf-diff1.txt
|
|
|
|
printf "line1\nline2\nline3y" >nolf-diff2.txt
|
|
|
|
|
|
|
|
test_expect_success 'conflict at EOF without LF resolved by --ours' \
|
|
|
|
'git merge-file -p --ours nolf-diff1.txt nolf-orig.txt nolf-diff2.txt >output.txt &&
|
|
|
|
printf "line1\nline2\nline3x" >expect.txt &&
|
|
|
|
test_cmp expect.txt output.txt'
|
|
|
|
|
|
|
|
test_expect_success 'conflict at EOF without LF resolved by --theirs' \
|
|
|
|
'git merge-file -p --theirs nolf-diff1.txt nolf-orig.txt nolf-diff2.txt >output.txt &&
|
|
|
|
printf "line1\nline2\nline3y" >expect.txt &&
|
|
|
|
test_cmp expect.txt output.txt'
|
|
|
|
|
|
|
|
test_expect_success 'conflict at EOF without LF resolved by --union' \
|
|
|
|
'git merge-file -p --union nolf-diff1.txt nolf-orig.txt nolf-diff2.txt >output.txt &&
|
|
|
|
printf "line1\nline2\nline3x\nline3y" >expect.txt &&
|
|
|
|
test_cmp expect.txt output.txt'
|
|
|
|
|
|
|
|
test_expect_success 'conflict sections match existing line endings' '
|
merge-file: let conflict markers match end-of-line style of the context
When merging files with CR/LF line endings, the conflict markers should
match those, lest the output file has mixed line endings.
This is particularly of interest on Windows, where some editors get
*really* confused by mixed line endings.
The original version of this patch by Beat Bolli respected core.eol, and
a subsequent improvement by this developer also respected gitattributes.
This approach was suboptimal, though: `git merge-file` was invented as a
drop-in replacement for GNU merge and as such has no problem operating
outside of any repository at all!
Another problem with the original approach was pointed out by Junio
Hamano: legacy repositories might have their text files committed using
CR/LF line endings (and core.eol and the gitattributes would give us a
false impression there). Therefore, the much superior approach is to
simply match the context's line endings, if any.
We actually do not have to look at the *entire* context at all: if the
files are all LF-only, or if they all have CR/LF line endings, it is
sufficient to look at just a *single* line to match that style. And if
the line endings are mixed anyway, it is *still* okay to imitate just a
single line's eol: we will just add to the pile of mixed line endings,
and there is nothing we can do about that.
So what we do is: we look at the line preceding the conflict, falling
back to the line preceding that in case it was the last line and had no
line ending, falling back to the first line, first in the first
post-image, then the second post-image, and finally the pre-image.
If we find consistent CR/LF (or undecided) end-of-line style, we match
that, otherwise we use LF-only line endings for the conflict markers.
Note that while it is true that there have to be at least two lines we
can look at (otherwise there would be no conflict), the same is not true
for line *endings*: the three files in question could all consist of a
single line without any line ending, each. In this case we fall back to
using LF-only.
Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
9 years ago
|
|
|
printf "1\\r\\n2\\r\\n3" >crlf-orig.txt &&
|
|
|
|
printf "1\\r\\n2\\r\\n4" >crlf-diff1.txt &&
|
|
|
|
printf "1\\r\\n2\\r\\n5" >crlf-diff2.txt &&
|
|
|
|
test_must_fail git -c core.eol=crlf merge-file -p \
|
|
|
|
crlf-diff1.txt crlf-orig.txt crlf-diff2.txt >crlf.txt &&
|
|
|
|
test $(tr "\015" Q <crlf.txt | grep "^[<=>].*Q$" | wc -l) = 3 &&
|
|
|
|
test $(tr "\015" Q <crlf.txt | grep "[345]Q$" | wc -l) = 3 &&
|
merge-file: let conflict markers match end-of-line style of the context
When merging files with CR/LF line endings, the conflict markers should
match those, lest the output file has mixed line endings.
This is particularly of interest on Windows, where some editors get
*really* confused by mixed line endings.
The original version of this patch by Beat Bolli respected core.eol, and
a subsequent improvement by this developer also respected gitattributes.
This approach was suboptimal, though: `git merge-file` was invented as a
drop-in replacement for GNU merge and as such has no problem operating
outside of any repository at all!
Another problem with the original approach was pointed out by Junio
Hamano: legacy repositories might have their text files committed using
CR/LF line endings (and core.eol and the gitattributes would give us a
false impression there). Therefore, the much superior approach is to
simply match the context's line endings, if any.
We actually do not have to look at the *entire* context at all: if the
files are all LF-only, or if they all have CR/LF line endings, it is
sufficient to look at just a *single* line to match that style. And if
the line endings are mixed anyway, it is *still* okay to imitate just a
single line's eol: we will just add to the pile of mixed line endings,
and there is nothing we can do about that.
So what we do is: we look at the line preceding the conflict, falling
back to the line preceding that in case it was the last line and had no
line ending, falling back to the first line, first in the first
post-image, then the second post-image, and finally the pre-image.
If we find consistent CR/LF (or undecided) end-of-line style, we match
that, otherwise we use LF-only line endings for the conflict markers.
Note that while it is true that there have to be at least two lines we
can look at (otherwise there would be no conflict), the same is not true
for line *endings*: the three files in question could all consist of a
single line without any line ending, each. In this case we fall back to
using LF-only.
Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
9 years ago
|
|
|
test_must_fail git -c core.eol=crlf merge-file -p \
|
|
|
|
nolf-diff1.txt nolf-orig.txt nolf-diff2.txt >nolf.txt &&
|
|
|
|
test $(tr "\015" Q <nolf.txt | grep "^[<=>].*Q$" | wc -l) = 0
|
|
|
|
'
|
|
|
|
|
|
|
|
test_done
|