|
|
|
#include "cache.h"
|
|
|
|
#include "object-store.h"
|
|
|
|
#include "run-command.h"
|
|
|
|
#include "sigchain.h"
|
|
|
|
#include "connected.h"
|
|
|
|
#include "transport.h"
|
|
|
|
#include "packfile.h"
|
|
|
|
#include "promisor-remote.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we feed all the commits we want to verify to this command
|
|
|
|
*
|
|
|
|
* $ git rev-list --objects --stdin --not --all
|
|
|
|
*
|
|
|
|
* and if it does not error out, that means everything reachable from
|
|
|
|
* these commits locally exists and is connected to our existing refs.
|
|
|
|
* Note that this does _not_ validate the individual objects.
|
|
|
|
*
|
|
|
|
* Returns 0 if everything is connected, non-zero otherwise.
|
|
|
|
*/
|
|
|
|
int check_connected(oid_iterate_fn fn, void *cb_data,
|
check_everything_connected: use a struct with named options
The number of variants of check_everything_connected has
grown over the years, so that the "real" function takes
several possibly-zero, possibly-NULL arguments. We hid the
complexity behind some wrapper functions, but this doesn't
scale well when we want to add new options.
If we add more wrapper variants to handle the new options,
then we can get a combinatorial explosion when those options
might be used together (right now nobody wants to use both
"shallow" and "transport" together, so we get by with just a
few wrappers).
If instead we add new parameters to each function, each of
which can have a default value, then callers who want the
defaults end up with confusing invocations like:
check_everything_connected(fn, 0, data, -1, 0, NULL);
where it is unclear which parameter is which (and every
caller needs updated when we add new options).
Instead, let's add a struct to hold all of the optional
parameters. This is a little more verbose for the callers
(who have to declare the struct and fill it in), but it
makes their code much easier to follow, because every option
is named as it is set (and unused options do not have to be
mentioned at all).
Note that we could also stick the iteration function and its
callback data into the option struct, too. But since those
are required for each call, by avoiding doing so, we can let
very simple callers just pass "NULL" for the options and not
worry about the struct at all.
While we're touching each site, let's also rename the
function to check_connected(). The existing name was quite
long, and not all of the wrappers even used the full name.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
9 years ago
|
|
|
struct check_connected_options *opt)
|
|
|
|
{
|
|
|
|
struct child_process rev_list = CHILD_PROCESS_INIT;
|
check_everything_connected: use a struct with named options
The number of variants of check_everything_connected has
grown over the years, so that the "real" function takes
several possibly-zero, possibly-NULL arguments. We hid the
complexity behind some wrapper functions, but this doesn't
scale well when we want to add new options.
If we add more wrapper variants to handle the new options,
then we can get a combinatorial explosion when those options
might be used together (right now nobody wants to use both
"shallow" and "transport" together, so we get by with just a
few wrappers).
If instead we add new parameters to each function, each of
which can have a default value, then callers who want the
defaults end up with confusing invocations like:
check_everything_connected(fn, 0, data, -1, 0, NULL);
where it is unclear which parameter is which (and every
caller needs updated when we add new options).
Instead, let's add a struct to hold all of the optional
parameters. This is a little more verbose for the callers
(who have to declare the struct and fill it in), but it
makes their code much easier to follow, because every option
is named as it is set (and unused options do not have to be
mentioned at all).
Note that we could also stick the iteration function and its
callback data into the option struct, too. But since those
are required for each call, by avoiding doing so, we can let
very simple callers just pass "NULL" for the options and not
worry about the struct at all.
While we're touching each site, let's also rename the
function to check_connected(). The existing name was quite
long, and not all of the wrappers even used the full name.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
9 years ago
|
|
|
struct check_connected_options defaults = CHECK_CONNECTED_INIT;
|
|
|
|
char commit[GIT_MAX_HEXSZ + 1];
|
|
|
|
struct object_id oid;
|
|
|
|
int err = 0;
|
|
|
|
struct packed_git *new_pack = NULL;
|
check_everything_connected: use a struct with named options
The number of variants of check_everything_connected has
grown over the years, so that the "real" function takes
several possibly-zero, possibly-NULL arguments. We hid the
complexity behind some wrapper functions, but this doesn't
scale well when we want to add new options.
If we add more wrapper variants to handle the new options,
then we can get a combinatorial explosion when those options
might be used together (right now nobody wants to use both
"shallow" and "transport" together, so we get by with just a
few wrappers).
If instead we add new parameters to each function, each of
which can have a default value, then callers who want the
defaults end up with confusing invocations like:
check_everything_connected(fn, 0, data, -1, 0, NULL);
where it is unclear which parameter is which (and every
caller needs updated when we add new options).
Instead, let's add a struct to hold all of the optional
parameters. This is a little more verbose for the callers
(who have to declare the struct and fill it in), but it
makes their code much easier to follow, because every option
is named as it is set (and unused options do not have to be
mentioned at all).
Note that we could also stick the iteration function and its
callback data into the option struct, too. But since those
are required for each call, by avoiding doing so, we can let
very simple callers just pass "NULL" for the options and not
worry about the struct at all.
While we're touching each site, let's also rename the
function to check_connected(). The existing name was quite
long, and not all of the wrappers even used the full name.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
9 years ago
|
|
|
struct transport *transport;
|
|
|
|
size_t base_len;
|
|
|
|
const unsigned hexsz = the_hash_algo->hexsz;
|
|
|
|
|
check_everything_connected: use a struct with named options
The number of variants of check_everything_connected has
grown over the years, so that the "real" function takes
several possibly-zero, possibly-NULL arguments. We hid the
complexity behind some wrapper functions, but this doesn't
scale well when we want to add new options.
If we add more wrapper variants to handle the new options,
then we can get a combinatorial explosion when those options
might be used together (right now nobody wants to use both
"shallow" and "transport" together, so we get by with just a
few wrappers).
If instead we add new parameters to each function, each of
which can have a default value, then callers who want the
defaults end up with confusing invocations like:
check_everything_connected(fn, 0, data, -1, 0, NULL);
where it is unclear which parameter is which (and every
caller needs updated when we add new options).
Instead, let's add a struct to hold all of the optional
parameters. This is a little more verbose for the callers
(who have to declare the struct and fill it in), but it
makes their code much easier to follow, because every option
is named as it is set (and unused options do not have to be
mentioned at all).
Note that we could also stick the iteration function and its
callback data into the option struct, too. But since those
are required for each call, by avoiding doing so, we can let
very simple callers just pass "NULL" for the options and not
worry about the struct at all.
While we're touching each site, let's also rename the
function to check_connected(). The existing name was quite
long, and not all of the wrappers even used the full name.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
9 years ago
|
|
|
if (!opt)
|
|
|
|
opt = &defaults;
|
|
|
|
transport = opt->transport;
|
|
|
|
|
|
|
|
if (fn(cb_data, &oid)) {
|
|
|
|
if (opt->err_fd)
|
|
|
|
close(opt->err_fd);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (transport && transport->smart_options &&
|
|
|
|
transport->smart_options->self_contained_and_connected &&
|
|
|
|
transport->pack_lockfile &&
|
|
|
|
strip_suffix(transport->pack_lockfile, ".keep", &base_len)) {
|
|
|
|
struct strbuf idx_file = STRBUF_INIT;
|
|
|
|
strbuf_add(&idx_file, transport->pack_lockfile, base_len);
|
|
|
|
strbuf_addstr(&idx_file, ".idx");
|
|
|
|
new_pack = add_packed_git(idx_file.buf, idx_file.len, 1);
|
|
|
|
strbuf_release(&idx_file);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt->check_refs_only) {
|
|
|
|
/*
|
|
|
|
* For partial clones, we don't want to have to do a regular
|
|
|
|
* connectivity check because we have to enumerate and exclude
|
|
|
|
* all promisor objects (slow), and then the connectivity check
|
|
|
|
* itself becomes a no-op because in a partial clone every
|
|
|
|
* object is a promisor object. Instead, just make sure we
|
|
|
|
* received the objects pointed to by each wanted ref.
|
|
|
|
*/
|
|
|
|
do {
|
|
|
|
if (!repo_has_object_file(the_repository, &oid))
|
|
|
|
return 1;
|
|
|
|
} while (!fn(cb_data, &oid));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
check_everything_connected: use a struct with named options
The number of variants of check_everything_connected has
grown over the years, so that the "real" function takes
several possibly-zero, possibly-NULL arguments. We hid the
complexity behind some wrapper functions, but this doesn't
scale well when we want to add new options.
If we add more wrapper variants to handle the new options,
then we can get a combinatorial explosion when those options
might be used together (right now nobody wants to use both
"shallow" and "transport" together, so we get by with just a
few wrappers).
If instead we add new parameters to each function, each of
which can have a default value, then callers who want the
defaults end up with confusing invocations like:
check_everything_connected(fn, 0, data, -1, 0, NULL);
where it is unclear which parameter is which (and every
caller needs updated when we add new options).
Instead, let's add a struct to hold all of the optional
parameters. This is a little more verbose for the callers
(who have to declare the struct and fill it in), but it
makes their code much easier to follow, because every option
is named as it is set (and unused options do not have to be
mentioned at all).
Note that we could also stick the iteration function and its
callback data into the option struct, too. But since those
are required for each call, by avoiding doing so, we can let
very simple callers just pass "NULL" for the options and not
worry about the struct at all.
While we're touching each site, let's also rename the
function to check_connected(). The existing name was quite
long, and not all of the wrappers even used the full name.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
9 years ago
|
|
|
if (opt->shallow_file) {
|
|
|
|
argv_array_push(&rev_list.args, "--shallow-file");
|
check_everything_connected: use a struct with named options
The number of variants of check_everything_connected has
grown over the years, so that the "real" function takes
several possibly-zero, possibly-NULL arguments. We hid the
complexity behind some wrapper functions, but this doesn't
scale well when we want to add new options.
If we add more wrapper variants to handle the new options,
then we can get a combinatorial explosion when those options
might be used together (right now nobody wants to use both
"shallow" and "transport" together, so we get by with just a
few wrappers).
If instead we add new parameters to each function, each of
which can have a default value, then callers who want the
defaults end up with confusing invocations like:
check_everything_connected(fn, 0, data, -1, 0, NULL);
where it is unclear which parameter is which (and every
caller needs updated when we add new options).
Instead, let's add a struct to hold all of the optional
parameters. This is a little more verbose for the callers
(who have to declare the struct and fill it in), but it
makes their code much easier to follow, because every option
is named as it is set (and unused options do not have to be
mentioned at all).
Note that we could also stick the iteration function and its
callback data into the option struct, too. But since those
are required for each call, by avoiding doing so, we can let
very simple callers just pass "NULL" for the options and not
worry about the struct at all.
While we're touching each site, let's also rename the
function to check_connected(). The existing name was quite
long, and not all of the wrappers even used the full name.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
9 years ago
|
|
|
argv_array_push(&rev_list.args, opt->shallow_file);
|
|
|
|
}
|
|
|
|
argv_array_push(&rev_list.args,"rev-list");
|
|
|
|
argv_array_push(&rev_list.args, "--objects");
|
|
|
|
argv_array_push(&rev_list.args, "--stdin");
|
|
|
|
if (has_promisor_remote())
|
|
|
|
argv_array_push(&rev_list.args, "--exclude-promisor-objects");
|
fetch-pack: write shallow, then check connectivity
When fetching, connectivity is checked after the shallow file is
updated. There are 2 issues with this: (1) the connectivity check is
only performed up to ancestors of existing refs (which is not thorough
enough if we were deepening an existing ref in the first place), and (2)
there is no rollback of the shallow file if the connectivity check
fails.
To solve (1), update the connectivity check to check the ancestry chain
completely in the case of a deepening fetch by refraining from passing
"--not --all" when invoking rev-list in connected.c.
To solve (2), have fetch_pack() perform its own connectivity check
before updating the shallow file. To support existing use cases in which
"git fetch-pack" is used to download objects without much regard as to
the connectivity of the resulting objects with respect to the existing
repository, the connectivity check is only done if necessary (that is,
the fetch is not a clone, and the fetch involves shallow/deepen
functionality). "git fetch" still performs its own connectivity check,
preserving correctness but sometimes performing redundant work. This
redundancy is mitigated by the fact that fetch_pack() reports if it has
performed a connectivity check itself, and if the transport supports
connect or stateless-connect, it will bubble up that report so that "git
fetch" knows not to perform the connectivity check in such a case.
This was noticed when a user tried to deepen an existing repository by
fetching with --no-shallow from a server that did not send all necessary
objects - the connectivity check as run by "git fetch" succeeded, but a
subsequent "git fsck" failed.
Signed-off-by: Jonathan Tan <jonathantanmy@google.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
7 years ago
|
|
|
if (!opt->is_deepening_fetch) {
|
|
|
|
argv_array_push(&rev_list.args, "--not");
|
|
|
|
argv_array_push(&rev_list.args, "--all");
|
|
|
|
}
|
|
|
|
argv_array_push(&rev_list.args, "--quiet");
|
|
|
|
argv_array_push(&rev_list.args, "--alternate-refs");
|
|
|
|
if (opt->progress)
|
|
|
|
argv_array_pushf(&rev_list.args, "--progress=%s",
|
|
|
|
_("Checking connectivity"));
|
|
|
|
|
|
|
|
rev_list.git_cmd = 1;
|
|
|
|
rev_list.env = opt->env;
|
|
|
|
rev_list.in = -1;
|
|
|
|
rev_list.no_stdout = 1;
|
|
|
|
if (opt->err_fd)
|
|
|
|
rev_list.err = opt->err_fd;
|
|
|
|
else
|
|
|
|
rev_list.no_stderr = opt->quiet;
|
|
|
|
|
|
|
|
if (start_command(&rev_list))
|
|
|
|
return error(_("Could not run 'git rev-list'"));
|
|
|
|
|
|
|
|
sigchain_push(SIGPIPE, SIG_IGN);
|
|
|
|
|
|
|
|
commit[hexsz] = '\n';
|
|
|
|
do {
|
|
|
|
/*
|
|
|
|
* If index-pack already checked that:
|
|
|
|
* - there are no dangling pointers in the new pack
|
|
|
|
* - the pack is self contained
|
|
|
|
* Then if the updated ref is in the new pack, then we
|
|
|
|
* are sure the ref is good and not sending it to
|
|
|
|
* rev-list for verification.
|
|
|
|
*/
|
|
|
|
if (new_pack && find_pack_entry_one(oid.hash, new_pack))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
memcpy(commit, oid_to_hex(&oid), hexsz);
|
|
|
|
if (write_in_full(rev_list.in, commit, hexsz + 1) < 0) {
|
|
|
|
if (errno != EPIPE && errno != EINVAL)
|
|
|
|
error_errno(_("failed write to rev-list"));
|
|
|
|
err = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while (!fn(cb_data, &oid));
|
|
|
|
|
|
|
|
if (close(rev_list.in))
|
|
|
|
err = error_errno(_("failed to close rev-list's stdin"));
|
|
|
|
|
|
|
|
sigchain_pop(SIGPIPE);
|
|
|
|
return finish_command(&rev_list) || err;
|
|
|
|
}
|