for-each-ref: introduce %(upstream:track[short])
Introduce %(upstream:track) to display "[ahead M, behind N]" and %(upstream:trackshort) to display "=", ">", "<", or "<>" appropriately (inspired by contrib/completion/git-prompt.sh). Now you can use the following format in for-each-ref: %(refname:short)%(upstream:trackshort) to display refs with terse tracking information. Note that :track and :trackshort only work with "upstream", and error out when used with anything else. Signed-off-by: Ramkumar Ramachandra <artagnon@gmail.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
7a48b83219
commit
b28061ce0d
@ -91,7 +91,11 @@ objectname::
|
|||||||
upstream::
|
upstream::
|
||||||
The name of a local ref which can be considered ``upstream''
|
The name of a local ref which can be considered ``upstream''
|
||||||
from the displayed ref. Respects `:short` in the same way as
|
from the displayed ref. Respects `:short` in the same way as
|
||||||
`refname` above.
|
`refname` above. Additionally respects `:track` to show
|
||||||
|
"[ahead N, behind M]" and `:trackshort` to show the terse
|
||||||
|
version: ">" (ahead), "<" (behind), "<>" (ahead and behind),
|
||||||
|
or "=" (in sync). Has no effect if the ref does not have
|
||||||
|
tracking information associated with it.
|
||||||
|
|
||||||
HEAD::
|
HEAD::
|
||||||
'*' if HEAD matches current ref (the checked out branch), ' '
|
'*' if HEAD matches current ref (the checked out branch), ' '
|
||||||
|
@ -641,6 +641,7 @@ static void populate_value(struct refinfo *ref)
|
|||||||
int deref = 0;
|
int deref = 0;
|
||||||
const char *refname;
|
const char *refname;
|
||||||
const char *formatp;
|
const char *formatp;
|
||||||
|
struct branch *branch = NULL;
|
||||||
|
|
||||||
if (*name == '*') {
|
if (*name == '*') {
|
||||||
deref = 1;
|
deref = 1;
|
||||||
@ -652,7 +653,6 @@ static void populate_value(struct refinfo *ref)
|
|||||||
else if (!prefixcmp(name, "symref"))
|
else if (!prefixcmp(name, "symref"))
|
||||||
refname = ref->symref ? ref->symref : "";
|
refname = ref->symref ? ref->symref : "";
|
||||||
else if (!prefixcmp(name, "upstream")) {
|
else if (!prefixcmp(name, "upstream")) {
|
||||||
struct branch *branch;
|
|
||||||
/* only local branches may have an upstream */
|
/* only local branches may have an upstream */
|
||||||
if (prefixcmp(ref->refname, "refs/heads/"))
|
if (prefixcmp(ref->refname, "refs/heads/"))
|
||||||
continue;
|
continue;
|
||||||
@ -679,6 +679,7 @@ static void populate_value(struct refinfo *ref)
|
|||||||
} else if (!strcmp(name, "HEAD")) {
|
} else if (!strcmp(name, "HEAD")) {
|
||||||
const char *head;
|
const char *head;
|
||||||
unsigned char sha1[20];
|
unsigned char sha1[20];
|
||||||
|
|
||||||
head = resolve_ref_unsafe("HEAD", sha1, 1, NULL);
|
head = resolve_ref_unsafe("HEAD", sha1, 1, NULL);
|
||||||
if (!strcmp(ref->refname, head))
|
if (!strcmp(ref->refname, head))
|
||||||
v->s = "*";
|
v->s = "*";
|
||||||
@ -689,13 +690,46 @@ static void populate_value(struct refinfo *ref)
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
formatp = strchr(name, ':');
|
formatp = strchr(name, ':');
|
||||||
/* look for "short" refname format */
|
|
||||||
if (formatp) {
|
if (formatp) {
|
||||||
|
int num_ours, num_theirs;
|
||||||
|
|
||||||
formatp++;
|
formatp++;
|
||||||
if (!strcmp(formatp, "short"))
|
if (!strcmp(formatp, "short"))
|
||||||
refname = shorten_unambiguous_ref(refname,
|
refname = shorten_unambiguous_ref(refname,
|
||||||
warn_ambiguous_refs);
|
warn_ambiguous_refs);
|
||||||
|
else if (!strcmp(formatp, "track") &&
|
||||||
|
!prefixcmp(name, "upstream")) {
|
||||||
|
char buf[40];
|
||||||
|
|
||||||
|
stat_tracking_info(branch, &num_ours, &num_theirs);
|
||||||
|
if (!num_ours && !num_theirs)
|
||||||
|
v->s = "";
|
||||||
|
else if (!num_ours) {
|
||||||
|
sprintf(buf, "[behind %d]", num_theirs);
|
||||||
|
v->s = xstrdup(buf);
|
||||||
|
} else if (!num_theirs) {
|
||||||
|
sprintf(buf, "[ahead %d]", num_ours);
|
||||||
|
v->s = xstrdup(buf);
|
||||||
|
} else {
|
||||||
|
sprintf(buf, "[ahead %d, behind %d]",
|
||||||
|
num_ours, num_theirs);
|
||||||
|
v->s = xstrdup(buf);
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
} else if (!strcmp(formatp, "trackshort") &&
|
||||||
|
!prefixcmp(name, "upstream")) {
|
||||||
|
assert(branch);
|
||||||
|
stat_tracking_info(branch, &num_ours, &num_theirs);
|
||||||
|
if (!num_ours && !num_theirs)
|
||||||
|
v->s = "=";
|
||||||
|
else if (!num_ours)
|
||||||
|
v->s = "<";
|
||||||
|
else if (!num_theirs)
|
||||||
|
v->s = ">";
|
||||||
else
|
else
|
||||||
|
v->s = "<>";
|
||||||
|
continue;
|
||||||
|
} else
|
||||||
die("unknown %.*s format %s",
|
die("unknown %.*s format %s",
|
||||||
(int)(formatp - name), name, formatp);
|
(int)(formatp - name), name, formatp);
|
||||||
}
|
}
|
||||||
|
@ -303,6 +303,33 @@ test_expect_success 'Check short upstream format' '
|
|||||||
test_cmp expected actual
|
test_cmp expected actual
|
||||||
'
|
'
|
||||||
|
|
||||||
|
test_expect_success 'setup for upstream:track[short]' '
|
||||||
|
test_commit two
|
||||||
|
'
|
||||||
|
|
||||||
|
cat >expected <<EOF
|
||||||
|
[ahead 1]
|
||||||
|
EOF
|
||||||
|
|
||||||
|
test_expect_success 'Check upstream:track format' '
|
||||||
|
git for-each-ref --format="%(upstream:track)" refs/heads >actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
cat >expected <<EOF
|
||||||
|
>
|
||||||
|
EOF
|
||||||
|
|
||||||
|
test_expect_success 'Check upstream:trackshort format' '
|
||||||
|
git for-each-ref --format="%(upstream:trackshort)" refs/heads >actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'Check that :track[short] cannot be used with other atoms' '
|
||||||
|
test_must_fail git for-each-ref --format="%(refname:track)" 2>/dev/null &&
|
||||||
|
test_must_fail git for-each-ref --format="%(refname:trackshort)" 2>/dev/null
|
||||||
|
'
|
||||||
|
|
||||||
cat >expected <<EOF
|
cat >expected <<EOF
|
||||||
$(git rev-parse --short HEAD)
|
$(git rev-parse --short HEAD)
|
||||||
EOF
|
EOF
|
||||||
|
Loading…
Reference in New Issue
Block a user