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:
Junio C Hamano 2013-07-15 10:28:34 -07:00
commit 9678ee7ba3
2 changed files with 42 additions and 25 deletions

View File

@ -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. */

View File

@ -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)' '