Skip to content

Commit 79c4759

Browse files
newrengitster
authored andcommitted
merge-recursive: avoid clobbering untracked files with directory renames
Reviewed-by: Stefan Beller <[email protected]> Signed-off-by: Elijah Newren <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 9c0743f commit 79c4759

File tree

2 files changed

+43
-5
lines changed

2 files changed

+43
-5
lines changed

merge-recursive.c

Lines changed: 40 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1151,6 +1151,26 @@ static int conflict_rename_dir(struct merge_options *o,
11511151
{
11521152
const struct diff_filespec *dest = pair->two;
11531153

1154+
if (!o->call_depth && would_lose_untracked(dest->path)) {
1155+
char *alt_path = unique_path(o, dest->path, rename_branch);
1156+
1157+
output(o, 1, _("Error: Refusing to lose untracked file at %s; "
1158+
"writing to %s instead."),
1159+
dest->path, alt_path);
1160+
/*
1161+
* Write the file in worktree at alt_path, but not in the
1162+
* index. Instead, write to dest->path for the index but
1163+
* only at the higher appropriate stage.
1164+
*/
1165+
if (update_file(o, 0, &dest->oid, dest->mode, alt_path))
1166+
return -1;
1167+
free(alt_path);
1168+
return update_stages(o, dest->path, NULL,
1169+
rename_branch == o->branch1 ? dest : NULL,
1170+
rename_branch == o->branch1 ? NULL : dest);
1171+
}
1172+
1173+
/* Update dest->path both in index and in worktree */
11541174
if (update_file(o, 1, &dest->oid, dest->mode, dest->path))
11551175
return -1;
11561176
return 0;
@@ -1169,7 +1189,8 @@ static int handle_change_delete(struct merge_options *o,
11691189
const char *update_path = path;
11701190
int ret = 0;
11711191

1172-
if (dir_in_way(path, !o->call_depth, 0)) {
1192+
if (dir_in_way(path, !o->call_depth, 0) ||
1193+
(!o->call_depth && would_lose_untracked(path))) {
11731194
update_path = alt_path = unique_path(o, path, change_branch);
11741195
}
11751196

@@ -1295,6 +1316,12 @@ static int handle_file(struct merge_options *o,
12951316
dst_name = unique_path(o, rename->path, cur_branch);
12961317
output(o, 1, _("%s is a directory in %s adding as %s instead"),
12971318
rename->path, other_branch, dst_name);
1319+
} else if (!o->call_depth &&
1320+
would_lose_untracked(rename->path)) {
1321+
dst_name = unique_path(o, rename->path, cur_branch);
1322+
output(o, 1, _("Refusing to lose untracked file at %s; "
1323+
"adding as %s instead"),
1324+
rename->path, dst_name);
12981325
}
12991326
}
13001327
if ((ret = update_file(o, 0, &rename->oid, rename->mode, dst_name)))
@@ -1420,7 +1447,18 @@ static int conflict_rename_rename_2to1(struct merge_options *o,
14201447
char *new_path2 = unique_path(o, path, ci->branch2);
14211448
output(o, 1, _("Renaming %s to %s and %s to %s instead"),
14221449
a->path, new_path1, b->path, new_path2);
1423-
remove_file(o, 0, path, 0);
1450+
if (would_lose_untracked(path))
1451+
/*
1452+
* Only way we get here is if both renames were from
1453+
* a directory rename AND user had an untracked file
1454+
* at the location where both files end up after the
1455+
* two directory renames. See testcase 10d of t6043.
1456+
*/
1457+
output(o, 1, _("Refusing to lose untracked file at "
1458+
"%s, even though it's in the way."),
1459+
path);
1460+
else
1461+
remove_file(o, 0, path, 0);
14241462
ret = update_file(o, 0, &mfi_c1.oid, mfi_c1.mode, new_path1);
14251463
if (!ret)
14261464
ret = update_file(o, 0, &mfi_c2.oid, mfi_c2.mode,

t/t6043-merge-rename-directories.sh

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -2992,7 +2992,7 @@ test_expect_success '10b-setup: Overwrite untracked with dir rename + delete' '
29922992
)
29932993
'
29942994

2995-
test_expect_failure '10b-check: Overwrite untracked with dir rename + delete' '
2995+
test_expect_success '10b-check: Overwrite untracked with dir rename + delete' '
29962996
(
29972997
cd 10b &&
29982998
@@ -3070,7 +3070,7 @@ test_expect_success '10c-setup: Overwrite untracked with dir rename/rename(1to2)
30703070
)
30713071
'
30723072

3073-
test_expect_failure '10c-check: Overwrite untracked with dir rename/rename(1to2)' '
3073+
test_expect_success '10c-check: Overwrite untracked with dir rename/rename(1to2)' '
30743074
(
30753075
cd 10c &&
30763076
@@ -3145,7 +3145,7 @@ test_expect_success '10d-setup: Delete untracked with dir rename/rename(2to1)' '
31453145
)
31463146
'
31473147

3148-
test_expect_failure '10d-check: Delete untracked with dir rename/rename(2to1)' '
3148+
test_expect_success '10d-check: Delete untracked with dir rename/rename(2to1)' '
31493149
(
31503150
cd 10d &&
31513151

0 commit comments

Comments
 (0)