Skip to content

Commit faaab8d

Browse files
davvidgitster
authored andcommitted
mergetool: support delete/delete conflicts
If two branches each move a file into different directories then mergetool will fail because it assumes that the file being merged, and its parent directory, are present in the worktree. Create the merge file's parent directory to allow using the deleted base version of the file for merge resolution when encountering a delete/delete conflict. The end result is that a delete/delete conflict is presented for the user to resolve. Reported-by: Joe Einertson <[email protected]> Signed-off-by: David Aguilar <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 326e5bc commit faaab8d

File tree

2 files changed

+50
-3
lines changed

2 files changed

+50
-3
lines changed

git-mergetool.sh

Lines changed: 11 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -282,8 +282,14 @@ merge_file () {
282282
return
283283
fi
284284

285-
mv -- "$MERGED" "$BACKUP"
286-
cp -- "$BACKUP" "$MERGED"
285+
if test -f "$MERGED"
286+
then
287+
mv -- "$MERGED" "$BACKUP"
288+
cp -- "$BACKUP" "$MERGED"
289+
fi
290+
# Create a parent directory to handle delete/delete conflicts
291+
# where the base's directory no longer exists.
292+
mkdir -p "$(dirname "$MERGED")"
287293

288294
checkout_staged_file 1 "$MERGED" "$BASE"
289295
checkout_staged_file 2 "$MERGED" "$LOCAL"
@@ -295,7 +301,9 @@ merge_file () {
295301
describe_file "$local_mode" "local" "$LOCAL"
296302
describe_file "$remote_mode" "remote" "$REMOTE"
297303
resolve_deleted_merge
298-
return
304+
status=$?
305+
rmdir -p "$(dirname "$MERGED")" 2>/dev/null
306+
return $status
299307
fi
300308

301309
if is_symlink "$local_mode" || is_symlink "$remote_mode"

t/t7610-mergetool.sh

Lines changed: 39 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -243,6 +243,45 @@ test_expect_success 'mergetool takes partial path' '
243243
git reset --hard
244244
'
245245

246+
test_expect_success 'mergetool delete/delete conflict' '
247+
git checkout -b delete-base branch1 &&
248+
mkdir -p a/a &&
249+
(echo one; echo two; echo 3; echo 4) >a/a/file.txt &&
250+
git add a/a/file.txt &&
251+
git commit -m"base file" &&
252+
git checkout -b move-to-b delete-base &&
253+
mkdir -p b/b &&
254+
git mv a/a/file.txt b/b/file.txt &&
255+
(echo one; echo two; echo 4) >b/b/file.txt &&
256+
git commit -a -m"move to b" &&
257+
git checkout -b move-to-c delete-base &&
258+
mkdir -p c/c &&
259+
git mv a/a/file.txt c/c/file.txt &&
260+
(echo one; echo two; echo 3) >c/c/file.txt &&
261+
git commit -a -m"move to c" &&
262+
test_must_fail git merge move-to-b &&
263+
echo d | git mergetool a/a/file.txt &&
264+
! test -f a/a/file.txt &&
265+
git reset --hard HEAD &&
266+
test_must_fail git merge move-to-b &&
267+
echo m | git mergetool a/a/file.txt &&
268+
test -f b/b/file.txt &&
269+
git reset --hard HEAD &&
270+
test_must_fail git merge move-to-b &&
271+
! echo a | git mergetool a/a/file.txt &&
272+
! test -f a/a/file.txt &&
273+
git reset --hard HEAD
274+
'
275+
276+
test_expect_success 'mergetool produces no errors when keepBackup is used' '
277+
test_config mergetool.keepBackup true &&
278+
test_must_fail git merge move-to-b &&
279+
: >expect &&
280+
echo d | git mergetool a/a/file.txt 2>actual &&
281+
test_cmp expect actual &&
282+
git reset --hard HEAD
283+
'
284+
246285
test_expect_success 'deleted vs modified submodule' '
247286
git checkout -b test6 branch1 &&
248287
git submodule update -N &&

0 commit comments

Comments
 (0)