Skip to content

Commit 9678ee7

Browse files
committed
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
2 parents dd28abc + a54841e commit 9678ee7

File tree

2 files changed

+42
-25
lines changed

2 files changed

+42
-25
lines changed

builtin/merge.c

Lines changed: 33 additions & 22 deletions
Original file line numberDiff line numberDiff line change
@@ -47,8 +47,8 @@ static const char * const builtin_merge_usage[] = {
4747
};
4848

4949
static int show_diffstat = 1, shortlog_len = -1, squash;
50-
static int option_commit = 1, allow_fast_forward = 1;
51-
static int fast_forward_only, option_edit = -1;
50+
static int option_commit = 1;
51+
static int option_edit = -1;
5252
static int allow_trivial = 1, have_message, verify_signatures;
5353
static int overwrite_ignore = 1;
5454
static struct strbuf merge_msg = STRBUF_INIT;
@@ -76,6 +76,14 @@ static struct strategy all_strategy[] = {
7676

7777
static const char *pull_twohead, *pull_octopus;
7878

79+
enum ff_type {
80+
FF_NO,
81+
FF_ALLOW,
82+
FF_ONLY
83+
};
84+
85+
static enum ff_type fast_forward = FF_ALLOW;
86+
7987
static int option_parse_message(const struct option *opt,
8088
const char *arg, int unset)
8189
{
@@ -178,6 +186,13 @@ static int option_parse_n(const struct option *opt,
178186
return 0;
179187
}
180188

189+
static int option_parse_ff_only(const struct option *opt,
190+
const char *arg, int unset)
191+
{
192+
fast_forward = FF_ONLY;
193+
return 0;
194+
}
195+
181196
static struct option builtin_merge_options[] = {
182197
{ OPTION_CALLBACK, 'n', NULL, NULL, NULL,
183198
N_("do not show a diffstat at the end of the merge"),
@@ -194,10 +209,10 @@ static struct option builtin_merge_options[] = {
194209
N_("perform a commit if the merge succeeds (default)")),
195210
OPT_BOOL('e', "edit", &option_edit,
196211
N_("edit message before committing")),
197-
OPT_BOOLEAN(0, "ff", &allow_fast_forward,
198-
N_("allow fast-forward (default)")),
199-
OPT_BOOLEAN(0, "ff-only", &fast_forward_only,
200-
N_("abort if fast-forward is not possible")),
212+
OPT_SET_INT(0, "ff", &fast_forward, N_("allow fast-forward (default)"), FF_ALLOW),
213+
{ OPTION_CALLBACK, 0, "ff-only", NULL, NULL,
214+
N_("abort if fast-forward is not possible"),
215+
PARSE_OPT_NOARG | PARSE_OPT_NONEG, option_parse_ff_only },
201216
OPT_RERERE_AUTOUPDATE(&allow_rerere_auto),
202217
OPT_BOOL(0, "verify-signatures", &verify_signatures,
203218
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)
581596
else if (!strcmp(k, "merge.ff")) {
582597
int boolval = git_config_maybe_bool(k, v);
583598
if (0 <= boolval) {
584-
allow_fast_forward = boolval;
599+
fast_forward = boolval ? FF_ALLOW : FF_NO;
585600
} else if (v && !strcmp(v, "only")) {
586-
allow_fast_forward = 1;
587-
fast_forward_only = 1;
601+
fast_forward = FF_ONLY;
588602
} /* do not barf on values from future versions of git */
589603
return 0;
590604
} else if (!strcmp(k, "merge.defaulttoupstream")) {
@@ -863,7 +877,7 @@ static int finish_automerge(struct commit *head,
863877

864878
free_commit_list(common);
865879
parents = remoteheads;
866-
if (!head_subsumed || !allow_fast_forward)
880+
if (!head_subsumed || fast_forward == FF_NO)
867881
commit_list_insert(head, &parents);
868882
strbuf_addch(&merge_msg, '\n');
869883
prepare_to_commit(remoteheads);
@@ -1008,7 +1022,7 @@ static void write_merge_state(struct commit_list *remoteheads)
10081022
if (fd < 0)
10091023
die_errno(_("Could not open '%s' for writing"), filename);
10101024
strbuf_reset(&buf);
1011-
if (!allow_fast_forward)
1025+
if (fast_forward == FF_NO)
10121026
strbuf_addf(&buf, "no-ff");
10131027
if (write_in_full(fd, buf.buf, buf.len) != buf.len)
10141028
die_errno(_("Could not write to '%s'"), filename);
@@ -1157,14 +1171,11 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
11571171
show_diffstat = 0;
11581172

11591173
if (squash) {
1160-
if (!allow_fast_forward)
1174+
if (fast_forward == FF_NO)
11611175
die(_("You cannot combine --squash with --no-ff."));
11621176
option_commit = 0;
11631177
}
11641178

1165-
if (!allow_fast_forward && fast_forward_only)
1166-
die(_("You cannot combine --no-ff with --ff-only."));
1167-
11681179
if (!abort_current_merge) {
11691180
if (!argc) {
11701181
if (default_to_upstream)
@@ -1206,7 +1217,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
12061217
"empty head"));
12071218
if (squash)
12081219
die(_("Squash commit into empty head not supported yet"));
1209-
if (!allow_fast_forward)
1220+
if (fast_forward == FF_NO)
12101221
die(_("Non-fast-forward commit does not make sense into "
12111222
"an empty head"));
12121223
remoteheads = collect_parents(head_commit, &head_subsumed, argc, argv);
@@ -1294,11 +1305,11 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
12941305
sha1_to_hex(commit->object.sha1));
12951306
setenv(buf.buf, merge_remote_util(commit)->name, 1);
12961307
strbuf_reset(&buf);
1297-
if (!fast_forward_only &&
1308+
if (fast_forward != FF_ONLY &&
12981309
merge_remote_util(commit) &&
12991310
merge_remote_util(commit)->obj &&
13001311
merge_remote_util(commit)->obj->type == OBJ_TAG)
1301-
allow_fast_forward = 0;
1312+
fast_forward = FF_NO;
13021313
}
13031314

13041315
if (option_edit < 0)
@@ -1315,7 +1326,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
13151326

13161327
for (i = 0; i < use_strategies_nr; i++) {
13171328
if (use_strategies[i]->attr & NO_FAST_FORWARD)
1318-
allow_fast_forward = 0;
1329+
fast_forward = FF_NO;
13191330
if (use_strategies[i]->attr & NO_TRIVIAL)
13201331
allow_trivial = 0;
13211332
}
@@ -1345,7 +1356,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
13451356
*/
13461357
finish_up_to_date("Already up-to-date.");
13471358
goto done;
1348-
} else if (allow_fast_forward && !remoteheads->next &&
1359+
} else if (fast_forward != FF_NO && !remoteheads->next &&
13491360
!common->next &&
13501361
!hashcmp(common->item->object.sha1, head_commit->object.sha1)) {
13511362
/* Again the most common case of merging one remote. */
@@ -1392,7 +1403,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
13921403
* only one common.
13931404
*/
13941405
refresh_cache(REFRESH_QUIET);
1395-
if (allow_trivial && !fast_forward_only) {
1406+
if (allow_trivial && fast_forward != FF_ONLY) {
13961407
/* See if it is really trivial. */
13971408
git_committer_info(IDENT_STRICT);
13981409
printf(_("Trying really trivial in-index merge...\n"));
@@ -1433,7 +1444,7 @@ int cmd_merge(int argc, const char **argv, const char *prefix)
14331444
}
14341445
}
14351446

1436-
if (fast_forward_only)
1447+
if (fast_forward == FF_ONLY)
14371448
die(_("Not possible to fast-forward, aborting."));
14381449

14391450
/* We are going to make a new commit. */

t/t7600-merge.sh

Lines changed: 9 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -497,9 +497,15 @@ test_expect_success 'combining --squash and --no-ff is refused' '
497497
test_must_fail git merge --no-ff --squash c1
498498
'
499499

500-
test_expect_success 'combining --ff-only and --no-ff is refused' '
501-
test_must_fail git merge --ff-only --no-ff c1 &&
502-
test_must_fail git merge --no-ff --ff-only c1
500+
test_expect_success 'option --ff-only overwrites --no-ff' '
501+
git merge --no-ff --ff-only c1 &&
502+
test_must_fail git merge --no-ff --ff-only c2
503+
'
504+
505+
test_expect_success 'option --ff-only overwrites merge.ff=only config' '
506+
git reset --hard c0 &&
507+
test_config merge.ff only &&
508+
git merge --no-ff c1
503509
'
504510

505511
test_expect_success 'merge c0 with c1 (ff overrides no-ff)' '

0 commit comments

Comments
 (0)