serve: advertise object-format capability for protocol v2

In order to communicate the protocol supported by the server side, add
support for advertising the object-format capability.  We check that the
client side sends us an identical algorithm if it sends us its own
object-format capability, and assume it speaks SHA-1 if not.

In the test, when we're using an algorithm other than SHA-1, we need to
specify the algorithm in use so we don't get a failure with an "unknown
format" message.  Add a test that we handle a mismatched algorithm.
Remove the test_oid_init call since it's no longer necessary.

Signed-off-by: brian m. carlson <sandals@crustytoothpaste.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
brian m. carlson 2020-05-25 19:59:17 +00:00 committed by Junio C Hamano
parent ab67235bc4
commit 9de0dd361c
3 changed files with 54 additions and 0 deletions

View File

@ -460,6 +460,8 @@ struct ref **get_remote_refs(int fd_out, struct packet_reader *reader,
die(_("unknown object format '%s' specified by server"), hash_name); die(_("unknown object format '%s' specified by server"), hash_name);
reader->hash_algo = &hash_algos[hash_algo]; reader->hash_algo = &hash_algos[hash_algo];
packet_write_fmt(fd_out, "object-format=%s", reader->hash_algo->name); packet_write_fmt(fd_out, "object-format=%s", reader->hash_algo->name);
} else {
reader->hash_algo = &hash_algos[GIT_HASH_SHA1];
} }
if (server_options && server_options->nr && if (server_options && server_options->nr &&

27
serve.c
View File

@ -22,6 +22,14 @@ static int agent_advertise(struct repository *r,
return 1; return 1;
} }
static int object_format_advertise(struct repository *r,
struct strbuf *value)
{
if (value)
strbuf_addstr(value, r->hash_algo->name);
return 1;
}
struct protocol_capability { struct protocol_capability {
/* /*
* The name of the capability. The server uses this name when * The name of the capability. The server uses this name when
@ -57,6 +65,7 @@ static struct protocol_capability capabilities[] = {
{ "ls-refs", always_advertise, ls_refs }, { "ls-refs", always_advertise, ls_refs },
{ "fetch", upload_pack_advertise, upload_pack_v2 }, { "fetch", upload_pack_advertise, upload_pack_v2 },
{ "server-option", always_advertise, NULL }, { "server-option", always_advertise, NULL },
{ "object-format", object_format_advertise, NULL },
}; };
static void advertise_capabilities(void) static void advertise_capabilities(void)
@ -153,6 +162,22 @@ int has_capability(const struct argv_array *keys, const char *capability,
return 0; return 0;
} }
static void check_algorithm(struct repository *r, struct argv_array *keys)
{
int client = GIT_HASH_SHA1, server = hash_algo_by_ptr(r->hash_algo);
const char *algo_name;
if (has_capability(keys, "object-format", &algo_name)) {
client = hash_algo_by_name(algo_name);
if (client == GIT_HASH_UNKNOWN)
die("unknown object format '%s'", algo_name);
}
if (client != server)
die("mismatched object format: server %s; client %s\n",
r->hash_algo->name, hash_algos[client].name);
}
enum request_state { enum request_state {
PROCESS_REQUEST_KEYS, PROCESS_REQUEST_KEYS,
PROCESS_REQUEST_DONE, PROCESS_REQUEST_DONE,
@ -223,6 +248,8 @@ static int process_request(void)
if (!command) if (!command)
die("no command requested"); die("no command requested");
check_algorithm(the_repository, &keys);
command->command(the_repository, &keys, &reader); command->command(the_repository, &keys, &reader);
argv_array_clear(&keys); argv_array_clear(&keys);

View File

@ -5,12 +5,17 @@ test_description='test protocol v2 server commands'
. ./test-lib.sh . ./test-lib.sh
test_expect_success 'test capability advertisement' ' test_expect_success 'test capability advertisement' '
test_oid_cache <<-EOF &&
wrong_algo sha1:sha256
wrong_algo sha256:sha1
EOF
cat >expect <<-EOF && cat >expect <<-EOF &&
version 2 version 2
agent=git/$(git version | cut -d" " -f3) agent=git/$(git version | cut -d" " -f3)
ls-refs ls-refs
fetch=shallow fetch=shallow
server-option server-option
object-format=$(test_oid algo)
0000 0000
EOF EOF
@ -45,6 +50,7 @@ test_expect_success 'request invalid capability' '
test_expect_success 'request with no command' ' test_expect_success 'request with no command' '
test-tool pkt-line pack >in <<-EOF && test-tool pkt-line pack >in <<-EOF &&
agent=git/test agent=git/test
object-format=$(test_oid algo)
0000 0000
EOF EOF
test_must_fail test-tool serve-v2 --stateless-rpc 2>err <in && test_must_fail test-tool serve-v2 --stateless-rpc 2>err <in &&
@ -54,6 +60,7 @@ test_expect_success 'request with no command' '
test_expect_success 'request invalid command' ' test_expect_success 'request invalid command' '
test-tool pkt-line pack >in <<-EOF && test-tool pkt-line pack >in <<-EOF &&
command=foo command=foo
object-format=$(test_oid algo)
agent=git/test agent=git/test
0000 0000
EOF EOF
@ -61,6 +68,17 @@ test_expect_success 'request invalid command' '
test_i18ngrep "invalid command" err test_i18ngrep "invalid command" err
' '
test_expect_success 'wrong object-format' '
test-tool pkt-line pack >in <<-EOF &&
command=fetch
agent=git/test
object-format=$(test_oid wrong_algo)
0000
EOF
test_must_fail test-tool serve-v2 --stateless-rpc 2>err <in &&
test_i18ngrep "mismatched object format" err
'
# Test the basics of ls-refs # Test the basics of ls-refs
# #
test_expect_success 'setup some refs and tags' ' test_expect_success 'setup some refs and tags' '
@ -74,6 +92,7 @@ test_expect_success 'setup some refs and tags' '
test_expect_success 'basics of ls-refs' ' test_expect_success 'basics of ls-refs' '
test-tool pkt-line pack >in <<-EOF && test-tool pkt-line pack >in <<-EOF &&
command=ls-refs command=ls-refs
object-format=$(test_oid algo)
0000 0000
EOF EOF
@ -96,6 +115,7 @@ test_expect_success 'basics of ls-refs' '
test_expect_success 'basic ref-prefixes' ' test_expect_success 'basic ref-prefixes' '
test-tool pkt-line pack >in <<-EOF && test-tool pkt-line pack >in <<-EOF &&
command=ls-refs command=ls-refs
object-format=$(test_oid algo)
0001 0001
ref-prefix refs/heads/master ref-prefix refs/heads/master
ref-prefix refs/tags/one ref-prefix refs/tags/one
@ -116,6 +136,7 @@ test_expect_success 'basic ref-prefixes' '
test_expect_success 'refs/heads prefix' ' test_expect_success 'refs/heads prefix' '
test-tool pkt-line pack >in <<-EOF && test-tool pkt-line pack >in <<-EOF &&
command=ls-refs command=ls-refs
object-format=$(test_oid algo)
0001 0001
ref-prefix refs/heads/ ref-prefix refs/heads/
0000 0000
@ -136,6 +157,7 @@ test_expect_success 'refs/heads prefix' '
test_expect_success 'peel parameter' ' test_expect_success 'peel parameter' '
test-tool pkt-line pack >in <<-EOF && test-tool pkt-line pack >in <<-EOF &&
command=ls-refs command=ls-refs
object-format=$(test_oid algo)
0001 0001
peel peel
ref-prefix refs/tags/ ref-prefix refs/tags/
@ -157,6 +179,7 @@ test_expect_success 'peel parameter' '
test_expect_success 'symrefs parameter' ' test_expect_success 'symrefs parameter' '
test-tool pkt-line pack >in <<-EOF && test-tool pkt-line pack >in <<-EOF &&
command=ls-refs command=ls-refs
object-format=$(test_oid algo)
0001 0001
symrefs symrefs
ref-prefix refs/heads/ ref-prefix refs/heads/
@ -178,6 +201,7 @@ test_expect_success 'symrefs parameter' '
test_expect_success 'sending server-options' ' test_expect_success 'sending server-options' '
test-tool pkt-line pack >in <<-EOF && test-tool pkt-line pack >in <<-EOF &&
command=ls-refs command=ls-refs
object-format=$(test_oid algo)
server-option=hello server-option=hello
server-option=world server-option=world
0001 0001
@ -200,6 +224,7 @@ test_expect_success 'unexpected lines are not allowed in fetch request' '
test-tool pkt-line pack >in <<-EOF && test-tool pkt-line pack >in <<-EOF &&
command=fetch command=fetch
object-format=$(test_oid algo)
0001 0001
this-is-not-a-command this-is-not-a-command
0000 0000