Recently git-status learned to display the state of the git sequencer in long form to help the user remember an interrupted command. This information is also useful in short form to humans and scripts, but no option is available to boil it down. Teach git-status to report the sequencer state in short form using a new --sequencer (-S) switch. Output zero or more simple state token strings indicating the deduced state of the git sequencer. Introduce a common function to determine the current sequencer state so the regular status function and this short version can share common code. Add a substate to wt_status_state to track more detailed information about a state, such as "conflicted" or "resolved". Move the am_empty_patch flage out of wt_status_state and into this new substate. State token strings which may be emitted and their meanings: merge a git-merge is in progress am a git-am is in progress rebase a git-rebase is in progress rebase-interactive a git-rebase--interactive is in progress cherry-pick a git-cherry-pick is in progress bisect a git-bisect is in progress conflicted there are unresolved conflicts resolved conflicts have been resolved editing interactive rebase stopped to edit a commit edited interactive rebase edit has been edited splitting interactive rebase, commit is being split I also considered adding these tokens, but I decided it was not appropriate since these changes are not sequencer-related. But it is possible I am being too short-sighted or have chosen the switch name poorly. clean index modified untracked --- Documentation/git-status.txt | 18 ++++++ builtin/commit.c | 12 +++- wt-status.c | 128 +++++++++++++++++++++++++++++++++++-------- wt-status.h | 13 ++++- 4 files changed, 146 insertions(+), 25 deletions(-) diff --git a/Documentation/git-status.txt b/Documentation/git-status.txt index 67e5f53..200a8e2 100644 --- a/Documentation/git-status.txt +++ b/Documentation/git-status.txt @@ -38,6 +38,24 @@ OPTIONS across git versions and regardless of user configuration. See below for details. +-S:: +--sequence:: + Show the git sequencer status. This shows zero or more tokens + describing the state of several git sequence operations. Each + token is separated by a newline. ++ + merge a git-merge is in progress + am a git-am is in progress + rebase a git-rebase is in progress + rebase-interactive a git-rebase--interactive is in progress + cherry-pick a git-cherry-pick is in progress + bisect a git-bisect is in progress + conflicted there are unresolved conflicts + resolved conflicts have been resolved + editing interactive rebase stopped to edit a commit + edited interactive rebase edit has been edited + splitting interactive rebase, commit is being split + -u[<mode>]:: --untracked-files[=<mode>]:: Show untracked files. diff --git a/builtin/commit.c b/builtin/commit.c index a17a5df..9706ed9 100644 --- a/builtin/commit.c +++ b/builtin/commit.c @@ -114,7 +114,8 @@ static struct strbuf message = STRBUF_INIT; static enum { STATUS_FORMAT_LONG, STATUS_FORMAT_SHORT, - STATUS_FORMAT_PORCELAIN + STATUS_FORMAT_PORCELAIN, + STATUS_FORMAT_SEQUENCER } status_format = STATUS_FORMAT_LONG; static int opt_parse_m(const struct option *opt, const char *arg, int unset) @@ -454,6 +455,9 @@ static int run_status(FILE *fp, const char *index_file, const char *prefix, int case STATUS_FORMAT_PORCELAIN: wt_porcelain_print(s); break; + case STATUS_FORMAT_SEQUENCER: + wt_sequencer_print(s); + break; case STATUS_FORMAT_LONG: wt_status_print(s); break; @@ -1156,6 +1160,9 @@ int cmd_status(int argc, const char **argv, const char *prefix) N_("show status concisely"), STATUS_FORMAT_SHORT), OPT_BOOLEAN('b', "branch", &s.show_branch, N_("show branch information")), + OPT_SET_INT('S', "sequence", &status_format, + N_("show sequencer state"), + STATUS_FORMAT_SEQUENCER), OPT_SET_INT(0, "porcelain", &status_format, N_("machine-readable output"), STATUS_FORMAT_PORCELAIN), @@ -1216,6 +1223,9 @@ int cmd_status(int argc, const char **argv, const char *prefix) case STATUS_FORMAT_PORCELAIN: wt_porcelain_print(&s); break; + case STATUS_FORMAT_SEQUENCER: + wt_sequencer_print(&s); + break; case STATUS_FORMAT_LONG: s.verbose = verbose; s.ignore_submodule_arg = ignore_submodule_arg; diff --git a/wt-status.c b/wt-status.c index 2a9658b..996b454 100644 --- a/wt-status.c +++ b/wt-status.c @@ -781,7 +781,7 @@ static void show_merge_in_progress(struct wt_status *s, struct wt_status_state *state, const char *color) { - if (has_unmerged(s)) { + if (state->substate == WT_SUBSTATE_CONFLICTED) { status_printf_ln(s, color, _("You have unmerged paths.")); if (advice_status_hints) status_printf_ln(s, color, @@ -802,11 +802,11 @@ static void show_am_in_progress(struct wt_status *s, { status_printf_ln(s, color, _("You are in the middle of an am session.")); - if (state->am_empty_patch) + if (state->substate==WT_SUBSTATE_NOMINAL) status_printf_ln(s, color, _("The current patch is empty.")); if (advice_status_hints) { - if (!state->am_empty_patch) + if (state->substate==WT_SUBSTATE_CONFLICTED) status_printf_ln(s, color, _(" (fix conflicts and then run \"git am --resolved\")")); status_printf_ln(s, color, @@ -867,9 +867,7 @@ static void show_rebase_in_progress(struct wt_status *s, struct wt_status_state *state, const char *color) { - struct stat st; - - if (has_unmerged(s)) { + if (state->substate == WT_SUBSTATE_CONFLICTED) { status_printf_ln(s, color, _("You are currently rebasing.")); if (advice_status_hints) { status_printf_ln(s, color, @@ -879,19 +877,19 @@ static void show_rebase_in_progress(struct wt_status *s, status_printf_ln(s, color, _(" (use \"git rebase --abort\" to check out the original branch)")); } - } else if (state->rebase_in_progress || !stat(git_path("MERGE_MSG"), &st)) { + } else if (state->substate == WT_SUBSTATE_RESOLVED) { status_printf_ln(s, color, _("You are currently rebasing.")); if (advice_status_hints) status_printf_ln(s, color, _(" (all conflicts fixed: run \"git rebase --continue\")")); - } else if (split_commit_in_progress(s)) { + } else if (state->substate == WT_SUBSTATE_SPLITTING) { status_printf_ln(s, color, _("You are currently splitting a commit during a rebase.")); if (advice_status_hints) status_printf_ln(s, color, _(" (Once your working directory is clean, run \"git rebase --continue\")")); } else { status_printf_ln(s, color, _("You are currently editing a commit during a rebase.")); - if (advice_status_hints && !s->amend) { + if (advice_status_hints && state->substate == WT_SUBSTATE_EDITING) { status_printf_ln(s, color, _(" (use \"git commit --amend\" to amend the current commit)")); status_printf_ln(s, color, @@ -907,7 +905,7 @@ static void show_cherry_pick_in_progress(struct wt_status *s, { status_printf_ln(s, color, _("You are currently cherry-picking.")); if (advice_status_hints) { - if (has_unmerged(s)) + if (state->substate == WT_SUBSTATE_CONFLICTED) status_printf_ln(s, color, _(" (fix conflicts and run \"git commit\")")); else @@ -928,34 +926,66 @@ static void show_bisect_in_progress(struct wt_status *s, wt_status_print_trailer(s); } -static void wt_status_print_state(struct wt_status *s) +static void wt_status_get_state(struct wt_status *s , struct wt_status_state *state) { - const char *state_color = color(WT_STATUS_HEADER, s); - struct wt_status_state state; struct stat st; - memset(&state, 0, sizeof(state)); + memset(state, 0, sizeof(*state)); + // Determine main sequencer activity if (!stat(git_path("MERGE_HEAD"), &st)) { - state.merge_in_progress = 1; + state->merge_in_progress = 1; } else if (!stat(git_path("rebase-apply"), &st)) { if (!stat(git_path("rebase-apply/applying"), &st)) { - state.am_in_progress = 1; - if (!stat(git_path("rebase-apply/patch"), &st) && !st.st_size) - state.am_empty_patch = 1; + state->am_in_progress = 1; + if (stat(git_path("rebase-apply/patch"), &st) || st.st_size) + state->substate = WT_SUBSTATE_CONFLICTED; } else { - state.rebase_in_progress = 1; + state->rebase_in_progress = 1; } } else if (!stat(git_path("rebase-merge"), &st)) { if (!stat(git_path("rebase-merge/interactive"), &st)) - state.rebase_interactive_in_progress = 1; + state->rebase_interactive_in_progress = 1; else - state.rebase_in_progress = 1; + state->rebase_in_progress = 1; } else if (!stat(git_path("CHERRY_PICK_HEAD"), &st)) { - state.cherry_pick_in_progress = 1; + state->cherry_pick_in_progress = 1; } if (!stat(git_path("BISECT_LOG"), &st)) - state.bisect_in_progress = 1; + state->bisect_in_progress = 1; + + // Check for unmerged files + if (state->rebase_in_progress || state->rebase_interactive_in_progress || + state->cherry_pick_in_progress || state->merge_in_progress) { + if (has_unmerged(s)) { + state->substate = WT_SUBSTATE_CONFLICTED; + } else { + state->substate = WT_SUBSTATE_RESOLVED; + } + } + + // Interactive Rebase is more nuanced + if (state->rebase_interactive_in_progress && state->substate != WT_SUBSTATE_CONFLICTED) { + if (!stat(git_path("MERGE_MSG"), &st)) { + state->substate = WT_SUBSTATE_RESOLVED; + } else if (split_commit_in_progress(s)) { + state->substate = WT_SUBSTATE_SPLITTING; + } else { + if (s->amend) { + state->substate = WT_SUBSTATE_EDITED; + } else { + state->substate = WT_SUBSTATE_EDITING; + } + } + } +} + +static void wt_status_print_state(struct wt_status *s) +{ + const char *state_color = color(WT_STATUS_HEADER, s); + struct wt_status_state state; + + wt_status_get_state(s, &state); if (state.merge_in_progress) show_merge_in_progress(s, &state, state_color); @@ -1192,6 +1222,53 @@ static void wt_shortstatus_print_tracking(struct wt_status *s) fputc(s->null_termination ? '\0' : '\n', s->fp); } +static void wt_print_token(struct wt_status *s, const char *color, const char *token) +{ + color_fprintf(s->fp, color, "%s", token); + fputc(s->null_termination ? '\0' : '\n', s->fp); +} + +static void wt_shortstatus_print_sequencer(struct wt_status *s) +{ + struct wt_status_state state; + const char *state_color = color(WT_STATUS_HEADER, s); + + wt_status_get_state(s, &state); + + if (state.merge_in_progress) + wt_print_token(s, state_color, "merge"); + if (state.am_in_progress) + wt_print_token(s, state_color, "am"); + if (state.rebase_in_progress) + wt_print_token(s, state_color, "rebase"); + if (state.rebase_interactive_in_progress) + wt_print_token(s, state_color, "rebase-interactive"); + if (state.cherry_pick_in_progress) + wt_print_token(s, state_color, "cherry-pick"); + if (state.bisect_in_progress) + wt_print_token(s, state_color, "bisect"); + + switch (state.substate) { + case WT_SUBSTATE_NOMINAL: + break; + case WT_SUBSTATE_CONFLICTED: + wt_print_token(s, state_color, "conflicted"); + break; + case WT_SUBSTATE_RESOLVED: + wt_print_token(s, state_color, "resolved"); + break; + case WT_SUBSTATE_EDITED: + wt_print_token(s, state_color, "edited"); + break; + case WT_SUBSTATE_EDITING: + wt_print_token(s, state_color, "editing"); + break; + case WT_SUBSTATE_SPLITTING: + wt_print_token(s, state_color, "splitting"); + break; + } +} + void wt_shortstatus_print(struct wt_status *s) { int i; @@ -1224,6 +1301,11 @@ void wt_shortstatus_print(struct wt_status *s) } } +void wt_sequencer_print(struct wt_status *s) +{ + wt_shortstatus_print_sequencer(s); +} + void wt_porcelain_print(struct wt_status *s) { s->use_color = 0; diff --git a/wt-status.h b/wt-status.h index 236b41f..3de4b26 100644 --- a/wt-status.h +++ b/wt-status.h @@ -59,6 +59,7 @@ struct wt_status { unsigned colopts; int null_termination; int show_branch; + int show_sequencer; /* These are computed during processing of the individual sections */ int commitable; @@ -71,14 +72,23 @@ struct wt_status { struct string_list ignored; }; +enum wt_status_substate { + WT_SUBSTATE_NOMINAL = 0, + WT_SUBSTATE_CONFLICTED, + WT_SUBSTATE_RESOLVED, + WT_SUBSTATE_SPLITTING, + WT_SUBSTATE_EDITING, + WT_SUBSTATE_EDITED, +}; + struct wt_status_state { int merge_in_progress; int am_in_progress; - int am_empty_patch; int rebase_in_progress; int rebase_interactive_in_progress; int cherry_pick_in_progress; int bisect_in_progress; + enum wt_status_substate substate; }; void wt_status_prepare(struct wt_status *s); @@ -86,6 +96,7 @@ void wt_status_print(struct wt_status *s); void wt_status_collect(struct wt_status *s); void wt_shortstatus_print(struct wt_status *s); +void wt_sequencer_print(struct wt_status *s); void wt_porcelain_print(struct wt_status *s); void status_printf_ln(struct wt_status *s, const char *color, const char *fmt, ...) -- 1.8.0.2.ge1a3bdd -- To unsubscribe from this list: send the line "unsubscribe git" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html