connect: request remote refs using v2
Teach the client to be able to request a remote's refs using protocol v2. This is done by having a client issue a 'ls-refs' request to a v2 server. Signed-off-by: Brandon Williams <bmwill@google.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
72d0ea0056
commit
e52449b672
@ -5,6 +5,7 @@
|
||||
#include "parse-options.h"
|
||||
#include "protocol.h"
|
||||
#include "upload-pack.h"
|
||||
#include "serve.h"
|
||||
|
||||
static const char * const upload_pack_usage[] = {
|
||||
N_("git upload-pack [<options>] <dir>"),
|
||||
@ -16,6 +17,7 @@ int cmd_upload_pack(int argc, const char **argv, const char *prefix)
|
||||
const char *dir;
|
||||
int strict = 0;
|
||||
struct upload_pack_options opts = { 0 };
|
||||
struct serve_options serve_opts = SERVE_OPTIONS_INIT;
|
||||
struct option options[] = {
|
||||
OPT_BOOL(0, "stateless-rpc", &opts.stateless_rpc,
|
||||
N_("quit after a single request/response exchange")),
|
||||
@ -48,11 +50,9 @@ int cmd_upload_pack(int argc, const char **argv, const char *prefix)
|
||||
|
||||
switch (determine_protocol_version_server()) {
|
||||
case protocol_v2:
|
||||
/*
|
||||
* fetch support for protocol v2 has not been implemented yet,
|
||||
* so ignore the request to use v2 and fallback to using v0.
|
||||
*/
|
||||
upload_pack(&opts);
|
||||
serve_opts.advertise_capabilities = opts.advertise_refs;
|
||||
serve_opts.stateless_rpc = opts.stateless_rpc;
|
||||
serve(&serve_opts);
|
||||
break;
|
||||
case protocol_v1:
|
||||
/*
|
||||
|
138
connect.c
138
connect.c
@ -12,9 +12,11 @@
|
||||
#include "sha1-array.h"
|
||||
#include "transport.h"
|
||||
#include "strbuf.h"
|
||||
#include "version.h"
|
||||
#include "protocol.h"
|
||||
|
||||
static char *server_capabilities;
|
||||
static char *server_capabilities_v1;
|
||||
static struct argv_array server_capabilities_v2 = ARGV_ARRAY_INIT;
|
||||
static const char *parse_feature_value(const char *, const char *, int *);
|
||||
|
||||
static int check_ref(const char *name, unsigned int flags)
|
||||
@ -62,6 +64,33 @@ static void die_initial_contact(int unexpected)
|
||||
"and the repository exists."));
|
||||
}
|
||||
|
||||
/* Checks if the server supports the capability 'c' */
|
||||
int server_supports_v2(const char *c, int die_on_error)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < server_capabilities_v2.argc; i++) {
|
||||
const char *out;
|
||||
if (skip_prefix(server_capabilities_v2.argv[i], c, &out) &&
|
||||
(!*out || *out == '='))
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (die_on_error)
|
||||
die("server doesn't support '%s'", c);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void process_capabilities_v2(struct packet_reader *reader)
|
||||
{
|
||||
while (packet_reader_read(reader) == PACKET_READ_NORMAL)
|
||||
argv_array_push(&server_capabilities_v2, reader->line);
|
||||
|
||||
if (reader->status != PACKET_READ_FLUSH)
|
||||
die("expected flush after capabilities");
|
||||
}
|
||||
|
||||
enum protocol_version discover_version(struct packet_reader *reader)
|
||||
{
|
||||
enum protocol_version version = protocol_unknown_version;
|
||||
@ -84,7 +113,7 @@ enum protocol_version discover_version(struct packet_reader *reader)
|
||||
|
||||
switch (version) {
|
||||
case protocol_v2:
|
||||
die("support for protocol v2 not implemented yet");
|
||||
process_capabilities_v2(reader);
|
||||
break;
|
||||
case protocol_v1:
|
||||
/* Read the peeked version line */
|
||||
@ -128,7 +157,7 @@ reject:
|
||||
static void annotate_refs_with_symref_info(struct ref *ref)
|
||||
{
|
||||
struct string_list symref = STRING_LIST_INIT_DUP;
|
||||
const char *feature_list = server_capabilities;
|
||||
const char *feature_list = server_capabilities_v1;
|
||||
|
||||
while (feature_list) {
|
||||
int len;
|
||||
@ -157,7 +186,7 @@ static void process_capabilities(const char *line, int *len)
|
||||
int nul_location = strlen(line);
|
||||
if (nul_location == *len)
|
||||
return;
|
||||
server_capabilities = xstrdup(line + nul_location + 1);
|
||||
server_capabilities_v1 = xstrdup(line + nul_location + 1);
|
||||
*len = nul_location;
|
||||
}
|
||||
|
||||
@ -292,6 +321,105 @@ struct ref **get_remote_heads(struct packet_reader *reader,
|
||||
return list;
|
||||
}
|
||||
|
||||
/* Returns 1 when a valid ref has been added to `list`, 0 otherwise */
|
||||
static int process_ref_v2(const char *line, struct ref ***list)
|
||||
{
|
||||
int ret = 1;
|
||||
int i = 0;
|
||||
struct object_id old_oid;
|
||||
struct ref *ref;
|
||||
struct string_list line_sections = STRING_LIST_INIT_DUP;
|
||||
const char *end;
|
||||
|
||||
/*
|
||||
* Ref lines have a number of fields which are space deliminated. The
|
||||
* first field is the OID of the ref. The second field is the ref
|
||||
* name. Subsequent fields (symref-target and peeled) are optional and
|
||||
* don't have a particular order.
|
||||
*/
|
||||
if (string_list_split(&line_sections, line, ' ', -1) < 2) {
|
||||
ret = 0;
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (parse_oid_hex(line_sections.items[i++].string, &old_oid, &end) ||
|
||||
*end) {
|
||||
ret = 0;
|
||||
goto out;
|
||||
}
|
||||
|
||||
ref = alloc_ref(line_sections.items[i++].string);
|
||||
|
||||
oidcpy(&ref->old_oid, &old_oid);
|
||||
**list = ref;
|
||||
*list = &ref->next;
|
||||
|
||||
for (; i < line_sections.nr; i++) {
|
||||
const char *arg = line_sections.items[i].string;
|
||||
if (skip_prefix(arg, "symref-target:", &arg))
|
||||
ref->symref = xstrdup(arg);
|
||||
|
||||
if (skip_prefix(arg, "peeled:", &arg)) {
|
||||
struct object_id peeled_oid;
|
||||
char *peeled_name;
|
||||
struct ref *peeled;
|
||||
if (parse_oid_hex(arg, &peeled_oid, &end) || *end) {
|
||||
ret = 0;
|
||||
goto out;
|
||||
}
|
||||
|
||||
peeled_name = xstrfmt("%s^{}", ref->name);
|
||||
peeled = alloc_ref(peeled_name);
|
||||
|
||||
oidcpy(&peeled->old_oid, &peeled_oid);
|
||||
**list = peeled;
|
||||
*list = &peeled->next;
|
||||
|
||||
free(peeled_name);
|
||||
}
|
||||
}
|
||||
|
||||
out:
|
||||
string_list_clear(&line_sections, 0);
|
||||
return ret;
|
||||
}
|
||||
|
||||
struct ref **get_remote_refs(int fd_out, struct packet_reader *reader,
|
||||
struct ref **list, int for_push,
|
||||
const struct argv_array *ref_prefixes)
|
||||
{
|
||||
int i;
|
||||
*list = NULL;
|
||||
|
||||
if (server_supports_v2("ls-refs", 1))
|
||||
packet_write_fmt(fd_out, "command=ls-refs\n");
|
||||
|
||||
if (server_supports_v2("agent", 0))
|
||||
packet_write_fmt(fd_out, "agent=%s", git_user_agent_sanitized());
|
||||
|
||||
packet_delim(fd_out);
|
||||
/* When pushing we don't want to request the peeled tags */
|
||||
if (!for_push)
|
||||
packet_write_fmt(fd_out, "peel\n");
|
||||
packet_write_fmt(fd_out, "symrefs\n");
|
||||
for (i = 0; ref_prefixes && i < ref_prefixes->argc; i++) {
|
||||
packet_write_fmt(fd_out, "ref-prefix %s\n",
|
||||
ref_prefixes->argv[i]);
|
||||
}
|
||||
packet_flush(fd_out);
|
||||
|
||||
/* Process response from server */
|
||||
while (packet_reader_read(reader) == PACKET_READ_NORMAL) {
|
||||
if (!process_ref_v2(reader->line, &list))
|
||||
die("invalid ls-refs response: %s", reader->line);
|
||||
}
|
||||
|
||||
if (reader->status != PACKET_READ_FLUSH)
|
||||
die("expected flush after ref listing");
|
||||
|
||||
return list;
|
||||
}
|
||||
|
||||
static const char *parse_feature_value(const char *feature_list, const char *feature, int *lenp)
|
||||
{
|
||||
int len;
|
||||
@ -336,7 +464,7 @@ int parse_feature_request(const char *feature_list, const char *feature)
|
||||
|
||||
const char *server_feature_value(const char *feature, int *len)
|
||||
{
|
||||
return parse_feature_value(server_capabilities, feature, len);
|
||||
return parse_feature_value(server_capabilities_v1, feature, len);
|
||||
}
|
||||
|
||||
int server_supports(const char *feature)
|
||||
|
@ -16,4 +16,6 @@ extern int url_is_local_not_ssh(const char *url);
|
||||
struct packet_reader;
|
||||
extern enum protocol_version discover_version(struct packet_reader *reader);
|
||||
|
||||
extern int server_supports_v2(const char *c, int die_on_error);
|
||||
|
||||
#endif
|
||||
|
6
remote.h
6
remote.h
@ -151,11 +151,17 @@ void free_refs(struct ref *ref);
|
||||
|
||||
struct oid_array;
|
||||
struct packet_reader;
|
||||
struct argv_array;
|
||||
extern struct ref **get_remote_heads(struct packet_reader *reader,
|
||||
struct ref **list, unsigned int flags,
|
||||
struct oid_array *extra_have,
|
||||
struct oid_array *shallow_points);
|
||||
|
||||
/* Used for protocol v2 in order to retrieve refs from a remote */
|
||||
extern struct ref **get_remote_refs(int fd_out, struct packet_reader *reader,
|
||||
struct ref **list, int for_push,
|
||||
const struct argv_array *ref_prefixes);
|
||||
|
||||
int resolve_remote_symref(struct ref *ref, struct ref *list);
|
||||
int ref_newer(const struct object_id *new_oid, const struct object_id *old_oid);
|
||||
|
||||
|
57
t/t5702-protocol-v2.sh
Executable file
57
t/t5702-protocol-v2.sh
Executable file
@ -0,0 +1,57 @@
|
||||
#!/bin/sh
|
||||
|
||||
test_description='test git wire-protocol version 2'
|
||||
|
||||
TEST_NO_CREATE_REPO=1
|
||||
|
||||
. ./test-lib.sh
|
||||
|
||||
# Test protocol v2 with 'git://' transport
|
||||
#
|
||||
. "$TEST_DIRECTORY"/lib-git-daemon.sh
|
||||
start_git_daemon --export-all --enable=receive-pack
|
||||
daemon_parent=$GIT_DAEMON_DOCUMENT_ROOT_PATH/parent
|
||||
|
||||
test_expect_success 'create repo to be served by git-daemon' '
|
||||
git init "$daemon_parent" &&
|
||||
test_commit -C "$daemon_parent" one
|
||||
'
|
||||
|
||||
test_expect_success 'list refs with git:// using protocol v2' '
|
||||
test_when_finished "rm -f log" &&
|
||||
|
||||
GIT_TRACE_PACKET="$(pwd)/log" git -c protocol.version=2 \
|
||||
ls-remote --symref "$GIT_DAEMON_URL/parent" >actual &&
|
||||
|
||||
# Client requested to use protocol v2
|
||||
grep "git> .*\\\0\\\0version=2\\\0$" log &&
|
||||
# Server responded using protocol v2
|
||||
grep "git< version 2" log &&
|
||||
|
||||
git ls-remote --symref "$GIT_DAEMON_URL/parent" >expect &&
|
||||
test_cmp actual expect
|
||||
'
|
||||
|
||||
stop_git_daemon
|
||||
|
||||
# Test protocol v2 with 'file://' transport
|
||||
#
|
||||
test_expect_success 'create repo to be served by file:// transport' '
|
||||
git init file_parent &&
|
||||
test_commit -C file_parent one
|
||||
'
|
||||
|
||||
test_expect_success 'list refs with file:// using protocol v2' '
|
||||
test_when_finished "rm -f log" &&
|
||||
|
||||
GIT_TRACE_PACKET="$(pwd)/log" git -c protocol.version=2 \
|
||||
ls-remote --symref "file://$(pwd)/file_parent" >actual &&
|
||||
|
||||
# Server responded using protocol v2
|
||||
grep "git< version 2" log &&
|
||||
|
||||
git ls-remote --symref "file://$(pwd)/file_parent" >expect &&
|
||||
test_cmp actual expect
|
||||
'
|
||||
|
||||
test_done
|
@ -204,7 +204,7 @@ static struct ref *get_refs_via_connect(struct transport *transport, int for_pus
|
||||
data->version = discover_version(&reader);
|
||||
switch (data->version) {
|
||||
case protocol_v2:
|
||||
die("support for protocol v2 not implemented yet");
|
||||
get_remote_refs(data->fd[1], &reader, &refs, for_push, NULL);
|
||||
break;
|
||||
case protocol_v1:
|
||||
case protocol_v0:
|
||||
|
Loading…
Reference in New Issue
Block a user