git-apply: Loosen "match_beginning" logic
Even after a handfle attempts, match_beginning logic still has corner cases:1bf1a85
(apply: treat EOF as proper context., 2006-05-23)65aadb9
(apply: force matching at the beginning., 2006-05-24)4be6096
(apply --unidiff-zero: loosen sanity checks ..., 2006-09-17)ee5a317
(Fix "git apply" to correctly enforce "match ..., 2008-04-06) This is a tricky piece of code. We still incorrectly enforce "match_beginning" for -U0 matches. I noticed this while trying out an example sequence from Clemens Buchacher: $ echo a >victim $ git add victim $ echo b >>victim $ git diff -U0 >patch $ cat patch diff --git i/victim w/victim index 7898192..422c2b7 100644 --- i/victim +++ w/victim @@ -1,0 +2 @@ a +b $ git apply --cached --unidiff-zero <patch $ git show :victim b a The change inserts a new line before the second line, but we insist it to be applied at the beginning. As the result, the code refuses to apply it at the original offset, and we end up adding the line at the beginning. Updates to the test script are by Clemens Buchacher. Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
ee837244df
commit
ed0f47a8c4
@ -1996,6 +1996,8 @@ static int apply_one_fragment(struct image *img, struct fragment *frag,
|
|||||||
/*
|
/*
|
||||||
* A hunk to change lines at the beginning would begin with
|
* A hunk to change lines at the beginning would begin with
|
||||||
* @@ -1,L +N,M @@
|
* @@ -1,L +N,M @@
|
||||||
|
* but we need to be careful. -U0 that inserts before the second
|
||||||
|
* line also has this pattern.
|
||||||
*
|
*
|
||||||
* And a hunk to add to an empty file would begin with
|
* And a hunk to add to an empty file would begin with
|
||||||
* @@ -0,0 +N,M @@
|
* @@ -0,0 +N,M @@
|
||||||
@ -2003,7 +2005,8 @@ static int apply_one_fragment(struct image *img, struct fragment *frag,
|
|||||||
* In other words, a hunk that is (frag->oldpos <= 1) with or
|
* In other words, a hunk that is (frag->oldpos <= 1) with or
|
||||||
* without leading context must match at the beginning.
|
* without leading context must match at the beginning.
|
||||||
*/
|
*/
|
||||||
match_beginning = frag->oldpos <= 1;
|
match_beginning = (!frag->oldpos ||
|
||||||
|
(frag->oldpos == 1 && !unidiff_zero));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* A hunk without trailing lines must match at the end.
|
* A hunk without trailing lines must match at the end.
|
||||||
|
@ -27,6 +27,15 @@ test_expect_success setup '
|
|||||||
git diff victim >add-a-patch.with &&
|
git diff victim >add-a-patch.with &&
|
||||||
git diff --unified=0 >add-a-patch.without &&
|
git diff --unified=0 >add-a-patch.without &&
|
||||||
|
|
||||||
|
: insert at line two
|
||||||
|
for i in b a '"$L"' y
|
||||||
|
do
|
||||||
|
echo $i
|
||||||
|
done >victim &&
|
||||||
|
cat victim >insert-a-expect &&
|
||||||
|
git diff victim >insert-a-patch.with &&
|
||||||
|
git diff --unified=0 >insert-a-patch.without &&
|
||||||
|
|
||||||
: modify at the head
|
: modify at the head
|
||||||
for i in a '"$L"' y
|
for i in a '"$L"' y
|
||||||
do
|
do
|
||||||
@ -55,7 +64,7 @@ test_expect_success setup '
|
|||||||
git diff --unified=0 >add-z-patch.without &&
|
git diff --unified=0 >add-z-patch.without &&
|
||||||
|
|
||||||
: modify at the tail
|
: modify at the tail
|
||||||
for i in a '"$L"' y
|
for i in b '"$L"' z
|
||||||
do
|
do
|
||||||
echo $i
|
echo $i
|
||||||
done >victim &&
|
done >victim &&
|
||||||
@ -81,7 +90,7 @@ do
|
|||||||
with) u= ;;
|
with) u= ;;
|
||||||
without) u='--unidiff-zero ' ;;
|
without) u='--unidiff-zero ' ;;
|
||||||
esac
|
esac
|
||||||
for kind in add-a add-z mod-a mod-z del-a del-z
|
for kind in add-a add-z insert-a mod-a mod-z del-a del-z
|
||||||
do
|
do
|
||||||
test_expect_success "apply $kind-patch $with context" '
|
test_expect_success "apply $kind-patch $with context" '
|
||||||
cat original >victim &&
|
cat original >victim &&
|
||||||
@ -95,7 +104,7 @@ do
|
|||||||
done
|
done
|
||||||
done
|
done
|
||||||
|
|
||||||
for kind in add-a add-z mod-a mod-z del-a del-z
|
for kind in add-a add-z insert-a mod-a mod-z del-a del-z
|
||||||
do
|
do
|
||||||
rm -f $kind-ng.without
|
rm -f $kind-ng.without
|
||||||
sed -e "s/^diff --git /diff /" \
|
sed -e "s/^diff --git /diff /" \
|
||||||
|
Loading…
Reference in New Issue
Block a user