push: allow unqualified dest refspecs to DWIM
Previously, a push like: git push remote src:dst would go through the following steps: 1. check for an unambiguous 'dst' on the remote; if it exists, then push to that ref 2. otherwise, check if 'dst' begins with 'refs/'; if it does, create a new ref 3. otherwise, complain because we don't know where in the refs hierarchy to put 'dst' However, in some cases, we can guess about the ref type of 'dst' based on the ref type of 'src'. Specifically, before complaining we now check: 2.5. if 'src' resolves to a ref starting with refs/heads or refs/tags, then prepend that to 'dst' So now this creates a new branch on the remote, whereas it previously failed with an error message: git push master:newbranch Note that, by design, we limit this DWIM behavior only to source refs which resolve exactly (including symrefs which resolve to existing refs). We still complain on a partial destination refspec if the source is a raw sha1, or a ref expression such as 'master~10'. Signed-off-by: Jeff King <peff@peff.net> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
31c6390d40
commit
f8aae12034
32
remote.c
32
remote.c
@ -812,6 +812,26 @@ static struct ref *make_linked_ref(const char *name, struct ref ***tail)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static char *guess_ref(const char *name, struct ref *peer)
|
||||||
|
{
|
||||||
|
struct strbuf buf = STRBUF_INIT;
|
||||||
|
unsigned char sha1[20];
|
||||||
|
|
||||||
|
const char *r = resolve_ref(peer->name, sha1, 1, NULL);
|
||||||
|
if (!r)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (!prefixcmp(r, "refs/heads/"))
|
||||||
|
strbuf_addstr(&buf, "refs/heads/");
|
||||||
|
else if (!prefixcmp(r, "refs/tags/"))
|
||||||
|
strbuf_addstr(&buf, "refs/tags/");
|
||||||
|
else
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
strbuf_addstr(&buf, name);
|
||||||
|
return strbuf_detach(&buf, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
static int match_explicit(struct ref *src, struct ref *dst,
|
static int match_explicit(struct ref *src, struct ref *dst,
|
||||||
struct ref ***dst_tail,
|
struct ref ***dst_tail,
|
||||||
struct refspec *rs,
|
struct refspec *rs,
|
||||||
@ -820,6 +840,7 @@ static int match_explicit(struct ref *src, struct ref *dst,
|
|||||||
struct ref *matched_src, *matched_dst;
|
struct ref *matched_src, *matched_dst;
|
||||||
|
|
||||||
const char *dst_value = rs->dst;
|
const char *dst_value = rs->dst;
|
||||||
|
char *dst_guess;
|
||||||
|
|
||||||
if (rs->pattern)
|
if (rs->pattern)
|
||||||
return errs;
|
return errs;
|
||||||
@ -866,10 +887,15 @@ static int match_explicit(struct ref *src, struct ref *dst,
|
|||||||
case 0:
|
case 0:
|
||||||
if (!memcmp(dst_value, "refs/", 5))
|
if (!memcmp(dst_value, "refs/", 5))
|
||||||
matched_dst = make_linked_ref(dst_value, dst_tail);
|
matched_dst = make_linked_ref(dst_value, dst_tail);
|
||||||
|
else if((dst_guess = guess_ref(dst_value, matched_src)))
|
||||||
|
matched_dst = make_linked_ref(dst_guess, dst_tail);
|
||||||
else
|
else
|
||||||
error("dst refspec %s does not match any "
|
error("unable to push to unqualified destination: %s\n"
|
||||||
"existing ref on the remote and does "
|
"The destination refspec neither matches an "
|
||||||
"not start with refs/.", dst_value);
|
"existing ref on the remote nor\n"
|
||||||
|
"begins with refs/, and we are unable to "
|
||||||
|
"guess a prefix based on the source ref.",
|
||||||
|
dst_value);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
matched_dst = NULL;
|
matched_dst = NULL;
|
||||||
|
@ -273,6 +273,37 @@ test_expect_success 'push with colon-less refspec (4)' '
|
|||||||
|
|
||||||
'
|
'
|
||||||
|
|
||||||
|
test_expect_success 'push head with non-existant, incomplete dest' '
|
||||||
|
|
||||||
|
mk_test &&
|
||||||
|
git push testrepo master:branch &&
|
||||||
|
check_push_result $the_commit heads/branch
|
||||||
|
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'push tag with non-existant, incomplete dest' '
|
||||||
|
|
||||||
|
mk_test &&
|
||||||
|
git tag -f v1.0 &&
|
||||||
|
git push testrepo v1.0:tag &&
|
||||||
|
check_push_result $the_commit tags/tag
|
||||||
|
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'push sha1 with non-existant, incomplete dest' '
|
||||||
|
|
||||||
|
mk_test &&
|
||||||
|
test_must_fail git push testrepo `git rev-parse master`:foo
|
||||||
|
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'push ref expression with non-existant, incomplete dest' '
|
||||||
|
|
||||||
|
mk_test &&
|
||||||
|
test_must_fail git push testrepo master^:branch
|
||||||
|
|
||||||
|
'
|
||||||
|
|
||||||
test_expect_success 'push with HEAD' '
|
test_expect_success 'push with HEAD' '
|
||||||
|
|
||||||
mk_test heads/master &&
|
mk_test heads/master &&
|
||||||
@ -311,6 +342,15 @@ test_expect_success 'push with +HEAD' '
|
|||||||
|
|
||||||
'
|
'
|
||||||
|
|
||||||
|
test_expect_success 'push HEAD with non-existant, incomplete dest' '
|
||||||
|
|
||||||
|
mk_test &&
|
||||||
|
git checkout master &&
|
||||||
|
git push testrepo HEAD:branch &&
|
||||||
|
check_push_result $the_commit heads/branch
|
||||||
|
|
||||||
|
'
|
||||||
|
|
||||||
test_expect_success 'push with config remote.*.push = HEAD' '
|
test_expect_success 'push with config remote.*.push = HEAD' '
|
||||||
|
|
||||||
mk_test heads/local &&
|
mk_test heads/local &&
|
||||||
|
Loading…
Reference in New Issue
Block a user