This allows cherry-picking a set of commits, some of which may be
redundant, without stopping to ask for the user intervention.

Signed-off-by: Giuseppe Bilotta <giuseppe.bilo...@gmail.com>
---
 Documentation/git-cherry-pick.txt |  4 ++++
 builtin/revert.c                  |  1 +
 sequencer.c                       | 45 +++++++++++++++++++++++++++++++--------
 sequencer.h                       |  1 +
 4 files changed, 42 insertions(+), 9 deletions(-)

diff --git a/Documentation/git-cherry-pick.txt 
b/Documentation/git-cherry-pick.txt
index d35d771fc8..ffced816d6 100644
--- a/Documentation/git-cherry-pick.txt
+++ b/Documentation/git-cherry-pick.txt
@@ -138,6 +138,10 @@ effect to your index in a row.
        examine the commit. This option overrides that behavior and
        creates an empty commit object.  Implies `--allow-empty`.
 
+--skip-empty::
+       This option causes empty and redundant cherry-picked commits to
+       be skipped without requesting the user intervention.
+
 --strategy=<strategy>::
        Use the given merge strategy.  Should only be used once.
        See the MERGE STRATEGIES section in linkgit:git-merge[1]
diff --git a/builtin/revert.c b/builtin/revert.c
index 4ca5b51544..ffdd367f99 100644
--- a/builtin/revert.c
+++ b/builtin/revert.c
@@ -102,6 +102,7 @@ static int run_sequencer(int argc, const char **argv, 
struct replay_opts *opts)
                        OPT_BOOL(0, "allow-empty", &opts->allow_empty, 
N_("preserve initially empty commits")),
                        OPT_BOOL(0, "allow-empty-message", 
&opts->allow_empty_message, N_("allow commits with empty messages")),
                        OPT_BOOL(0, "keep-redundant-commits", 
&opts->keep_redundant_commits, N_("keep redundant, empty commits")),
+                       OPT_BOOL(0, "skip-empty", &opts->skip_empty, N_("skip 
redundant, empty commits")),
                        OPT_END(),
                };
                options = parse_options_concat(options, cp_extra);
diff --git a/sequencer.c b/sequencer.c
index 3d2f61c979..9c01310162 100644
--- a/sequencer.c
+++ b/sequencer.c
@@ -550,22 +550,32 @@ static int is_original_commit_empty(struct commit *commit)
 
 /*
  * Do we run "git commit" with "--allow-empty"?
+ *
+ * Or do we just skip this empty commit?
+ *
+ * Returns 1 if a commit should be done with --allow-empty,
+ *         0 if a commit should be done without --allow-empty,
+ *         2 if no commit should be done at all (skip empty commit)
+ *         negative values in case of error
+ *
  */
-static int allow_empty(struct replay_opts *opts, struct commit *commit)
+static int allow_or_skip_empty(struct replay_opts *opts, struct commit *commit)
 {
        int index_unchanged, empty_commit;
 
        /*
-        * Three cases:
+        * Four cases:
         *
-        * (1) we do not allow empty at all and error out.
+        * (1) we do not allow empty at all and error out;
         *
-        * (2) we allow ones that were initially empty, but
+        * (2) we skip empty commits altogether;
+        *
+        * (3) we allow ones that were initially empty, but
         * forbid the ones that become empty;
         *
-        * (3) we allow both.
+        * (4) we allow both.
         */
-       if (!opts->allow_empty)
+       if (!opts->allow_empty && !opts->skip_empty)
                return 0; /* let "git commit" barf as necessary */
 
        index_unchanged = is_index_unchanged();
@@ -574,6 +584,9 @@ static int allow_empty(struct replay_opts *opts, struct 
commit *commit)
        if (!index_unchanged)
                return 0; /* we do not have to say --allow-empty */
 
+       if (opts->skip_empty)
+               return 2;
+
        if (opts->keep_redundant_commits)
                return 1;
 
@@ -612,7 +625,7 @@ static int do_pick_commit(enum todo_command command, struct 
commit *commit,
        const char *base_label, *next_label;
        struct commit_message msg = { NULL, NULL, NULL, NULL };
        struct strbuf msgbuf = STRBUF_INIT;
-       int res, unborn = 0, allow;
+       int res = 0, unborn = 0, allow;
 
        if (opts->no_commit) {
                /*
@@ -771,12 +784,13 @@ static int do_pick_commit(enum todo_command command, 
struct commit *commit,
                goto leave;
        }
 
-       allow = allow_empty(opts, commit);
+       allow = allow_or_skip_empty(opts, commit);
        if (allow < 0) {
                res = allow;
                goto leave;
        }
-       if (!opts->no_commit)
+       /* allow == 2 means skip this commit */
+       if (allow != 2 && !opts->no_commit)
                res = run_git_commit(opts->edit ? NULL : git_path_merge_msg(),
                                     opts, allow, opts->edit, 0, 0);
 
@@ -993,6 +1007,8 @@ static int populate_opts_cb(const char *key, const char 
*value, void *data)
                opts->allow_empty_message = git_config_bool_or_int(key, value, 
&error_flag);
        else if (!strcmp(key, "options.keep-redundant-commits"))
                opts->keep_redundant_commits = git_config_bool_or_int(key, 
value, &error_flag);
+       else if (!strcmp(key, "options.skip-empty"))
+               opts->skip_empty = git_config_bool_or_int(key, value, 
&error_flag);
        else if (!strcmp(key, "options.mainline"))
                opts->mainline = git_config_int(key, value);
        else if (!strcmp(key, "options.gpg-sign"))
@@ -1249,6 +1265,8 @@ static int save_opts(struct replay_opts *opts)
                res |= git_config_set_in_file_gently(opts_file, 
"options.allow-empty-message", "true");
        if (opts->keep_redundant_commits)
                res |= git_config_set_in_file_gently(opts_file, 
"options.keep-redundant-commits", "true");
+       if (opts->skip_empty)
+               res |= git_config_set_in_file_gently(opts_file, 
"options.skip-empty", "true");
        if (opts->mainline) {
                struct strbuf buf = STRBUF_INIT;
                strbuf_addf(&buf, "%d", opts->mainline);
@@ -1322,6 +1340,14 @@ int sequencer_continue(struct replay_opts *opts)
        if ((res = read_populate_todo(&todo_list, opts)))
                goto release_todo_list;
 
+       /* check if there is something to commit */
+       res = is_index_unchanged();
+       if (res < 0)
+               goto release_todo_list;
+
+       if (res && opts->skip_empty)
+               goto skip_this_commit;
+
        /* Verify that the conflict has been resolved */
        if (file_exists(git_path_cherry_pick_head()) ||
            file_exists(git_path_revert_head())) {
@@ -1333,6 +1359,7 @@ int sequencer_continue(struct replay_opts *opts)
                res = error_dirty_index(opts);
                goto release_todo_list;
        }
+skip_this_commit:
        todo_list.current++;
        res = pick_commits(&todo_list, opts);
 release_todo_list:
diff --git a/sequencer.h b/sequencer.h
index 7a513c576b..c747cfcfc7 100644
--- a/sequencer.h
+++ b/sequencer.h
@@ -23,6 +23,7 @@ struct replay_opts {
        int allow_empty;
        int allow_empty_message;
        int keep_redundant_commits;
+       int skip_empty;
 
        int mainline;
 
-- 
2.11.0.616.gd72966cf44.dirty

Reply via email to