Pushing from a shallow clone using today's send-pack and receive-pack may work, if the transferred pack does not ends up at any graft points. If it does, recent receive-pack that does connectivity check will reject the push. If receive-pack is old, the upstream repo becomes corrupt. The pack protocol is updated and send-pack now sends all shallow grafts before it sends the commands, if the repo is shallow. This protocol extension will break current receive-pack, which is intended, mostly to stop corrupting the upstream repo. The receiver end, the newreceive-pack, does something similar to fetch-pack: it creates a temporary shallow file with grafts from send-pack, then receives the pack, and finally writes the refined shallow file down. shadow file is not cleaned up after deleting (or force updating) a ref if that ref is the only way to reach the graft points. The reason is once we delete graft points, we can't recover. That may make reflog entries on server useless. Leave that for the administrators to decide when to clean up shadow file (maybe at repack/gc time). Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@xxxxxxxxx> --- Documentation/technical/pack-protocol.txt | 4 +- builtin/receive-pack.c | 49 ++++++++++++++++++---- send-pack.c | 3 ++ t/t5537-push-shallow.sh (new +x) | 67 +++++++++++++++++++++++++++++++ 4 files changed, 114 insertions(+), 9 deletions(-) create mode 100755 t/t5537-push-shallow.sh diff --git a/Documentation/technical/pack-protocol.txt b/Documentation/technical/pack-protocol.txt index eb8edd1..c73b62f 100644 --- a/Documentation/technical/pack-protocol.txt +++ b/Documentation/technical/pack-protocol.txt @@ -464,7 +464,9 @@ contain all the objects that the server will need to complete the new references. ---- - update-request = command-list [pack-file] + update-request = *shallow command-list [pack-file] + + shallow = PKT-LINE("shallow" SP obj-id) command-list = PKT-LINE(command NUL capability-list LF) *PKT-LINE(command LF) diff --git a/builtin/receive-pack.c b/builtin/receive-pack.c index 2d8e19b..0537e26 100644 --- a/builtin/receive-pack.c +++ b/builtin/receive-pack.c @@ -41,6 +41,10 @@ static int auto_gc = 1; static const char *head_name; static void *head_name_to_free; static int sent_capabilities; +static int shallow_changed; +static const char* alternate_shallow_file; +static struct lock_file shallow_lock; +static struct extra_have_objects shallow; static enum deny_action parse_deny_action(const char *var, const char *value) { @@ -751,6 +755,13 @@ static void execute_commands(struct command *commands, const char *unpacker_erro } } +static void add_extra_have(struct extra_have_objects *extra, unsigned char *sha1) +{ + ALLOC_GROW(extra->array, extra->nr + 1, extra->alloc); + hashcpy(&(extra->array[extra->nr][0]), sha1); + extra->nr++; +} + static struct command *read_head_info(void) { struct command *commands = NULL; @@ -765,6 +776,17 @@ static struct command *read_head_info(void) line = packet_read_line(0, &len); if (!line) break; + + if (len == 48 && !prefixcmp(line, "shallow ")) { + if (get_sha1_hex(line + 8, old_sha1)) + die("protocol error: expected shallow sha, got '%s'", line + 8); + if (!has_sha1_file(old_sha1)) { + add_extra_have(&shallow, old_sha1); + shallow_changed = 1; + } + continue; + } + if (len < 83 || line[40] != ' ' || line[81] != ' ' || @@ -827,6 +849,12 @@ static const char *unpack(int err_fd) ? transfer_fsck_objects : 0); + if (shallow_changed) + setup_alternate_shallow(&shallow_lock, + &alternate_shallow_file, + &shallow, 0); + + hdr_err = parse_pack_header(&hdr); if (hdr_err) { if (err_fd > 0) @@ -854,9 +882,8 @@ static const char *unpack(int err_fd) child.err = err_fd; child.git_cmd = 1; code = run_command(&child); - if (!code) - return NULL; - return "unpack-objects abnormal exit"; + if (code) + return "unpack-objects abnormal exit"; } else { const char *keeper[7]; int s, status, i = 0; @@ -887,12 +914,18 @@ static const char *unpack(int err_fd) pack_lockfile = index_pack_lockfile(ip.out); close(ip.out); status = finish_command(&ip); - if (!status) { - reprepare_packed_git(); - return NULL; - } - return "index-pack abnormal exit"; + if (status) + return "index-pack abnormal exit"; + reprepare_packed_git(); + } + + if (shallow_changed) { + setup_alternate_shallow(&shallow_lock, + &alternate_shallow_file, + &shallow, 1); + commit_lock_file(&shallow_lock); } + return NULL; } static const char *unpack_with_sideband(void) diff --git a/send-pack.c b/send-pack.c index 7d172ef..81d4b1c 100644 --- a/send-pack.c +++ b/send-pack.c @@ -212,6 +212,9 @@ int send_pack(struct send_pack_args *args, return 0; } + if (!args->dry_run && !args->stateless_rpc) + advertise_shallow_grafts(out); + /* * Finally, tell the other end! */ diff --git a/t/t5537-push-shallow.sh b/t/t5537-push-shallow.sh new file mode 100755 index 0000000..30fdbc8 --- /dev/null +++ b/t/t5537-push-shallow.sh @@ -0,0 +1,67 @@ +#!/bin/sh + +test_description='push from/to a shallow clone' + +. ./test-lib.sh + +test_expect_success 'setup' ' + test_commit 1 && + test_commit 2 && + test_commit 3 && + test_commit 4 +' + +test_expect_success 'setup shallow clone' ' + git clone --no-local --depth=2 .git shallow && + git --git-dir=shallow/.git log --format=%s >actual && + cat <<EOF >expect && +4 +3 +EOF + test_cmp expect actual +' + +test_expect_success 'push from shallow clone' ' + ( + cd shallow && + test_commit 5 && + git push ../.git +master:refs/remotes/shallow/master + ) && + git log --format=%s shallow/master >actual && + git fsck && + cat <<EOF >expect && +5 +4 +3 +2 +1 +EOF + test_cmp expect actual +' + +test_expect_success 'push from shallow clone, with grafted roots' ' + git init shallow2 && + ( + cd shallow2 && + test_commit a && + test_commit b && + test_commit c && + git rev-parse b > .git/shallow && + git log --format=%s >actual && + cat <<EOF >expect && +c +b +EOF + test_cmp expect actual && + git push ../.git +master:refs/remotes/shallow2/master + ) && + git log --format=%s shallow2/master >actual && + git fsck && + cat <<EOF >expect && +c +b +EOF + test_cmp expect actual +' + +test_done -- 1.8.2.83.gc99314b -- To unsubscribe from this list: send the line "unsubscribe git" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html