Merge branch 'mm/merge-rename-delete-message'
When "git merge" detects a path that is renamed in one history while the other history deleted (or modified) it, it now reports both paths to help the user understand what is going on in the two histories being merged. * mm/merge-rename-delete-message: merge-recursive: make "CONFLICT (rename/delete)" message show both paths
This commit is contained in:
commit
74aabf41ce
@ -1061,16 +1061,20 @@ static int merge_file_one(struct merge_options *o,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int handle_change_delete(struct merge_options *o,
|
static int handle_change_delete(struct merge_options *o,
|
||||||
const char *path,
|
const char *path, const char *old_path,
|
||||||
const struct object_id *o_oid, int o_mode,
|
const struct object_id *o_oid, int o_mode,
|
||||||
const struct object_id *a_oid, int a_mode,
|
const struct object_id *changed_oid,
|
||||||
const struct object_id *b_oid, int b_mode,
|
int changed_mode,
|
||||||
|
const char *change_branch,
|
||||||
|
const char *delete_branch,
|
||||||
const char *change, const char *change_past)
|
const char *change, const char *change_past)
|
||||||
{
|
{
|
||||||
char *renamed = NULL;
|
char *alt_path = NULL;
|
||||||
|
const char *update_path = path;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
if (dir_in_way(path, !o->call_depth, 0)) {
|
if (dir_in_way(path, !o->call_depth, 0)) {
|
||||||
renamed = unique_path(o, path, a_oid ? o->branch1 : o->branch2);
|
update_path = alt_path = unique_path(o, path, change_branch);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (o->call_depth) {
|
if (o->call_depth) {
|
||||||
@ -1081,43 +1085,43 @@ static int handle_change_delete(struct merge_options *o,
|
|||||||
*/
|
*/
|
||||||
ret = remove_file_from_cache(path);
|
ret = remove_file_from_cache(path);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
ret = update_file(o, 0, o_oid, o_mode,
|
ret = update_file(o, 0, o_oid, o_mode, update_path);
|
||||||
renamed ? renamed : path);
|
} else {
|
||||||
} else if (!a_oid) {
|
if (!alt_path) {
|
||||||
if (!renamed) {
|
if (!old_path) {
|
||||||
output(o, 1, _("CONFLICT (%s/delete): %s deleted in %s "
|
output(o, 1, _("CONFLICT (%s/delete): %s deleted in %s "
|
||||||
"and %s in %s. Version %s of %s left in tree."),
|
"and %s in %s. Version %s of %s left in tree."),
|
||||||
change, path, o->branch1, change_past,
|
change, path, delete_branch, change_past,
|
||||||
o->branch2, o->branch2, path);
|
change_branch, change_branch, path);
|
||||||
ret = update_file(o, 0, b_oid, b_mode, path);
|
|
||||||
} else {
|
} else {
|
||||||
output(o, 1, _("CONFLICT (%s/delete): %s deleted in %s "
|
output(o, 1, _("CONFLICT (%s/delete): %s deleted in %s "
|
||||||
"and %s in %s. Version %s of %s left in tree at %s."),
|
"and %s to %s in %s. Version %s of %s left in tree."),
|
||||||
change, path, o->branch1, change_past,
|
change, old_path, delete_branch, change_past, path,
|
||||||
o->branch2, o->branch2, path, renamed);
|
change_branch, change_branch, path);
|
||||||
ret = update_file(o, 0, b_oid, b_mode, renamed);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (!renamed) {
|
if (!old_path) {
|
||||||
output(o, 1, _("CONFLICT (%s/delete): %s deleted in %s "
|
|
||||||
"and %s in %s. Version %s of %s left in tree."),
|
|
||||||
change, path, o->branch2, change_past,
|
|
||||||
o->branch1, o->branch1, path);
|
|
||||||
} else {
|
|
||||||
output(o, 1, _("CONFLICT (%s/delete): %s deleted in %s "
|
output(o, 1, _("CONFLICT (%s/delete): %s deleted in %s "
|
||||||
"and %s in %s. Version %s of %s left in tree at %s."),
|
"and %s in %s. Version %s of %s left in tree at %s."),
|
||||||
change, path, o->branch2, change_past,
|
change, path, delete_branch, change_past,
|
||||||
o->branch1, o->branch1, path, renamed);
|
change_branch, change_branch, path, alt_path);
|
||||||
ret = update_file(o, 0, a_oid, a_mode, renamed);
|
} else {
|
||||||
|
output(o, 1, _("CONFLICT (%s/delete): %s deleted in %s "
|
||||||
|
"and %s to %s in %s. Version %s of %s left in tree at %s."),
|
||||||
|
change, old_path, delete_branch, change_past, path,
|
||||||
|
change_branch, change_branch, path, alt_path);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* No need to call update_file() on path when !renamed, since
|
* No need to call update_file() on path when change_branch ==
|
||||||
* that would needlessly touch path. We could call
|
* o->branch1 && !alt_path, since that would needlessly touch
|
||||||
* update_file_flags() with update_cache=0 and update_wd=0,
|
* path. We could call update_file_flags() with update_cache=0
|
||||||
* but that's a no-op.
|
* and update_wd=0, but that's a no-op.
|
||||||
*/
|
*/
|
||||||
|
if (change_branch != o->branch1 || alt_path)
|
||||||
|
ret = update_file(o, 0, changed_oid, changed_mode, update_path);
|
||||||
}
|
}
|
||||||
free(renamed);
|
free(alt_path);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -1125,28 +1129,17 @@ static int handle_change_delete(struct merge_options *o,
|
|||||||
static int conflict_rename_delete(struct merge_options *o,
|
static int conflict_rename_delete(struct merge_options *o,
|
||||||
struct diff_filepair *pair,
|
struct diff_filepair *pair,
|
||||||
const char *rename_branch,
|
const char *rename_branch,
|
||||||
const char *other_branch)
|
const char *delete_branch)
|
||||||
{
|
{
|
||||||
const struct diff_filespec *orig = pair->one;
|
const struct diff_filespec *orig = pair->one;
|
||||||
const struct diff_filespec *dest = pair->two;
|
const struct diff_filespec *dest = pair->two;
|
||||||
const struct object_id *a_oid = NULL;
|
|
||||||
const struct object_id *b_oid = NULL;
|
|
||||||
int a_mode = 0;
|
|
||||||
int b_mode = 0;
|
|
||||||
|
|
||||||
if (rename_branch == o->branch1) {
|
|
||||||
a_oid = &dest->oid;
|
|
||||||
a_mode = dest->mode;
|
|
||||||
} else {
|
|
||||||
b_oid = &dest->oid;
|
|
||||||
b_mode = dest->mode;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (handle_change_delete(o,
|
if (handle_change_delete(o,
|
||||||
o->call_depth ? orig->path : dest->path,
|
o->call_depth ? orig->path : dest->path,
|
||||||
|
o->call_depth ? NULL : orig->path,
|
||||||
&orig->oid, orig->mode,
|
&orig->oid, orig->mode,
|
||||||
a_oid, a_mode,
|
&dest->oid, dest->mode,
|
||||||
b_oid, b_mode,
|
rename_branch, delete_branch,
|
||||||
_("rename"), _("renamed")))
|
_("rename"), _("renamed")))
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
@ -1662,11 +1655,27 @@ static int handle_modify_delete(struct merge_options *o,
|
|||||||
struct object_id *a_oid, int a_mode,
|
struct object_id *a_oid, int a_mode,
|
||||||
struct object_id *b_oid, int b_mode)
|
struct object_id *b_oid, int b_mode)
|
||||||
{
|
{
|
||||||
|
const char *modify_branch, *delete_branch;
|
||||||
|
struct object_id *changed_oid;
|
||||||
|
int changed_mode;
|
||||||
|
|
||||||
|
if (a_oid) {
|
||||||
|
modify_branch = o->branch1;
|
||||||
|
delete_branch = o->branch2;
|
||||||
|
changed_oid = a_oid;
|
||||||
|
changed_mode = a_mode;
|
||||||
|
} else {
|
||||||
|
modify_branch = o->branch2;
|
||||||
|
delete_branch = o->branch1;
|
||||||
|
changed_oid = b_oid;
|
||||||
|
changed_mode = b_mode;
|
||||||
|
}
|
||||||
|
|
||||||
return handle_change_delete(o,
|
return handle_change_delete(o,
|
||||||
path,
|
path, NULL,
|
||||||
o_oid, o_mode,
|
o_oid, o_mode,
|
||||||
a_oid, a_mode,
|
changed_oid, changed_mode,
|
||||||
b_oid, b_mode,
|
modify_branch, delete_branch,
|
||||||
_("modify"), _("modified"));
|
_("modify"), _("modified"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
23
t/t6045-merge-rename-delete.sh
Executable file
23
t/t6045-merge-rename-delete.sh
Executable file
@ -0,0 +1,23 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
test_description='Merge-recursive rename/delete conflict message'
|
||||||
|
. ./test-lib.sh
|
||||||
|
|
||||||
|
test_expect_success 'rename/delete' '
|
||||||
|
echo foo >A &&
|
||||||
|
git add A &&
|
||||||
|
git commit -m "initial" &&
|
||||||
|
|
||||||
|
git checkout -b rename &&
|
||||||
|
git mv A B &&
|
||||||
|
git commit -m "rename" &&
|
||||||
|
|
||||||
|
git checkout master &&
|
||||||
|
git rm A &&
|
||||||
|
git commit -m "delete" &&
|
||||||
|
|
||||||
|
test_must_fail git merge --strategy=recursive rename >output &&
|
||||||
|
test_i18ngrep "CONFLICT (rename/delete): A deleted in HEAD and renamed to B in rename. Version rename of B left in tree." output
|
||||||
|
'
|
||||||
|
|
||||||
|
test_done
|
Loading…
Reference in New Issue
Block a user