* Since 2.7.0 git branch displays symbolic references as ref -> ref instead of ref -> branch
@ 2016-04-03 2:54 Phil Sainty
2016-04-03 4:14 ` [PATCH] branch: fix shortening of non-remote symrefs Jeff King
0 siblings, 1 reply; 3+ messages in thread
From: Phil Sainty @ 2016-04-03 2:54 UTC (permalink / raw)
To: git
Given the following symbolic reference:
$ git symbolic-ref refs/heads/m refs/heads/master
Correct in 2.6.6:
$ PATH=~/git/git-2.6.6:$PATH git branch
m -> master
* master
Wrong in 2.7.0:
$ PATH=~/git/git-2.7.0:$PATH git branch
m -> m
* master
Still wrong in current version 2.8.0:
$ PATH=~/git/git-2.8.0:$PATH git branch
m -> m
* master
As the new output isn't very useful, I assume this is an
unintentional bug/regression.
The 2.6.6 output has been used since at least version 1.7.12.4
(when I first started making use of this ability).
-Phil
^ permalink raw reply [flat|nested] 3+ messages in thread
* [PATCH] branch: fix shortening of non-remote symrefs
2016-04-03 2:54 Since 2.7.0 git branch displays symbolic references as ref -> ref instead of ref -> branch Phil Sainty
@ 2016-04-03 4:14 ` Jeff King
2016-04-05 6:21 ` Karthik Nayak
0 siblings, 1 reply; 3+ messages in thread
From: Jeff King @ 2016-04-03 4:14 UTC (permalink / raw)
To: Phil Sainty; +Cc: Karthik Nayak, Junio C Hamano, git
On Sun, Apr 03, 2016 at 02:54:22PM +1200, Phil Sainty wrote:
> Given the following symbolic reference:
>
> $ git symbolic-ref refs/heads/m refs/heads/master
>
>
> Correct in 2.6.6:
>
> $ PATH=~/git/git-2.6.6:$PATH git branch
> m -> master
> * master
>
>
> Wrong in 2.7.0:
>
> $ PATH=~/git/git-2.7.0:$PATH git branch
> m -> m
> * master
Thanks for an easy test case. Though we don't officially support
arbitrary symrefs in the ref namespace, they do mostly work. And
certainly the current output is nonsense, and it worked before. This
bisects to aedcb7d (branch.c: use 'ref-filter' APIs, 2015-09-23).
The fix is below. Karthik, I didn't look at all how this interacts with
your work to convert branch to ref-filter for printing. I imagine it
drops this code completely, but we should make sure that ref-filter gets
this case right. I almost didn't prepare this patch at all, but I
suspect we may want it for "maint", while the full conversion would wait
for "master".
-- >8 --
Subject: branch: fix shortening of non-remote symrefs
Commit aedcb7d (branch.c: use 'ref-filter' APIs, 2015-09-23)
adjusted the symref-printing code to look like this:
if (item->symref) {
skip_prefix(item->symref, "refs/remotes/", &desc);
strbuf_addf(&out, " -> %s", desc);
}
This has three bugs in it:
1. It always skips past "refs/remotes/", instead of
skipping past the prefix associated with the branch we
are showing (so commonly we see "refs/remotes/" for the
refs/remotes/origin/HEAD symref, but the previous code
would skip "refs/heads/" when showing a symref it found
in refs/heads/.
2. If skip_prefix() does not match, it leaves "desc"
untouched, and we show whatever happened to be in it
(which is the refname from a call to skip_prefix()
earlier in the function).
3. If we do match with skip_prefix(), we stomp on the
"desc" variable, which is later passed to
add_verbose_info(). We probably want to retain the
original refname there (though it likely doesn't matter
in practice, since after all, one points to the other).
The fix to match the original code is fairly easy: record
the prefix to strip based on item->kind, and use it here.
However, since we already have a local variable named "prefix",
let's give the two prefixes verbose names so we don't
confuse them.
Signed-off-by: Jeff King <peff@peff.net>
---
The test makes sure we restored the v2.6.x behavior, namely that
cross-prefix symrefs will not be shortened at all. It might be nice to
show:
ref-to-remote -> remotes/origin/branch-one
or something, but that should be separate from the fix (and I don't
overly care either way, so I probably won't work on it).
builtin/branch.c | 19 ++++++++++++-------
t/t3203-branch-output.sh | 12 ++++++++++++
2 files changed, 24 insertions(+), 7 deletions(-)
diff --git a/builtin/branch.c b/builtin/branch.c
index 7b45b6b..f6c23bf 100644
--- a/builtin/branch.c
+++ b/builtin/branch.c
@@ -393,22 +393,25 @@ static void format_and_print_ref_item(struct ref_array_item *item, int maxwidth,
int current = 0;
int color;
struct strbuf out = STRBUF_INIT, name = STRBUF_INIT;
- const char *prefix = "";
+ const char *prefix_to_show = "";
+ const char *prefix_to_skip = NULL;
const char *desc = item->refname;
char *to_free = NULL;
switch (item->kind) {
case FILTER_REFS_BRANCHES:
- skip_prefix(desc, "refs/heads/", &desc);
+ prefix_to_skip = "refs/heads/";
+ skip_prefix(desc, prefix_to_skip, &desc);
if (!filter->detached && !strcmp(desc, head))
current = 1;
else
color = BRANCH_COLOR_LOCAL;
break;
case FILTER_REFS_REMOTES:
- skip_prefix(desc, "refs/remotes/", &desc);
+ prefix_to_skip = "refs/remotes/";
+ skip_prefix(desc, prefix_to_skip, &desc);
color = BRANCH_COLOR_REMOTE;
- prefix = remote_prefix;
+ prefix_to_show = remote_prefix;
break;
case FILTER_REFS_DETACHED_HEAD:
desc = to_free = get_head_description();
@@ -425,7 +428,7 @@ static void format_and_print_ref_item(struct ref_array_item *item, int maxwidth,
color = BRANCH_COLOR_CURRENT;
}
- strbuf_addf(&name, "%s%s", prefix, desc);
+ strbuf_addf(&name, "%s%s", prefix_to_show, desc);
if (filter->verbose) {
int utf8_compensation = strlen(name.buf) - utf8_strwidth(name.buf);
strbuf_addf(&out, "%c %s%-*s%s", c, branch_get_color(color),
@@ -436,8 +439,10 @@ static void format_and_print_ref_item(struct ref_array_item *item, int maxwidth,
name.buf, branch_get_color(BRANCH_COLOR_RESET));
if (item->symref) {
- skip_prefix(item->symref, "refs/remotes/", &desc);
- strbuf_addf(&out, " -> %s", desc);
+ const char *symref = item->symref;
+ if (prefix_to_skip)
+ skip_prefix(symref, prefix_to_skip, &symref);
+ strbuf_addf(&out, " -> %s", symref);
}
else if (filter->verbose)
/* " f7c0c00 [ahead 58, behind 197] vcs-svn: drop obj_pool.h" */
diff --git a/t/t3203-branch-output.sh b/t/t3203-branch-output.sh
index 4261403..c6a3ccb 100755
--- a/t/t3203-branch-output.sh
+++ b/t/t3203-branch-output.sh
@@ -184,4 +184,16 @@ test_expect_success 'ambiguous branch/tag not marked' '
test_cmp expect actual
'
+test_expect_success 'local-branch symrefs shortened properly' '
+ git symbolic-ref refs/heads/ref-to-branch refs/heads/branch-one &&
+ git symbolic-ref refs/heads/ref-to-remote refs/remotes/origin/branch-one &&
+ cat >expect <<-\EOF &&
+ ref-to-branch -> branch-one
+ ref-to-remote -> refs/remotes/origin/branch-one
+ EOF
+ git branch >actual.raw &&
+ grep ref-to <actual.raw >actual &&
+ test_cmp expect actual
+'
+
test_done
--
2.8.0.429.gaaf8de0
^ permalink raw reply related [flat|nested] 3+ messages in thread
* Re: [PATCH] branch: fix shortening of non-remote symrefs
2016-04-03 4:14 ` [PATCH] branch: fix shortening of non-remote symrefs Jeff King
@ 2016-04-05 6:21 ` Karthik Nayak
0 siblings, 0 replies; 3+ messages in thread
From: Karthik Nayak @ 2016-04-05 6:21 UTC (permalink / raw)
To: Jeff King; +Cc: Phil Sainty, Junio C Hamano, Git
Hello,
On Sun, Apr 3, 2016 at 9:44 AM, Jeff King <peff@peff.net> wrote:
> On Sun, Apr 03, 2016 at 02:54:22PM +1200, Phil Sainty wrote:
>
>> Given the following symbolic reference:
>>
>> $ git symbolic-ref refs/heads/m refs/heads/master
>>
>>
>> Correct in 2.6.6:
>>
>> $ PATH=~/git/git-2.6.6:$PATH git branch
>> m -> master
>> * master
>>
>>
>> Wrong in 2.7.0:
>>
>> $ PATH=~/git/git-2.7.0:$PATH git branch
>> m -> m
>> * master
Thanks for reporting this.
>
> Thanks for an easy test case. Though we don't officially support
> arbitrary symrefs in the ref namespace, they do mostly work. And
> certainly the current output is nonsense, and it worked before. This
> bisects to aedcb7d (branch.c: use 'ref-filter' APIs, 2015-09-23).
>
> The fix is below. Karthik, I didn't look at all how this interacts with
> your work to convert branch to ref-filter for printing. I imagine it
> drops this code completely, but we should make sure that ref-filter gets
> this case right. I almost didn't prepare this patch at all, but I
> suspect we may want it for "maint", while the full conversion would wait
> for "master".
>
It's dropped in my latest series. I should be able to replicate what you've done
here onto ref-filter.c. Since I'm re-rolling my patches, I'll add this
one along too.
> -- >8 --
> Subject: branch: fix shortening of non-remote symrefs
>
> Commit aedcb7d (branch.c: use 'ref-filter' APIs, 2015-09-23)
> adjusted the symref-printing code to look like this:
>
> if (item->symref) {
> skip_prefix(item->symref, "refs/remotes/", &desc);
> strbuf_addf(&out, " -> %s", desc);
> }
>
> This has three bugs in it:
>
> 1. It always skips past "refs/remotes/", instead of
> skipping past the prefix associated with the branch we
> are showing (so commonly we see "refs/remotes/" for the
> refs/remotes/origin/HEAD symref, but the previous code
> would skip "refs/heads/" when showing a symref it found
> in refs/heads/.
>
> 2. If skip_prefix() does not match, it leaves "desc"
> untouched, and we show whatever happened to be in it
> (which is the refname from a call to skip_prefix()
> earlier in the function).
>
> 3. If we do match with skip_prefix(), we stomp on the
> "desc" variable, which is later passed to
> add_verbose_info(). We probably want to retain the
> original refname there (though it likely doesn't matter
> in practice, since after all, one points to the other).
>
> The fix to match the original code is fairly easy: record
> the prefix to strip based on item->kind, and use it here.
> However, since we already have a local variable named "prefix",
> let's give the two prefixes verbose names so we don't
> confuse them.
>
> Signed-off-by: Jeff King <peff@peff.net>
> ---
> The test makes sure we restored the v2.6.x behavior, namely that
> cross-prefix symrefs will not be shortened at all. It might be nice to
> show:
>
> ref-to-remote -> remotes/origin/branch-one
>
> or something, but that should be separate from the fix (and I don't
> overly care either way, so I probably won't work on it).
>
> builtin/branch.c | 19 ++++++++++++-------
> t/t3203-branch-output.sh | 12 ++++++++++++
> 2 files changed, 24 insertions(+), 7 deletions(-)
>
> diff --git a/builtin/branch.c b/builtin/branch.c
> index 7b45b6b..f6c23bf 100644
> --- a/builtin/branch.c
> +++ b/builtin/branch.c
> @@ -393,22 +393,25 @@ static void format_and_print_ref_item(struct ref_array_item *item, int maxwidth,
> int current = 0;
> int color;
> struct strbuf out = STRBUF_INIT, name = STRBUF_INIT;
> - const char *prefix = "";
> + const char *prefix_to_show = "";
> + const char *prefix_to_skip = NULL;
> const char *desc = item->refname;
> char *to_free = NULL;
>
> switch (item->kind) {
> case FILTER_REFS_BRANCHES:
> - skip_prefix(desc, "refs/heads/", &desc);
> + prefix_to_skip = "refs/heads/";
> + skip_prefix(desc, prefix_to_skip, &desc);
> if (!filter->detached && !strcmp(desc, head))
> current = 1;
> else
> color = BRANCH_COLOR_LOCAL;
> break;
> case FILTER_REFS_REMOTES:
> - skip_prefix(desc, "refs/remotes/", &desc);
> + prefix_to_skip = "refs/remotes/";
> + skip_prefix(desc, prefix_to_skip, &desc);
> color = BRANCH_COLOR_REMOTE;
> - prefix = remote_prefix;
> + prefix_to_show = remote_prefix;
> break;
> case FILTER_REFS_DETACHED_HEAD:
> desc = to_free = get_head_description();
> @@ -425,7 +428,7 @@ static void format_and_print_ref_item(struct ref_array_item *item, int maxwidth,
> color = BRANCH_COLOR_CURRENT;
> }
>
> - strbuf_addf(&name, "%s%s", prefix, desc);
> + strbuf_addf(&name, "%s%s", prefix_to_show, desc);
> if (filter->verbose) {
> int utf8_compensation = strlen(name.buf) - utf8_strwidth(name.buf);
> strbuf_addf(&out, "%c %s%-*s%s", c, branch_get_color(color),
> @@ -436,8 +439,10 @@ static void format_and_print_ref_item(struct ref_array_item *item, int maxwidth,
> name.buf, branch_get_color(BRANCH_COLOR_RESET));
>
> if (item->symref) {
> - skip_prefix(item->symref, "refs/remotes/", &desc);
> - strbuf_addf(&out, " -> %s", desc);
> + const char *symref = item->symref;
> + if (prefix_to_skip)
> + skip_prefix(symref, prefix_to_skip, &symref);
> + strbuf_addf(&out, " -> %s", symref);
> }
> else if (filter->verbose)
> /* " f7c0c00 [ahead 58, behind 197] vcs-svn: drop obj_pool.h" */
> diff --git a/t/t3203-branch-output.sh b/t/t3203-branch-output.sh
> index 4261403..c6a3ccb 100755
> --- a/t/t3203-branch-output.sh
> +++ b/t/t3203-branch-output.sh
> @@ -184,4 +184,16 @@ test_expect_success 'ambiguous branch/tag not marked' '
> test_cmp expect actual
> '
>
> +test_expect_success 'local-branch symrefs shortened properly' '
> + git symbolic-ref refs/heads/ref-to-branch refs/heads/branch-one &&
> + git symbolic-ref refs/heads/ref-to-remote refs/remotes/origin/branch-one &&
> + cat >expect <<-\EOF &&
> + ref-to-branch -> branch-one
> + ref-to-remote -> refs/remotes/origin/branch-one
> + EOF
> + git branch >actual.raw &&
> + grep ref-to <actual.raw >actual &&
> + test_cmp expect actual
> +'
> +
> test_done
> --
> 2.8.0.429.gaaf8de0
>
This seems to fix the bug very easily and effectively. Thanks for the patch :)
--
Regards,
Karthik Nayak
^ permalink raw reply [flat|nested] 3+ messages in thread
end of thread, other threads:[~2016-04-05 6:21 UTC | newest]
Thread overview: 3+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2016-04-03 2:54 Since 2.7.0 git branch displays symbolic references as ref -> ref instead of ref -> branch Phil Sainty
2016-04-03 4:14 ` [PATCH] branch: fix shortening of non-remote symrefs Jeff King
2016-04-05 6:21 ` Karthik Nayak
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).