diff options
author | Congyi Wu <congyiwu@gmail.com> | 2013-01-03 13:26:11 -0500 |
---|---|---|
committer | Congyi Wu <congyiwu@gmail.com> | 2013-01-03 17:19:55 -0500 |
commit | 4128f5aa31f4620c2393cc49e6e44e23d07fe58f (patch) | |
tree | e01d34b2564939adb9593c1d675dedfa4eb6b305 /src/push.c | |
parent | 07871d3adcfdaba7ad5f99f89299258d1dbd92f9 (diff) | |
download | libgit2-4128f5aa31f4620c2393cc49e6e44e23d07fe58f.tar.gz |
Fix bug in gen_pktline() for deletes of missing remote refs
* gen_pktline() in smart_protocol.c was skipping refspecs that deleted
refs that were not advertised by the server. The new behavior is to
send a delete command with an old-id of zero, which matches the behavior
of the official git client.
* Update test_network_push__delete() in reaction to above fix.
* Obviate messy logic that handles missing push_spec rrefs by canonicalizing
push_spec. After calculate_work(), loid, roid, and rref, are filled in with
exactly what is sent to the server
Diffstat (limited to 'src/push.c')
-rw-r--r-- | src/push.c | 53 |
1 files changed, 18 insertions, 35 deletions
diff --git a/src/push.c b/src/push.c index 1d63d574e..efca743b3 100644 --- a/src/push.c +++ b/src/push.c @@ -101,24 +101,27 @@ static int parse_refspec(push_spec **spec, const char *str) if (delim == NULL) { s->lref = git__strdup(str); check(s->lref); - s->rref = NULL; } else { if (delim - str) { s->lref = git__strndup(str, delim - str); check(s->lref); - } else - s->lref = NULL; + } if (strlen(delim + 1)) { s->rref = git__strdup(delim + 1); check(s->rref); - } else - s->rref = NULL; + } } if (!s->lref && !s->rref) goto on_error; + /* If rref is ommitted, use the same ref name as lref */ + if (!s->rref) { + s->rref = git__strdup(s->lref); + check(s->rref); + } + #undef check *spec = s; @@ -282,44 +285,24 @@ static int calculate_work(git_push *push) push_spec *spec; unsigned int i, j; + /* Update local and remote oids*/ + git_vector_foreach(&push->specs, i, spec) { if (spec->lref) { + /* This is a create or update. Local ref must exist. */ if (git_reference_name_to_id( &spec->loid, push->repo, spec->lref) < 0) { giterr_set(GIT_ENOTFOUND, "No such reference '%s'", spec->lref); return -1; } + } - if (!spec->rref) { - /* - * No remote reference given; if we find a remote - * reference with the same name we will update it, - * otherwise a new reference will be created. - */ - git_vector_foreach(&push->remote->refs, j, head) { - if (!strcmp(spec->lref, head->name)) { - /* - * Update remote reference - */ - git_oid_cpy(&spec->roid, &head->oid); - - break; - } - } - } else { - /* - * Remote reference given; update the given - * reference or create it. - */ - git_vector_foreach(&push->remote->refs, j, head) { - if (!strcmp(spec->rref, head->name)) { - /* - * Update remote reference - */ - git_oid_cpy(&spec->roid, &head->oid); - - break; - } + if (spec->rref) { + /* Remote ref may or may not (e.g. during create) already exist. */ + git_vector_foreach(&push->remote->refs, j, head) { + if (!strcmp(spec->rref, head->name)) { + git_oid_cpy(&spec->roid, &head->oid); + break; } } } |