Merge branch 'mv/merge-ff-tristate'
The configuration variable "merge.ff" was cleary a tri-state to choose one from "favor fast-forward when possible", "always create a merge even when the history could fast-forward" and "do not create any merge, only update when the history fast-forwards", but the command line parser did not implement the usual convention of "last one wins, and command line overrides the configuration" correctly. * mv/merge-ff-tristate: merge: handle --ff/--no-ff/--ff-only as a tri-state option
This commit is contained in:
commit
9678ee7ba3
@ -47,8 +47,8 @@ static const char * const builtin_merge_usage[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
static int show_diffstat = 1, shortlog_len = -1, squash;
|
static int show_diffstat = 1, shortlog_len = -1, squash;
|
||||||
static int option_commit = 1, allow_fast_forward = 1;
|
static int option_commit = 1;
|
||||||
static int fast_forward_only, option_edit = -1;
|
static int option_edit = -1;
|
||||||
static int allow_trivial = 1, have_message, verify_signatures;
|
static int allow_trivial = 1, have_message, verify_signatures;
|
||||||
static int overwrite_ignore = 1;
|
static int overwrite_ignore = 1;
|
||||||
static struct strbuf merge_msg = STRBUF_INIT;
|
static struct strbuf merge_msg = STRBUF_INIT;
|
||||||
@ -76,6 +76,14 @@ static struct strategy all_strategy[] = {
|
|||||||
|
|
||||||
static const char *pull_twohead, *pull_octopus;
|
static const char *pull_twohead, *pull_octopus;
|
||||||
|
|
||||||
|
enum ff_type {
|
||||||
|
FF_NO,
|
||||||
|
FF_ALLOW,
|
||||||
|
FF_ONLY
|
||||||
|
};
|
||||||
|
|
||||||
|
static enum ff_type fast_forward = FF_ALLOW;
|
||||||
|
|
||||||
static int option_parse_message(const struct option *opt,
|
static int option_parse_message(const struct option *opt,
|
||||||
const char *arg, int unset)
|
const char *arg, int unset)
|
||||||
{
|
{
|
||||||
@ -178,6 +186,13 @@ static int option_parse_n(const struct option *opt,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int option_parse_ff_only(const struct option *opt,
|
||||||
|
const char *arg, int unset)
|
||||||
|
{
|
||||||
|
fast_forward = FF_ONLY;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static struct option builtin_merge_options[] = {
|
static struct option builtin_merge_options[] = {
|
||||||
{ OPTION_CALLBACK, 'n', NULL, NULL, NULL,
|
{ OPTION_CALLBACK, 'n', NULL, NULL, NULL,
|
||||||
N_("do not show a diffstat at the end of the merge"),
|
N_("do not show a diffstat at the end of the merge"),
|
||||||
@ -194,10 +209,10 @@ static struct option builtin_merge_options[] = {
|
|||||||
N_("perform a commit if the merge succeeds (default)")),
|
N_("perform a commit if the merge succeeds (default)")),
|
||||||
OPT_BOOL('e', "edit", &option_edit,
|
OPT_BOOL('e', "edit", &option_edit,
|
||||||
N_("edit message before committing")),
|
N_("edit message before committing")),
|
||||||
OPT_BOOLEAN(0, "ff", &allow_fast_forward,
|
OPT_SET_INT(0, "ff", &fast_forward, N_("allow fast-forward (default)"), FF_ALLOW),
|
||||||
N_("allow fast-forward (default)")),
|
{ OPTION_CALLBACK, 0, "ff-only", NULL, NULL,
|
||||||
OPT_BOOLEAN(0, "ff-only", &fast_forward_only,
|
N_("abort if fast-forward is not possible"),
|
||||||
N_("abort if fast-forward is not possible")),
|
PARSE_OPT_NOARG | PARSE_OPT_NONEG, option_parse_ff_only },
|
||||||
OPT_RERERE_AUTOUPDATE(&allow_rerere_auto),
|
OPT_RERERE_AUTOUPDATE(&allow_rerere_auto),
|
||||||
OPT_BOOL(0, "verify-signatures", &verify_signatures,
|
OPT_BOOL(0, "verify-signatures", &verify_signatures,
|
||||||
N_("Verify that the named commit has a valid GPG signature")),
|
N_("Verify that the named commit has a valid GPG signature")),
|
||||||
@ -581,10 +596,9 @@ static int git_merge_config(const char *k, const char *v, void *cb)
|
|||||||
else if (!strcmp(k, "merge.ff")) {
|
else if (!strcmp(k, "merge.ff")) {
|
||||||
int boolval = git_config_maybe_bool(k, v);
|
int boolval = git_config_maybe_bool(k, v);
|
||||||
if (0 <= boolval) {
|
if (0 <= boolval) {
|
||||||
allow_fast_forward = boolval;
|
fast_forward = boolval ? FF_ALLOW : FF_NO;
|
||||||
} else if (v && !strcmp(v, "only")) {
|
} else if (v && !strcmp(v, "only")) {
|
||||||
allow_fast_forward = 1;
|
fast_forward = FF_ONLY;
|
||||||
fast_forward_only = 1;
|
|
||||||
} /* do not barf on values from future versions of git */
|
} /* do not barf on values from future versions of git */
|
||||||
return 0;
|
return 0;
|
||||||
} else if (!strcmp(k, "merge.defaulttoupstream")) {
|
} else if (!strcmp(k, "merge.defaulttoupstream")) {
|
||||||
@ -863,7 +877,7 @@ static int finish_automerge(struct commit *head,
|
|||||||
|
|
||||||
free_commit_list(common);
|
free_commit_list(common);
|
||||||
parents = remoteheads;
|
parents = remoteheads;
|
||||||
if (!head_subsumed || !allow_fast_forward)
|
if (!head_subsumed || fast_forward == FF_NO)
|
||||||
commit_list_insert(head, &parents);
|
commit_list_insert(head, &parents);
|
||||||
strbuf_addch(&merge_msg, '\n');
|
strbuf_addch(&merge_msg, '\n');
|
||||||
prepare_to_commit(remoteheads);
|
prepare_to_commit(remoteheads);
|
||||||
@ -1008,7 +1022,7 @@ static void write_merge_state(struct commit_list *remoteheads)
|
|||||||
if (fd < 0)
|
if (fd < 0)
|
||||||
die_errno(_("Could not open '%s' for writing"), filename);
|
die_errno(_("Could not open '%s' for writing"), filename);
|
||||||
strbuf_reset(&buf);
|
strbuf_reset(&buf);
|
||||||
if (!allow_fast_forward)
|
if (fast_forward == FF_NO)
|
||||||
strbuf_addf(&buf, "no-ff");
|
strbuf_addf(&buf, "no-ff");
|
||||||
if (write_in_full(fd, buf.buf, buf.len) != buf.len)
|
if (write_in_full(fd, buf.buf, buf.len) != buf.len)
|
||||||
die_errno(_("Could not write to '%s'"), filename);
|
die_errno(_("Could not write to '%s'"), filename);
|
||||||
@ -1157,14 +1171,11 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
|
|||||||
show_diffstat = 0;
|
show_diffstat = 0;
|
||||||
|
|
||||||
if (squash) {
|
if (squash) {
|
||||||
if (!allow_fast_forward)
|
if (fast_forward == FF_NO)
|
||||||
die(_("You cannot combine --squash with --no-ff."));
|
die(_("You cannot combine --squash with --no-ff."));
|
||||||
option_commit = 0;
|
option_commit = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!allow_fast_forward && fast_forward_only)
|
|
||||||
die(_("You cannot combine --no-ff with --ff-only."));
|
|
||||||
|
|
||||||
if (!abort_current_merge) {
|
if (!abort_current_merge) {
|
||||||
if (!argc) {
|
if (!argc) {
|
||||||
if (default_to_upstream)
|
if (default_to_upstream)
|
||||||
@ -1206,7 +1217,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
|
|||||||
"empty head"));
|
"empty head"));
|
||||||
if (squash)
|
if (squash)
|
||||||
die(_("Squash commit into empty head not supported yet"));
|
die(_("Squash commit into empty head not supported yet"));
|
||||||
if (!allow_fast_forward)
|
if (fast_forward == FF_NO)
|
||||||
die(_("Non-fast-forward commit does not make sense into "
|
die(_("Non-fast-forward commit does not make sense into "
|
||||||
"an empty head"));
|
"an empty head"));
|
||||||
remoteheads = collect_parents(head_commit, &head_subsumed, argc, argv);
|
remoteheads = collect_parents(head_commit, &head_subsumed, argc, argv);
|
||||||
@ -1294,11 +1305,11 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
|
|||||||
sha1_to_hex(commit->object.sha1));
|
sha1_to_hex(commit->object.sha1));
|
||||||
setenv(buf.buf, merge_remote_util(commit)->name, 1);
|
setenv(buf.buf, merge_remote_util(commit)->name, 1);
|
||||||
strbuf_reset(&buf);
|
strbuf_reset(&buf);
|
||||||
if (!fast_forward_only &&
|
if (fast_forward != FF_ONLY &&
|
||||||
merge_remote_util(commit) &&
|
merge_remote_util(commit) &&
|
||||||
merge_remote_util(commit)->obj &&
|
merge_remote_util(commit)->obj &&
|
||||||
merge_remote_util(commit)->obj->type == OBJ_TAG)
|
merge_remote_util(commit)->obj->type == OBJ_TAG)
|
||||||
allow_fast_forward = 0;
|
fast_forward = FF_NO;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (option_edit < 0)
|
if (option_edit < 0)
|
||||||
@ -1315,7 +1326,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
|
|||||||
|
|
||||||
for (i = 0; i < use_strategies_nr; i++) {
|
for (i = 0; i < use_strategies_nr; i++) {
|
||||||
if (use_strategies[i]->attr & NO_FAST_FORWARD)
|
if (use_strategies[i]->attr & NO_FAST_FORWARD)
|
||||||
allow_fast_forward = 0;
|
fast_forward = FF_NO;
|
||||||
if (use_strategies[i]->attr & NO_TRIVIAL)
|
if (use_strategies[i]->attr & NO_TRIVIAL)
|
||||||
allow_trivial = 0;
|
allow_trivial = 0;
|
||||||
}
|
}
|
||||||
@ -1345,7 +1356,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
|
|||||||
*/
|
*/
|
||||||
finish_up_to_date("Already up-to-date.");
|
finish_up_to_date("Already up-to-date.");
|
||||||
goto done;
|
goto done;
|
||||||
} else if (allow_fast_forward && !remoteheads->next &&
|
} else if (fast_forward != FF_NO && !remoteheads->next &&
|
||||||
!common->next &&
|
!common->next &&
|
||||||
!hashcmp(common->item->object.sha1, head_commit->object.sha1)) {
|
!hashcmp(common->item->object.sha1, head_commit->object.sha1)) {
|
||||||
/* Again the most common case of merging one remote. */
|
/* Again the most common case of merging one remote. */
|
||||||
@ -1392,7 +1403,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
|
|||||||
* only one common.
|
* only one common.
|
||||||
*/
|
*/
|
||||||
refresh_cache(REFRESH_QUIET);
|
refresh_cache(REFRESH_QUIET);
|
||||||
if (allow_trivial && !fast_forward_only) {
|
if (allow_trivial && fast_forward != FF_ONLY) {
|
||||||
/* See if it is really trivial. */
|
/* See if it is really trivial. */
|
||||||
git_committer_info(IDENT_STRICT);
|
git_committer_info(IDENT_STRICT);
|
||||||
printf(_("Trying really trivial in-index merge...\n"));
|
printf(_("Trying really trivial in-index merge...\n"));
|
||||||
@ -1433,7 +1444,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fast_forward_only)
|
if (fast_forward == FF_ONLY)
|
||||||
die(_("Not possible to fast-forward, aborting."));
|
die(_("Not possible to fast-forward, aborting."));
|
||||||
|
|
||||||
/* We are going to make a new commit. */
|
/* We are going to make a new commit. */
|
||||||
|
@ -497,9 +497,15 @@ test_expect_success 'combining --squash and --no-ff is refused' '
|
|||||||
test_must_fail git merge --no-ff --squash c1
|
test_must_fail git merge --no-ff --squash c1
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'combining --ff-only and --no-ff is refused' '
|
test_expect_success 'option --ff-only overwrites --no-ff' '
|
||||||
test_must_fail git merge --ff-only --no-ff c1 &&
|
git merge --no-ff --ff-only c1 &&
|
||||||
test_must_fail git merge --no-ff --ff-only c1
|
test_must_fail git merge --no-ff --ff-only c2
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'option --ff-only overwrites merge.ff=only config' '
|
||||||
|
git reset --hard c0 &&
|
||||||
|
test_config merge.ff only &&
|
||||||
|
git merge --no-ff c1
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'merge c0 with c1 (ff overrides no-ff)' '
|
test_expect_success 'merge c0 with c1 (ff overrides no-ff)' '
|
||||||
|
Loading…
Reference in New Issue
Block a user