Skip to content

Commit 183d797

Browse files
spearceJunio C Hamano
authored andcommitted
Keep untracked files not involved in a merge.
My earlier fix (8371234) to delete renamed tracked files from the working directory also caused merge-recursive to delete untracked files that were in the working directory. The problem here is merge-recursive is deleting the working directory file without regard for which branch it was associated with. What we really want to do during a merge is to only delete files that were renamed by the branch we are merging into the current branch, and that are still tracked by the current branch. These files definitely don't belong in the working directory anymore. Anything else is either a merge conflict (already handled in other parts of the code) or a file that is untracked by the current branch and thus is not even participating in the merge. Its this latter class that must be left alone. For this fix to work we are now assuming that the first non-base argument passed to git-merge-recursive always corresponds to the working directory. This is already true for all in-tree callers of merge-recursive. This assumption is also supported by the long time usage message of "<base> ... -- <head> <remote>", where "<head>" is implied to be HEAD, which is generally assumed to be the current tree-ish. Signed-off-by: Shawn O. Pearce <spearce@spearce.org> Signed-off-by: Junio C Hamano <junkio@cox.net>
1 parent 3dff537 commit 183d797

File tree

2 files changed

+43
-1
lines changed

2 files changed

+43
-1
lines changed

merge-recursive.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -891,7 +891,7 @@ static int process_renames(struct path_list *a_renames,
891891
struct diff_filespec src_other, dst_other;
892892
int try_merge, stage = a_renames == renames1 ? 3: 2;
893893

894-
remove_file(1, ren1_src, index_only);
894+
remove_file(1, ren1_src, index_only || stage == 3);
895895

896896
hashcpy(src_other.sha1, ren1->src_entry->stages[stage].sha);
897897
src_other.mode = ren1->src_entry->stages[stage].mode;

t/t6023-merge-rename-nocruft.sh

Lines changed: 42 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,7 @@ git add A M &&
4545
git commit -m "initial has A and M" &&
4646
git branch white &&
4747
git branch red &&
48+
git branch blue &&
4849
4950
git checkout white &&
5051
sed -e "/^g /s/.*/g : white changes a line/" <A >B &&
@@ -58,6 +59,13 @@ echo created by red >R &&
5859
git update-index --add R &&
5960
git commit -m "red creates R" &&
6061
62+
git checkout blue &&
63+
sed -e "/^o /s/.*/g : blue changes a line/" <A >B &&
64+
rm -f A &&
65+
mv B A &&
66+
git update-index A &&
67+
git commit -m "blue modify A" &&
68+
6169
git checkout master'
6270

6371
# This test broke in 65ac6e9c3f47807cb603af07a6a9e1a43bc119ae
@@ -94,4 +102,38 @@ test_expect_success 'merge white into red (A->B,M->N)' \
94102
return 0
95103
'
96104

105+
# This test broke in 8371234ecaaf6e14fe3f2082a855eff1bbd79ae9
106+
test_expect_success 'merge blue into white (A->B, mod A, A untracked)' \
107+
'
108+
git checkout -b white-blue white &&
109+
echo dirty >A &&
110+
git merge blue &&
111+
git write-tree >/dev/null || {
112+
echo "BAD: merge did not complete"
113+
return 1
114+
}
115+
116+
test -f A || {
117+
echo "BAD: A does not exist in working directory"
118+
return 1
119+
}
120+
test `cat A` = dirty || {
121+
echo "BAD: A content is wrong"
122+
return 1
123+
}
124+
test -f B || {
125+
echo "BAD: B does not exist in working directory"
126+
return 1
127+
}
128+
test -f N || {
129+
echo "BAD: N does not exist in working directory"
130+
return 1
131+
}
132+
test -f M && {
133+
echo "BAD: M still exists in working directory"
134+
return 1
135+
}
136+
return 0
137+
'
138+
97139
test_done

0 commit comments

Comments
 (0)