Skip to content

Commit

Permalink
rebase -i: add --ignore-whitespace flag
Browse files Browse the repository at this point in the history
There are two backends available for rebasing, viz, the am and the
interactive. Naturally, there shall be some features that are
implemented in one but not in the other. One such flag is
--ignore-whitespace which indicates merge mechanism to treat lines
with only whitespace changes as unchanged. Wire the interactive
rebase to also understand the --ignore-whitespace flag by
translating it to -Xignore-space-change.

Signed-off-by: Rohit Ashiwal <rohit.ashiwal265@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
  • Loading branch information
r1walz authored and gitster committed Aug 13, 2019
1 parent 9c9b961 commit e191cc8
Show file tree
Hide file tree
Showing 4 changed files with 97 additions and 8 deletions.
10 changes: 8 additions & 2 deletions Documentation/git-rebase.txt
Expand Up @@ -371,8 +371,13 @@ If either <upstream> or --root is given on the command line, then the
default is `--no-fork-point`, otherwise the default is `--fork-point`.

--ignore-whitespace::
This flag is either passed to the 'git apply' program
(see linkgit:git-apply[1]), or to 'git merge' program
(see linkgit:git-merge[1]) as `-Xignore-space-change`,
depending on which backend is selected by other options.

--whitespace=<option>::
These flag are passed to the 'git apply' program
This flag is passed to the 'git apply' program
(see linkgit:git-apply[1]) that applies the patch.
+
See also INCOMPATIBLE OPTIONS below.
Expand Down Expand Up @@ -520,7 +525,6 @@ The following options:
* --committer-date-is-author-date
* --ignore-date
* --whitespace
* --ignore-whitespace
* -C

are incompatible with the following options:
Expand All @@ -543,6 +547,8 @@ In addition, the following pairs of options are incompatible:
* --preserve-merges and --interactive
* --preserve-merges and --signoff
* --preserve-merges and --rebase-merges
* --preserve-merges and --ignore-whitespace
* --rebase-merges and --ignore-whitespace
* --rebase-merges and --strategy
* --rebase-merges and --strategy-option

Expand Down
29 changes: 24 additions & 5 deletions builtin/rebase.c
Expand Up @@ -79,6 +79,7 @@ struct rebase_options {
int allow_rerere_autoupdate;
int keep_empty;
int autosquash;
int ignore_whitespace;
char *gpg_sign_opt;
int autostash;
char *cmd;
Expand All @@ -99,6 +100,7 @@ struct rebase_options {

static struct replay_opts get_replay_opts(const struct rebase_options *opts)
{
char *strategy_opts = opts->strategy_opts;
struct replay_opts replay = REPLAY_OPTS_INIT;

replay.action = REPLAY_INTERACTIVE_REBASE;
Expand All @@ -114,8 +116,19 @@ static struct replay_opts get_replay_opts(const struct rebase_options *opts)
replay.reschedule_failed_exec = opts->reschedule_failed_exec;
replay.gpg_sign = xstrdup_or_null(opts->gpg_sign_opt);
replay.strategy = opts->strategy;
if (opts->strategy_opts)
parse_strategy_opts(&replay, opts->strategy_opts);

if (opts->ignore_whitespace) {
struct strbuf buf = STRBUF_INIT;

if (strategy_opts)
strbuf_addstr(&buf, strategy_opts);

strbuf_addstr(&buf, " --ignore-space-change");
free(strategy_opts);
strategy_opts = strbuf_detach(&buf, NULL);
}
if (strategy_opts)
parse_strategy_opts(&replay, strategy_opts);

return replay;
}
Expand Down Expand Up @@ -511,6 +524,8 @@ int cmd_rebase__interactive(int argc, const char **argv, const char *prefix)
argc = parse_options(argc, argv, prefix, options,
builtin_rebase_interactive_usage, PARSE_OPT_KEEP_ARGV0);

opts.strategy_opts = xstrdup_or_null(opts.strategy_opts);

if (!is_null_oid(&squash_onto))
opts.squash_onto = &squash_onto;

Expand Down Expand Up @@ -964,6 +979,8 @@ static int run_am(struct rebase_options *opts)
am.git_cmd = 1;
argv_array_push(&am.args, "am");

if (opts->ignore_whitespace)
argv_array_push(&am.args, "--ignore-whitespace");
if (opts->action && !strcmp("continue", opts->action)) {
argv_array_push(&am.args, "--resolved");
argv_array_pushf(&am.args, "--resolvemsg=%s", resolvemsg);
Expand Down Expand Up @@ -1407,16 +1424,15 @@ int cmd_rebase(int argc, const char **argv, const char *prefix)
PARSE_OPT_NOARG, NULL, REBASE_DIFFSTAT },
OPT_BOOL(0, "signoff", &options.signoff,
N_("add a Signed-off-by: line to each commit")),
OPT_PASSTHRU_ARGV(0, "ignore-whitespace", &options.git_am_opts,
NULL, N_("passed to 'git am'"),
PARSE_OPT_NOARG),
OPT_PASSTHRU_ARGV(0, "committer-date-is-author-date",
&options.git_am_opts, NULL,
N_("passed to 'git am'"), PARSE_OPT_NOARG),
OPT_PASSTHRU_ARGV(0, "ignore-date", &options.git_am_opts, NULL,
N_("passed to 'git am'"), PARSE_OPT_NOARG),
OPT_PASSTHRU_ARGV('C', NULL, &options.git_am_opts, N_("n"),
N_("passed to 'git apply'"), 0),
OPT_BOOL(0, "ignore-whitespace", &options.ignore_whitespace,
N_("ignore changes in whitespace")),
OPT_PASSTHRU_ARGV(0, "whitespace", &options.git_am_opts,
N_("action"), N_("passed to 'git apply'"), 0),
OPT_BIT('f', "force-rebase", &options.flags,
Expand Down Expand Up @@ -1838,6 +1854,9 @@ int cmd_rebase(int argc, const char **argv, const char *prefix)
}

if (options.rebase_merges) {
if (options.ignore_whitespace)
die(_("cannot combine '--rebase-merges' with "
"'--ignore-whitespace'"));
if (strategy_options.nr)
die(_("cannot combine '--rebase-merges' with "
"'--strategy-option'"));
Expand Down
1 change: 0 additions & 1 deletion t/t3422-rebase-incompatible-options.sh
Expand Up @@ -61,7 +61,6 @@ test_rebase_am_only () {
}

test_rebase_am_only --whitespace=fix
test_rebase_am_only --ignore-whitespace
test_rebase_am_only --committer-date-is-author-date
test_rebase_am_only -C4

Expand Down
65 changes: 65 additions & 0 deletions t/t3433-rebase-options-compatibility.sh
@@ -0,0 +1,65 @@
#!/bin/sh
#
# Copyright (c) 2019 Rohit Ashiwal
#

test_description='tests to ensure compatibility between am and interactive backends'

. ./test-lib.sh

# This is a special case in which both am and interactive backends
# provide the same output. It was done intentionally because
# both the backends fall short of optimal behaviour.
test_expect_success 'setup' '
git checkout -b topic &&
q_to_tab >file <<-\EOF &&
line 1
Qline 2
line 3
EOF
git add file &&
git commit -m "add file" &&
cat >file <<-\EOF &&
line 1
new line 2
line 3
EOF
git commit -am "update file" &&
git tag side &&
git checkout --orphan master &&
sed -e "s/^|//" >file <<-\EOF &&
|line 1
| line 2
|line 3
EOF
git add file &&
git commit -m "add file" &&
git tag main
'

test_expect_success '--ignore-whitespace works with am backend' '
cat >expect <<-\EOF &&
line 1
new line 2
line 3
EOF
test_must_fail git rebase main side &&
git rebase --abort &&
git rebase --ignore-whitespace main side &&
test_cmp expect file
'

test_expect_success '--ignore-whitespace works with interactive backend' '
cat >expect <<-\EOF &&
line 1
new line 2
line 3
EOF
test_must_fail git rebase --merge main side &&
git rebase --abort &&
git rebase --merge --ignore-whitespace main side &&
test_cmp expect file
'

test_done

0 comments on commit e191cc8

Please sign in to comment.