From: Jeff King <peff@peff.net>
To: Dana Dahlstrom <dahlstrom@google.com>
Cc: git@vger.kernel.org
Subject: Re: 'HEAD' is not a commit (according to git-checkout)
Date: Thu, 21 May 2020 15:16:26 -0400 [thread overview]
Message-ID: <20200521191626.GC1308489@coredump.intra.peff.net> (raw)
In-Reply-To: <CACqwCQiLpZ1HFzgJw0p0KR3jXNsxkhjXmF_huzhv+qkMZmybBQ@mail.gmail.com>
On Thu, May 21, 2020 at 12:00:00PM -0700, Dana Dahlstrom wrote:
> What did you do before the bug happened? (Steps to reproduce your issue)
>
> $ git clone https://github.com/githubtraining/hellogitworld.git
> $ cd hellogitworld
> $ git checkout -b work -t master HEAD
> fatal: 'HEAD' is not a commit and a branch 'work' cannot be created from it
> $ git show -s --oneline
> ef7bebf (HEAD -> master, origin/master, origin/HEAD) Fix groupId […]
> $ git checkout -b work -t master ef7bebf
> fatal: 'ef7bebf' is not a commit and a branch 'work' cannot be created from it
Thanks for a complete reproduction. There are a few things going on
here.
The "-t" option doesn't take an argument; it's a boolean that says
"track the branch we started from". So "master" is taken as the
start-point, and "HEAD" is tacked onto the end. I.e., equivalent to:
git checkout -b work master HEAD
That error message is wrong and misleading. It looks like what happens
is that we parse "master" as the start-point. And then we try to treat
remaining options (i.e., "HEAD") as a pathspec. That fails, because
there's no such path. But then instead of saying "hey, HEAD isn't a
pathspec" we try to be clever:
/*
* Try to give more helpful suggestion.
* new_branch && argc > 1 will be caught later.
*/
if (opts->new_branch && argc == 1)
die(_("'%s' is not a commit and a branch '%s' cannot be created from it"),
argv[0], opts->new_branch);
We know we're making a new branch and there's one argument, so we assume
that it didn't get parsed earlier as a start-point. But that misses the
fact that if we _did_ parse a start-point, it would have been removed
from argv, and our "single argument" is actually the former
second-argument.
Something like this works:
diff --git a/builtin/checkout.c b/builtin/checkout.c
index e9d111bb83..6559ac666b 100644
--- a/builtin/checkout.c
+++ b/builtin/checkout.c
@@ -1553,6 +1553,7 @@ static int checkout_main(int argc, const char **argv, const char *prefix,
{
struct branch_info new_branch_info;
int parseopt_flags = 0;
+ int got_start_point = 0;
memset(&new_branch_info, 0, sizeof(new_branch_info));
opts->overwrite_ignore = 1;
@@ -1661,6 +1662,8 @@ static int checkout_main(int argc, const char **argv, const char *prefix,
!opts->new_branch;
int n = parse_branchname_arg(argc, argv, dwim_ok,
&new_branch_info, opts, &rev);
+ if (n)
+ got_start_point = 1;
argv += n;
argc -= n;
} else if (!opts->accept_ref && opts->from_treeish) {
@@ -1689,7 +1692,7 @@ static int checkout_main(int argc, const char **argv, const char *prefix,
* Try to give more helpful suggestion.
* new_branch && argc > 1 will be caught later.
*/
- if (opts->new_branch && argc == 1)
+ if (opts->new_branch && !got_start_point && argc == 1)
die(_("'%s' is not a commit and a branch '%s' cannot be created from it"),
argv[0], opts->new_branch);
to produce:
$ git checkout -b work -t master HEAD
fatal: '--track' cannot be used with updating paths
$ git checkout -b work master HEAD
fatal: Cannot update paths and switch to branch 'work' at the same time.
which are both correct. I wonder if there's a more elegant way to do it,
though (probably not, as there's definitely some heuristic parsing going
on to determine which checkout mode we're in; the new switch/restore
alternatives don't suffer as much from this).
> What did you expect to happen? (Expected behavior)
>
> I expected a new branch named 'work' to be created and checked out,
> pointing to commit ef7bebf and with upstream branch set to 'master'.
So getting back to your actual goal: you can't do what you want with a
single checkout command. I think:
git checkout -b work HEAD
git branch --set-upstream-to=master
-Peff
next prev parent reply other threads:[~2020-05-21 19:16 UTC|newest]
Thread overview: 16+ messages / expand[flat|nested] mbox.gz Atom feed top
2020-05-21 19:00 'HEAD' is not a commit (according to git-checkout) Dana Dahlstrom
2020-05-21 19:16 ` Jeff King [this message]
2020-05-23 7:07 ` René Scharfe
2020-05-23 16:29 ` Jeff King
2020-05-24 7:22 ` [PATCH 1/2] checkout: add tests for -b and --track René Scharfe
2020-05-27 6:40 ` Jeff King
2020-05-28 13:53 ` René Scharfe
2020-05-24 7:23 ` [PATCH 2/2] checkout: improve error messages for -b with extra argument René Scharfe
2020-05-27 6:42 ` Jeff King
2020-05-24 7:23 ` 'HEAD' is not a commit (according to git-checkout) René Scharfe
2020-05-24 16:15 ` Junio C Hamano
2020-05-27 6:52 ` Jeff King
2020-05-27 15:44 ` Junio C Hamano
2020-05-27 15:52 ` Randall S. Becker
2020-05-27 17:31 ` Jeff King
2020-05-21 19:49 ` René Scharfe
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20200521191626.GC1308489@coredump.intra.peff.net \
--to=peff@peff.net \
--cc=dahlstrom@google.com \
--cc=git@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).