Re: [RFC/PATCH] log: add log.firstparent option

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Wed, Jul 22, 2015 at 6:23 PM, Jeff King <peff@xxxxxxxx> wrote:
> This patch adds an option to turn on --first-parent all the
> time, along with the corresponding --no-first-parent to
> disable it. The "why" of this requires a bit of backstory.
>
> Some projects (like git.git) encourage frequent rebasing to
> generate a set of clean, bisectable patches for each topic.
> The messy sequence of bug-ridden and bug-fixup commits is
> lost in the rebase, and not part of the final history.
>
> But other projects prefer to keep the messy history intact.
> For one thing, it makes collaboration on a topic easier, as
> developers can simply pull from each other during the messy
> development. And two, that history may later be useful when
> tracking down a bug, because it gives more insight into the
> actual thought process of the developer.
>
> But in this latter case you want _two_ views of history. You
> may want to see the "simple" version in which a series of
> fully-formed topics hit the branch (and you would like to
> see the diff of their final form). Or you may want to see
> the messy details, because you are digging into a bug
> related to the topic.
>
> One proposal we have seen in the past is to keep the messy
> history as a "shadow" parent of the real commits. That is,
> to introduce a new parent-like header into the commit
> object, but not traverse it by default in "git log". So it
> remains hidden until you ask to dig into a particular topic
> (presumably with a "log --show-messy-parents" option or
> similar). So by default you get the simple view, but can dig
> further if you wish.
>
> But we can observe that such shadow parents can be
> implemented as real parents; the problem isn't one of the
> underlying data structure, but how we present it in "git
> log". In other words, a perfectly reasonable workflow is:
>
>   - make your messy commits on a side branch
>
>   - do a non-fast-forward merge to bring them into master.
>     The commit message for this merge should be meaningful
>     and describe the topic as a whole.
>
>   - view the simple history with "git log --first-parent -m"
>
>   - view the complex history with "git log"
>
> But since you probably want to view the simple history most
> of the time, it would be nice to be able to default to that,
> and switch to the more complicated view with a command line
> option. Hence this patch.
>
> Suggested-by: Josh Bleecher Snyder <josharian@xxxxxxxxx>
> ---
> This came out of a discussion I had with Josh as OSCON. I
> don't think I would personally use it, because git.git is
> not a messy-workflow project. But I think that GitHub pushes
> people into this sort of workflow (the PR becomes the
> interesting unit of change), and my understanding is that
> Gerrit does, as well.

Github pull request messages
are similar to cover letters, so you could send a series with a
good cover letter, but crappy unfinished patches inside the series.
After applying all patches it could be all nice, i.e. compiles, tests, adds the
new functionality. It might be just a commit in between which may not even
compile. That's my understanding of the messy-workflow.

Gerrit cannot provide such a workflow easily as it's rather commit
centric and not branch centered. So you need to approve each
commit on its own and until 2 weeks ago you even needed to submit
each commit. (By now Gerrit has learned to submit a full branch, that
is you submit a commit and all its ancestors will integrate as well if they
were approved.)
Previously when the ancestors were not approved the commit would be
"submitted, merge pending", so it would wait for each commit to be approved
and submitted.
And because of this commit-centric workflow, the crappy commit in the series
is put into spot light.

Apart from that, the first-parent option gains some traction currently, Compare
https://git.eclipse.org/r/#/c/52381/ for example ;)

>
> There are probably some other things we (and others) could
> do to help support it:
>
>   - currently "--first-parent -p" needs "-m" to show
>     anything useful; this is being discussed elsewhere, and
>     it would be nice if it Just Worked (and showed the diff
>     between the merge and the first-parent)
>
>   - the commit messages for merges are often not great. A
>     few versions ago, I think, we started opening the editor
>     for merges, which is good. GitHub's PR-merge includes
>     the PR subject in the commit message, but not all of the
>     rationale and discussion. And in both git-generated and
>     GitHub-generated messages, the subject isn't amazing
>     (it's "merge topic jk/some-shorthand", which is barely
>     tolerable if you use good branch names; it could be
>     something like the subject-line of the cover letter for
>     the patch series).
>
>     So I think this could easily be improved by GitHub (we
>     have the PR subject and body, after all). It's harder
>     for a mailing list project like git.git, because Git
>     never actually sees the subject line. I think it would
>     require teaching git-am the concept of a patch series.

This would be cool I would imagine.

>
>     I don't know offhand what Gerrit merges look like.

Let's say it's complicated. Depending on configuration a few things
may happen. There are different integration strategies
* merge always
* merge if necessary (fastforward else)
* fastforward only
* rebase if necessary (to make it a linear history)
* cherrypick (which may add footers like Reviewed-by: to have that
information in the git history)

I think the "merge always" strategy would comfort from this patch.

>
>   - we already have merge.ff to default to making extra
>     merge commits. And if you use GitHub's UI to do the
>     merge, it uses --no-ff. I don't think we would want
>     these to become the default, so there's probably nothing
>     else to be done there.
>
> Signed-off-by: Jeff King <peff@xxxxxxxx>

The signoff is better placed above :)

> ---
>  Documentation/config.txt |  4 ++++
>  builtin/log.c            |  6 ++++++
>  revision.c               |  2 ++
>  t/t4202-log.sh           | 30 ++++++++++++++++++++++++++++++
>  4 files changed, 42 insertions(+)
>
> diff --git a/Documentation/config.txt b/Documentation/config.txt
> index 3e37b93..e9c3763 100644
> --- a/Documentation/config.txt
> +++ b/Documentation/config.txt
> @@ -1802,6 +1802,10 @@ log.mailmap::
>         If true, makes linkgit:git-log[1], linkgit:git-show[1], and
>         linkgit:git-whatchanged[1] assume `--use-mailmap`.
>
> +log.firstparent::
> +       If true, linkgit:git-log[1] will default to `--first-parent`;
> +       can be overridden by supplying `--no-first-parent`.
> +
>  mailinfo.scissors::
>         If true, makes linkgit:git-mailinfo[1] (and therefore
>         linkgit:git-am[1]) act by default as if the --scissors option
> diff --git a/builtin/log.c b/builtin/log.c
> index 8781049..3e9b034 100644
> --- a/builtin/log.c
> +++ b/builtin/log.c
> @@ -31,6 +31,7 @@ static const char *default_date_mode = NULL;
>
>  static int default_abbrev_commit;
>  static int default_show_root = 1;
> +static int default_first_parent;
>  static int decoration_style;
>  static int decoration_given;
>  static int use_mailmap_config;
> @@ -109,6 +110,7 @@ static void cmd_log_init_defaults(struct rev_info *rev)
>         rev->abbrev_commit = default_abbrev_commit;
>         rev->show_root_diff = default_show_root;
>         rev->subject_prefix = fmt_patch_subject_prefix;
> +       rev->first_parent_only = default_first_parent;
>         DIFF_OPT_SET(&rev->diffopt, ALLOW_TEXTCONV);
>
>         if (default_date_mode)
> @@ -396,6 +398,10 @@ static int git_log_config(const char *var, const char *value, void *cb)
>                 use_mailmap_config = git_config_bool(var, value);
>                 return 0;
>         }
> +       if (!strcmp(var, "log.firstparent")) {
> +               default_first_parent = git_config_bool(var, value);
> +               return 0;
> +       }
>
>         if (grep_config(var, value, cb) < 0)
>                 return -1;
> diff --git a/revision.c b/revision.c
> index ab97ffd..a03a84b 100644
> --- a/revision.c
> +++ b/revision.c
> @@ -1760,6 +1760,8 @@ static int handle_revision_opt(struct rev_info *revs, int argc, const char **arg
>                 return argcount;
>         } else if (!strcmp(arg, "--first-parent")) {
>                 revs->first_parent_only = 1;
> +       } else if (!strcmp(arg, "--no-first-parent")) {
> +               revs->first_parent_only = 0;
>         } else if (!strcmp(arg, "--ancestry-path")) {
>                 revs->ancestry_path = 1;
>                 revs->simplify_history = 0;
> diff --git a/t/t4202-log.sh b/t/t4202-log.sh
> index 1b2e981..de1c35d 100755
> --- a/t/t4202-log.sh
> +++ b/t/t4202-log.sh
> @@ -871,4 +871,34 @@ test_expect_success 'log --graph --no-walk is forbidden' '
>         test_must_fail git log --graph --no-walk
>  '
>
> +test_expect_success 'setup simple merge for first-parent tests' '
> +       git tag fp-base &&
> +       test_commit master &&
> +       git checkout -b fp-side &&
> +       test_commit side &&
> +       git checkout master &&
> +       git merge --no-ff fp-side
> +'
> +
> +test_expect_success 'log.firstparent config turns on first-parent' '
> +       test_config log.firstparent true &&
> +       cat >expect <<-\EOF &&
> +       Merge branch '\''fp-side'\''
> +       master
> +       EOF
> +       git log --format=%s fp-base.. >actual &&
> +       test_cmp expect actual
> +'
> +
> +test_expect_success 'log --no-first-parent override log.firstparent' '
> +       test_config log.firstparent true &&
> +       cat >expect <<-\EOF &&
> +       Merge branch '\''fp-side'\''
> +       side
> +       master
> +       EOF
> +       git log --no-first-parent --format=%s fp-base.. >actual &&
> +       test_cmp expect actual
> +'
> +
>  test_done
> --
> 2.5.0.rc2.540.ge5d4f14
> --
> 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
--
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



[Index of Archives]     [Linux Kernel Development]     [Gcc Help]     [IETF Annouce]     [DCCP]     [Netdev]     [Networking]     [Security]     [V4L]     [Bugtraq]     [Yosemite]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux SCSI]     [Fedora Users]