Merge branch 'nd/clone-connectivity-shortcut'
Special case "git clone" and use lighter-weight implementation to check the completeness of the history behind refs. * nd/clone-connectivity-shortcut: clone: open a shortcut for connectivity check index-pack: remove dead code (it should never happen) fetch-pack: prepare updated shallow file before fetching the pack clone: let the user know when check_everything_connected is run
This commit is contained in:
commit
72e719292d
@ -74,6 +74,9 @@ OPTIONS
|
|||||||
--strict::
|
--strict::
|
||||||
Die, if the pack contains broken objects or links.
|
Die, if the pack contains broken objects or links.
|
||||||
|
|
||||||
|
--check-self-contained-and-connected::
|
||||||
|
Die if the pack contains broken links. For internal use only.
|
||||||
|
|
||||||
--threads=<n>::
|
--threads=<n>::
|
||||||
Specifies the number of threads to spawn when resolving
|
Specifies the number of threads to spawn when resolving
|
||||||
deltas. This requires that index-pack be compiled with
|
deltas. This requires that index-pack be compiled with
|
||||||
|
@ -541,12 +541,18 @@ static void update_remote_refs(const struct ref *refs,
|
|||||||
const struct ref *mapped_refs,
|
const struct ref *mapped_refs,
|
||||||
const struct ref *remote_head_points_at,
|
const struct ref *remote_head_points_at,
|
||||||
const char *branch_top,
|
const char *branch_top,
|
||||||
const char *msg)
|
const char *msg,
|
||||||
|
struct transport *transport)
|
||||||
{
|
{
|
||||||
const struct ref *rm = mapped_refs;
|
const struct ref *rm = mapped_refs;
|
||||||
|
|
||||||
if (check_everything_connected(iterate_ref_map, 0, &rm))
|
if (0 <= option_verbosity)
|
||||||
|
printf(_("Checking connectivity... "));
|
||||||
|
if (check_everything_connected_with_transport(iterate_ref_map,
|
||||||
|
0, &rm, transport))
|
||||||
die(_("remote did not send all necessary objects"));
|
die(_("remote did not send all necessary objects"));
|
||||||
|
if (0 <= option_verbosity)
|
||||||
|
printf(_("done\n"));
|
||||||
|
|
||||||
if (refs) {
|
if (refs) {
|
||||||
write_remote_refs(mapped_refs);
|
write_remote_refs(mapped_refs);
|
||||||
@ -890,6 +896,9 @@ int cmd_clone(int argc, const char **argv, const char *prefix)
|
|||||||
if (option_upload_pack)
|
if (option_upload_pack)
|
||||||
transport_set_option(transport, TRANS_OPT_UPLOADPACK,
|
transport_set_option(transport, TRANS_OPT_UPLOADPACK,
|
||||||
option_upload_pack);
|
option_upload_pack);
|
||||||
|
|
||||||
|
if (transport->smart_options && !option_depth)
|
||||||
|
transport->smart_options->check_self_contained_and_connected = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
refs = transport_get_remote_refs(transport);
|
refs = transport_get_remote_refs(transport);
|
||||||
@ -951,7 +960,7 @@ int cmd_clone(int argc, const char **argv, const char *prefix)
|
|||||||
transport_fetch_refs(transport, mapped_refs);
|
transport_fetch_refs(transport, mapped_refs);
|
||||||
|
|
||||||
update_remote_refs(refs, mapped_refs, remote_head_points_at,
|
update_remote_refs(refs, mapped_refs, remote_head_points_at,
|
||||||
branch_top.buf, reflog_msg.buf);
|
branch_top.buf, reflog_msg.buf, transport);
|
||||||
|
|
||||||
update_head(our_head_points_at, remote_head, reflog_msg.buf);
|
update_head(our_head_points_at, remote_head, reflog_msg.buf);
|
||||||
|
|
||||||
|
@ -77,8 +77,10 @@ static int nr_threads;
|
|||||||
|
|
||||||
static int from_stdin;
|
static int from_stdin;
|
||||||
static int strict;
|
static int strict;
|
||||||
|
static int do_fsck_object;
|
||||||
static int verbose;
|
static int verbose;
|
||||||
static int show_stat;
|
static int show_stat;
|
||||||
|
static int check_self_contained_and_connected;
|
||||||
|
|
||||||
static struct progress *progress;
|
static struct progress *progress;
|
||||||
|
|
||||||
@ -187,13 +189,13 @@ static int mark_link(struct object *obj, int type, void *data)
|
|||||||
|
|
||||||
/* The content of each linked object must have been checked
|
/* The content of each linked object must have been checked
|
||||||
or it must be already present in the object database */
|
or it must be already present in the object database */
|
||||||
static void check_object(struct object *obj)
|
static unsigned check_object(struct object *obj)
|
||||||
{
|
{
|
||||||
if (!obj)
|
if (!obj)
|
||||||
return;
|
return 0;
|
||||||
|
|
||||||
if (!(obj->flags & FLAG_LINK))
|
if (!(obj->flags & FLAG_LINK))
|
||||||
return;
|
return 0;
|
||||||
|
|
||||||
if (!(obj->flags & FLAG_CHECKED)) {
|
if (!(obj->flags & FLAG_CHECKED)) {
|
||||||
unsigned long size;
|
unsigned long size;
|
||||||
@ -201,17 +203,20 @@ static void check_object(struct object *obj)
|
|||||||
if (type != obj->type || type <= 0)
|
if (type != obj->type || type <= 0)
|
||||||
die(_("object of unexpected type"));
|
die(_("object of unexpected type"));
|
||||||
obj->flags |= FLAG_CHECKED;
|
obj->flags |= FLAG_CHECKED;
|
||||||
return;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void check_objects(void)
|
static unsigned check_objects(void)
|
||||||
{
|
{
|
||||||
unsigned i, max;
|
unsigned i, max, foreign_nr = 0;
|
||||||
|
|
||||||
max = get_max_object_index();
|
max = get_max_object_index();
|
||||||
for (i = 0; i < max; i++)
|
for (i = 0; i < max; i++)
|
||||||
check_object(get_indexed_object(i));
|
foreign_nr += check_object(get_indexed_object(i));
|
||||||
|
return foreign_nr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -747,8 +752,7 @@ static void sha1_object(const void *data, struct object_entry *obj_entry,
|
|||||||
int eaten;
|
int eaten;
|
||||||
void *buf = (void *) data;
|
void *buf = (void *) data;
|
||||||
|
|
||||||
if (!buf)
|
assert(data && "data can only be NULL for large _blobs_");
|
||||||
buf = new_data = get_data_from_pack(obj_entry);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* we do not need to free the memory here, as the
|
* we do not need to free the memory here, as the
|
||||||
@ -757,7 +761,8 @@ static void sha1_object(const void *data, struct object_entry *obj_entry,
|
|||||||
obj = parse_object_buffer(sha1, type, size, buf, &eaten);
|
obj = parse_object_buffer(sha1, type, size, buf, &eaten);
|
||||||
if (!obj)
|
if (!obj)
|
||||||
die(_("invalid %s"), typename(type));
|
die(_("invalid %s"), typename(type));
|
||||||
if (fsck_object(obj, 1, fsck_error_function))
|
if (do_fsck_object &&
|
||||||
|
fsck_object(obj, 1, fsck_error_function))
|
||||||
die(_("Error in object"));
|
die(_("Error in object"));
|
||||||
if (fsck_walk(obj, mark_link, NULL))
|
if (fsck_walk(obj, mark_link, NULL))
|
||||||
die(_("Not all child objects of %s are reachable"), sha1_to_hex(obj->sha1));
|
die(_("Not all child objects of %s are reachable"), sha1_to_hex(obj->sha1));
|
||||||
@ -1491,6 +1496,7 @@ int cmd_index_pack(int argc, const char **argv, const char *prefix)
|
|||||||
struct pack_idx_entry **idx_objects;
|
struct pack_idx_entry **idx_objects;
|
||||||
struct pack_idx_option opts;
|
struct pack_idx_option opts;
|
||||||
unsigned char pack_sha1[20];
|
unsigned char pack_sha1[20];
|
||||||
|
unsigned foreign_nr = 1; /* zero is a "good" value, assume bad */
|
||||||
|
|
||||||
if (argc == 2 && !strcmp(argv[1], "-h"))
|
if (argc == 2 && !strcmp(argv[1], "-h"))
|
||||||
usage(index_pack_usage);
|
usage(index_pack_usage);
|
||||||
@ -1512,6 +1518,10 @@ int cmd_index_pack(int argc, const char **argv, const char *prefix)
|
|||||||
fix_thin_pack = 1;
|
fix_thin_pack = 1;
|
||||||
} else if (!strcmp(arg, "--strict")) {
|
} else if (!strcmp(arg, "--strict")) {
|
||||||
strict = 1;
|
strict = 1;
|
||||||
|
do_fsck_object = 1;
|
||||||
|
} else if (!strcmp(arg, "--check-self-contained-and-connected")) {
|
||||||
|
strict = 1;
|
||||||
|
check_self_contained_and_connected = 1;
|
||||||
} else if (!strcmp(arg, "--verify")) {
|
} else if (!strcmp(arg, "--verify")) {
|
||||||
verify = 1;
|
verify = 1;
|
||||||
} else if (!strcmp(arg, "--verify-stat")) {
|
} else if (!strcmp(arg, "--verify-stat")) {
|
||||||
@ -1625,7 +1635,7 @@ int cmd_index_pack(int argc, const char **argv, const char *prefix)
|
|||||||
conclude_pack(fix_thin_pack, curr_pack, pack_sha1);
|
conclude_pack(fix_thin_pack, curr_pack, pack_sha1);
|
||||||
free(deltas);
|
free(deltas);
|
||||||
if (strict)
|
if (strict)
|
||||||
check_objects();
|
foreign_nr = check_objects();
|
||||||
|
|
||||||
if (show_stat)
|
if (show_stat)
|
||||||
show_pack_info(stat_only);
|
show_pack_info(stat_only);
|
||||||
@ -1651,5 +1661,11 @@ int cmd_index_pack(int argc, const char **argv, const char *prefix)
|
|||||||
if (index_name == NULL)
|
if (index_name == NULL)
|
||||||
free((void *) curr_index);
|
free((void *) curr_index);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Let the caller know this pack is not self contained
|
||||||
|
*/
|
||||||
|
if (check_self_contained_and_connected && foreign_nr)
|
||||||
|
return 1;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
2
commit.h
2
commit.h
@ -176,6 +176,8 @@ extern int for_each_commit_graft(each_commit_graft_fn, void *);
|
|||||||
extern int is_repository_shallow(void);
|
extern int is_repository_shallow(void);
|
||||||
extern struct commit_list *get_shallow_commits(struct object_array *heads,
|
extern struct commit_list *get_shallow_commits(struct object_array *heads,
|
||||||
int depth, int shallow_flag, int not_shallow_flag);
|
int depth, int shallow_flag, int not_shallow_flag);
|
||||||
|
extern void check_shallow_file_for_update(void);
|
||||||
|
extern void set_alternate_shallow_file(const char *path);
|
||||||
|
|
||||||
int is_descendant_of(struct commit *, struct commit_list *);
|
int is_descendant_of(struct commit *, struct commit_list *);
|
||||||
int in_merge_bases(struct commit *, struct commit *);
|
int in_merge_bases(struct commit *, struct commit *);
|
||||||
|
34
connected.c
34
connected.c
@ -2,7 +2,12 @@
|
|||||||
#include "run-command.h"
|
#include "run-command.h"
|
||||||
#include "sigchain.h"
|
#include "sigchain.h"
|
||||||
#include "connected.h"
|
#include "connected.h"
|
||||||
|
#include "transport.h"
|
||||||
|
|
||||||
|
int check_everything_connected(sha1_iterate_fn fn, int quiet, void *cb_data)
|
||||||
|
{
|
||||||
|
return check_everything_connected_with_transport(fn, quiet, cb_data, NULL);
|
||||||
|
}
|
||||||
/*
|
/*
|
||||||
* If we feed all the commits we want to verify to this command
|
* If we feed all the commits we want to verify to this command
|
||||||
*
|
*
|
||||||
@ -14,7 +19,10 @@
|
|||||||
*
|
*
|
||||||
* Returns 0 if everything is connected, non-zero otherwise.
|
* Returns 0 if everything is connected, non-zero otherwise.
|
||||||
*/
|
*/
|
||||||
int check_everything_connected(sha1_iterate_fn fn, int quiet, void *cb_data)
|
int check_everything_connected_with_transport(sha1_iterate_fn fn,
|
||||||
|
int quiet,
|
||||||
|
void *cb_data,
|
||||||
|
struct transport *transport)
|
||||||
{
|
{
|
||||||
struct child_process rev_list;
|
struct child_process rev_list;
|
||||||
const char *argv[] = {"rev-list", "--objects",
|
const char *argv[] = {"rev-list", "--objects",
|
||||||
@ -22,10 +30,23 @@ int check_everything_connected(sha1_iterate_fn fn, int quiet, void *cb_data)
|
|||||||
char commit[41];
|
char commit[41];
|
||||||
unsigned char sha1[20];
|
unsigned char sha1[20];
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
struct packed_git *new_pack = NULL;
|
||||||
|
|
||||||
if (fn(cb_data, sha1))
|
if (fn(cb_data, sha1))
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
|
if (transport && transport->smart_options &&
|
||||||
|
transport->smart_options->self_contained_and_connected &&
|
||||||
|
transport->pack_lockfile &&
|
||||||
|
!suffixcmp(transport->pack_lockfile, ".keep")) {
|
||||||
|
struct strbuf idx_file = STRBUF_INIT;
|
||||||
|
strbuf_addstr(&idx_file, transport->pack_lockfile);
|
||||||
|
strbuf_setlen(&idx_file, idx_file.len - 5); /* ".keep" */
|
||||||
|
strbuf_addstr(&idx_file, ".idx");
|
||||||
|
new_pack = add_packed_git(idx_file.buf, idx_file.len, 1);
|
||||||
|
strbuf_release(&idx_file);
|
||||||
|
}
|
||||||
|
|
||||||
if (quiet)
|
if (quiet)
|
||||||
argv[5] = "--quiet";
|
argv[5] = "--quiet";
|
||||||
|
|
||||||
@ -42,6 +63,17 @@ int check_everything_connected(sha1_iterate_fn fn, int quiet, void *cb_data)
|
|||||||
|
|
||||||
commit[40] = '\n';
|
commit[40] = '\n';
|
||||||
do {
|
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(sha1, new_pack))
|
||||||
|
continue;
|
||||||
|
|
||||||
memcpy(commit, sha1_to_hex(sha1), 40);
|
memcpy(commit, sha1_to_hex(sha1), 40);
|
||||||
if (write_in_full(rev_list.in, commit, 41) < 0) {
|
if (write_in_full(rev_list.in, commit, 41) < 0) {
|
||||||
if (errno != EPIPE && errno != EINVAL)
|
if (errno != EPIPE && errno != EINVAL)
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
#ifndef CONNECTED_H
|
#ifndef CONNECTED_H
|
||||||
#define CONNECTED_H
|
#define CONNECTED_H
|
||||||
|
|
||||||
|
struct transport;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Take callback data, and return next object name in the buffer.
|
* Take callback data, and return next object name in the buffer.
|
||||||
* When called after returning the name for the last object, return -1
|
* When called after returning the name for the last object, return -1
|
||||||
@ -16,5 +18,8 @@ typedef int (*sha1_iterate_fn)(void *, unsigned char [20]);
|
|||||||
* Return 0 if Ok, non zero otherwise (i.e. some missing objects)
|
* Return 0 if Ok, non zero otherwise (i.e. some missing objects)
|
||||||
*/
|
*/
|
||||||
extern int check_everything_connected(sha1_iterate_fn, int quiet, void *cb_data);
|
extern int check_everything_connected(sha1_iterate_fn, int quiet, void *cb_data);
|
||||||
|
extern int check_everything_connected_with_transport(sha1_iterate_fn, int quiet,
|
||||||
|
void *cb_data,
|
||||||
|
struct transport *transport);
|
||||||
|
|
||||||
#endif /* CONNECTED_H */
|
#endif /* CONNECTED_H */
|
||||||
|
84
fetch-pack.c
84
fetch-pack.c
@ -20,6 +20,8 @@ static int no_done;
|
|||||||
static int fetch_fsck_objects = -1;
|
static int fetch_fsck_objects = -1;
|
||||||
static int transfer_fsck_objects = -1;
|
static int transfer_fsck_objects = -1;
|
||||||
static int agent_supported;
|
static int agent_supported;
|
||||||
|
static struct lock_file shallow_lock;
|
||||||
|
static const char *alternate_shallow_file;
|
||||||
|
|
||||||
#define COMPLETE (1U << 0)
|
#define COMPLETE (1U << 0)
|
||||||
#define COMMON (1U << 1)
|
#define COMMON (1U << 1)
|
||||||
@ -683,12 +685,13 @@ static int get_pack(struct fetch_pack_args *args,
|
|||||||
int xd[2], char **pack_lockfile)
|
int xd[2], char **pack_lockfile)
|
||||||
{
|
{
|
||||||
struct async demux;
|
struct async demux;
|
||||||
const char *argv[20];
|
const char *argv[22];
|
||||||
char keep_arg[256];
|
char keep_arg[256];
|
||||||
char hdr_arg[256];
|
char hdr_arg[256];
|
||||||
const char **av;
|
const char **av;
|
||||||
int do_keep = args->keep_pack;
|
int do_keep = args->keep_pack;
|
||||||
struct child_process cmd;
|
struct child_process cmd;
|
||||||
|
int ret;
|
||||||
|
|
||||||
memset(&demux, 0, sizeof(demux));
|
memset(&demux, 0, sizeof(demux));
|
||||||
if (use_sideband) {
|
if (use_sideband) {
|
||||||
@ -724,6 +727,11 @@ static int get_pack(struct fetch_pack_args *args,
|
|||||||
do_keep = 1;
|
do_keep = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (alternate_shallow_file) {
|
||||||
|
*av++ = "--shallow-file";
|
||||||
|
*av++ = alternate_shallow_file;
|
||||||
|
}
|
||||||
|
|
||||||
if (do_keep) {
|
if (do_keep) {
|
||||||
if (pack_lockfile)
|
if (pack_lockfile)
|
||||||
cmd.out = -1;
|
cmd.out = -1;
|
||||||
@ -740,11 +748,14 @@ static int get_pack(struct fetch_pack_args *args,
|
|||||||
strcpy(keep_arg + s, "localhost");
|
strcpy(keep_arg + s, "localhost");
|
||||||
*av++ = keep_arg;
|
*av++ = keep_arg;
|
||||||
}
|
}
|
||||||
|
if (args->check_self_contained_and_connected)
|
||||||
|
*av++ = "--check-self-contained-and-connected";
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
*av++ = "unpack-objects";
|
*av++ = "unpack-objects";
|
||||||
if (args->quiet || args->no_progress)
|
if (args->quiet || args->no_progress)
|
||||||
*av++ = "-q";
|
*av++ = "-q";
|
||||||
|
args->check_self_contained_and_connected = 0;
|
||||||
}
|
}
|
||||||
if (*hdr_arg)
|
if (*hdr_arg)
|
||||||
*av++ = hdr_arg;
|
*av++ = hdr_arg;
|
||||||
@ -765,7 +776,12 @@ static int get_pack(struct fetch_pack_args *args,
|
|||||||
close(cmd.out);
|
close(cmd.out);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (finish_command(&cmd))
|
ret = finish_command(&cmd);
|
||||||
|
if (!ret || (args->check_self_contained_and_connected && ret == 1))
|
||||||
|
args->self_contained_and_connected =
|
||||||
|
args->check_self_contained_and_connected &&
|
||||||
|
ret == 0;
|
||||||
|
else
|
||||||
die("%s failed", argv[0]);
|
die("%s failed", argv[0]);
|
||||||
if (use_sideband && finish_async(&demux))
|
if (use_sideband && finish_async(&demux))
|
||||||
die("error in sideband demultiplexer");
|
die("error in sideband demultiplexer");
|
||||||
@ -779,6 +795,27 @@ static int cmp_ref_by_name(const void *a_, const void *b_)
|
|||||||
return strcmp(a->name, b->name);
|
return strcmp(a->name, b->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void setup_alternate_shallow(void)
|
||||||
|
{
|
||||||
|
struct strbuf sb = STRBUF_INIT;
|
||||||
|
int fd;
|
||||||
|
|
||||||
|
check_shallow_file_for_update();
|
||||||
|
fd = hold_lock_file_for_update(&shallow_lock, git_path("shallow"),
|
||||||
|
LOCK_DIE_ON_ERROR);
|
||||||
|
if (write_shallow_commits(&sb, 0)) {
|
||||||
|
if (write_in_full(fd, sb.buf, sb.len) != sb.len)
|
||||||
|
die_errno("failed to write to %s", shallow_lock.filename);
|
||||||
|
alternate_shallow_file = shallow_lock.filename;
|
||||||
|
} else
|
||||||
|
/*
|
||||||
|
* is_repository_shallow() sees empty string as "no
|
||||||
|
* shallow file".
|
||||||
|
*/
|
||||||
|
alternate_shallow_file = "";
|
||||||
|
strbuf_release(&sb);
|
||||||
|
}
|
||||||
|
|
||||||
static struct ref *do_fetch_pack(struct fetch_pack_args *args,
|
static struct ref *do_fetch_pack(struct fetch_pack_args *args,
|
||||||
int fd[2],
|
int fd[2],
|
||||||
const struct ref *orig_ref,
|
const struct ref *orig_ref,
|
||||||
@ -858,6 +895,8 @@ static struct ref *do_fetch_pack(struct fetch_pack_args *args,
|
|||||||
|
|
||||||
if (args->stateless_rpc)
|
if (args->stateless_rpc)
|
||||||
packet_flush(fd[1]);
|
packet_flush(fd[1]);
|
||||||
|
if (args->depth > 0)
|
||||||
|
setup_alternate_shallow();
|
||||||
if (get_pack(args, fd, pack_lockfile))
|
if (get_pack(args, fd, pack_lockfile))
|
||||||
die("git fetch-pack: fetch failed.");
|
die("git fetch-pack: fetch failed.");
|
||||||
|
|
||||||
@ -936,15 +975,9 @@ struct ref *fetch_pack(struct fetch_pack_args *args,
|
|||||||
struct ref **sought, int nr_sought,
|
struct ref **sought, int nr_sought,
|
||||||
char **pack_lockfile)
|
char **pack_lockfile)
|
||||||
{
|
{
|
||||||
struct stat st;
|
|
||||||
struct ref *ref_cpy;
|
struct ref *ref_cpy;
|
||||||
|
|
||||||
fetch_pack_setup();
|
fetch_pack_setup();
|
||||||
if (args->depth > 0) {
|
|
||||||
if (stat(git_path("shallow"), &st))
|
|
||||||
st.st_mtime = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (nr_sought)
|
if (nr_sought)
|
||||||
nr_sought = remove_duplicates_in_refs(sought, nr_sought);
|
nr_sought = remove_duplicates_in_refs(sought, nr_sought);
|
||||||
|
|
||||||
@ -954,35 +987,12 @@ struct ref *fetch_pack(struct fetch_pack_args *args,
|
|||||||
}
|
}
|
||||||
ref_cpy = do_fetch_pack(args, fd, ref, sought, nr_sought, pack_lockfile);
|
ref_cpy = do_fetch_pack(args, fd, ref, sought, nr_sought, pack_lockfile);
|
||||||
|
|
||||||
if (args->depth > 0) {
|
if (alternate_shallow_file) {
|
||||||
static struct lock_file lock;
|
if (*alternate_shallow_file == '\0') { /* --unshallow */
|
||||||
struct cache_time mtime;
|
unlink_or_warn(git_path("shallow"));
|
||||||
struct strbuf sb = STRBUF_INIT;
|
rollback_lock_file(&shallow_lock);
|
||||||
char *shallow = git_path("shallow");
|
} else
|
||||||
int fd;
|
commit_lock_file(&shallow_lock);
|
||||||
|
|
||||||
mtime.sec = st.st_mtime;
|
|
||||||
mtime.nsec = ST_MTIME_NSEC(st);
|
|
||||||
if (stat(shallow, &st)) {
|
|
||||||
if (mtime.sec)
|
|
||||||
die("shallow file was removed during fetch");
|
|
||||||
} else if (st.st_mtime != mtime.sec
|
|
||||||
#ifdef USE_NSEC
|
|
||||||
|| ST_MTIME_NSEC(st) != mtime.nsec
|
|
||||||
#endif
|
|
||||||
)
|
|
||||||
die("shallow file was changed during fetch");
|
|
||||||
|
|
||||||
fd = hold_lock_file_for_update(&lock, shallow,
|
|
||||||
LOCK_DIE_ON_ERROR);
|
|
||||||
if (!write_shallow_commits(&sb, 0)
|
|
||||||
|| write_in_full(fd, sb.buf, sb.len) != sb.len) {
|
|
||||||
unlink_or_warn(shallow);
|
|
||||||
rollback_lock_file(&lock);
|
|
||||||
} else {
|
|
||||||
commit_lock_file(&lock);
|
|
||||||
}
|
|
||||||
strbuf_release(&sb);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
reprepare_packed_git();
|
reprepare_packed_git();
|
||||||
|
@ -16,7 +16,9 @@ struct fetch_pack_args {
|
|||||||
verbose:1,
|
verbose:1,
|
||||||
no_progress:1,
|
no_progress:1,
|
||||||
include_tag:1,
|
include_tag:1,
|
||||||
stateless_rpc:1;
|
stateless_rpc:1,
|
||||||
|
check_self_contained_and_connected:1,
|
||||||
|
self_contained_and_connected:1;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
7
git.c
7
git.c
@ -4,6 +4,7 @@
|
|||||||
#include "help.h"
|
#include "help.h"
|
||||||
#include "quote.h"
|
#include "quote.h"
|
||||||
#include "run-command.h"
|
#include "run-command.h"
|
||||||
|
#include "commit.h"
|
||||||
|
|
||||||
const char git_usage_string[] =
|
const char git_usage_string[] =
|
||||||
"git [--version] [--help] [-c name=value]\n"
|
"git [--version] [--help] [-c name=value]\n"
|
||||||
@ -146,6 +147,12 @@ static int handle_options(const char ***argv, int *argc, int *envchanged)
|
|||||||
setenv(GIT_LITERAL_PATHSPECS_ENVIRONMENT, "0", 1);
|
setenv(GIT_LITERAL_PATHSPECS_ENVIRONMENT, "0", 1);
|
||||||
if (envchanged)
|
if (envchanged)
|
||||||
*envchanged = 1;
|
*envchanged = 1;
|
||||||
|
} else if (!strcmp(cmd, "--shallow-file")) {
|
||||||
|
(*argv)++;
|
||||||
|
(*argc)--;
|
||||||
|
set_alternate_shallow_file((*argv)[0]);
|
||||||
|
if (envchanged)
|
||||||
|
*envchanged = 1;
|
||||||
} else {
|
} else {
|
||||||
fprintf(stderr, "Unknown option: %s\n", cmd);
|
fprintf(stderr, "Unknown option: %s\n", cmd);
|
||||||
usage(git_usage_string);
|
usage(git_usage_string);
|
||||||
|
42
shallow.c
42
shallow.c
@ -3,6 +3,16 @@
|
|||||||
#include "tag.h"
|
#include "tag.h"
|
||||||
|
|
||||||
static int is_shallow = -1;
|
static int is_shallow = -1;
|
||||||
|
static struct stat shallow_stat;
|
||||||
|
static char *alternate_shallow_file;
|
||||||
|
|
||||||
|
void set_alternate_shallow_file(const char *path)
|
||||||
|
{
|
||||||
|
if (is_shallow != -1)
|
||||||
|
die("BUG: is_repository_shallow must not be called before set_alternate_shallow_file");
|
||||||
|
free(alternate_shallow_file);
|
||||||
|
alternate_shallow_file = path ? xstrdup(path) : NULL;
|
||||||
|
}
|
||||||
|
|
||||||
int register_shallow(const unsigned char *sha1)
|
int register_shallow(const unsigned char *sha1)
|
||||||
{
|
{
|
||||||
@ -21,12 +31,21 @@ int is_repository_shallow(void)
|
|||||||
{
|
{
|
||||||
FILE *fp;
|
FILE *fp;
|
||||||
char buf[1024];
|
char buf[1024];
|
||||||
|
const char *path = alternate_shallow_file;
|
||||||
|
|
||||||
if (is_shallow >= 0)
|
if (is_shallow >= 0)
|
||||||
return is_shallow;
|
return is_shallow;
|
||||||
|
|
||||||
fp = fopen(git_path("shallow"), "r");
|
if (!path)
|
||||||
if (!fp) {
|
path = git_path("shallow");
|
||||||
|
/*
|
||||||
|
* fetch-pack sets '--shallow-file ""' as an indicator that no
|
||||||
|
* shallow file should be used. We could just open it and it
|
||||||
|
* will likely fail. But let's do an explicit check instead.
|
||||||
|
*/
|
||||||
|
if (!*path ||
|
||||||
|
stat(path, &shallow_stat) ||
|
||||||
|
(fp = fopen(path, "r")) == NULL) {
|
||||||
is_shallow = 0;
|
is_shallow = 0;
|
||||||
return is_shallow;
|
return is_shallow;
|
||||||
}
|
}
|
||||||
@ -108,3 +127,22 @@ struct commit_list *get_shallow_commits(struct object_array *heads, int depth,
|
|||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void check_shallow_file_for_update(void)
|
||||||
|
{
|
||||||
|
struct stat st;
|
||||||
|
|
||||||
|
if (!is_shallow)
|
||||||
|
return;
|
||||||
|
else if (is_shallow == -1)
|
||||||
|
die("BUG: shallow must be initialized by now");
|
||||||
|
|
||||||
|
if (stat(git_path("shallow"), &st))
|
||||||
|
die("shallow file was removed during fetch");
|
||||||
|
else if (st.st_mtime != shallow_stat.st_mtime
|
||||||
|
#ifdef USE_NSEC
|
||||||
|
|| ST_MTIME_NSEC(st) != ST_MTIME_NSEC(shallow_stat)
|
||||||
|
#endif
|
||||||
|
)
|
||||||
|
die("shallow file was changed during fetch");
|
||||||
|
}
|
||||||
|
@ -135,6 +135,13 @@ test_expect_success 'clone shallow depth 1' '
|
|||||||
test "`git --git-dir=shallow0/.git rev-list --count HEAD`" = 1
|
test "`git --git-dir=shallow0/.git rev-list --count HEAD`" = 1
|
||||||
'
|
'
|
||||||
|
|
||||||
|
test_expect_success 'clone shallow depth 1 with fsck' '
|
||||||
|
git config --global fetch.fsckobjects true &&
|
||||||
|
git clone --no-single-branch --depth 1 "file://$(pwd)/." shallow0fsck &&
|
||||||
|
test "`git --git-dir=shallow0fsck/.git rev-list --count HEAD`" = 1 &&
|
||||||
|
git config --global --unset fetch.fsckobjects
|
||||||
|
'
|
||||||
|
|
||||||
test_expect_success 'clone shallow' '
|
test_expect_success 'clone shallow' '
|
||||||
git clone --no-single-branch --depth 2 "file://$(pwd)/." shallow
|
git clone --no-single-branch --depth 2 "file://$(pwd)/." shallow
|
||||||
'
|
'
|
||||||
|
@ -534,6 +534,8 @@ static int fetch_refs_via_pack(struct transport *transport,
|
|||||||
args.quiet = (transport->verbose < 0);
|
args.quiet = (transport->verbose < 0);
|
||||||
args.no_progress = !transport->progress;
|
args.no_progress = !transport->progress;
|
||||||
args.depth = data->options.depth;
|
args.depth = data->options.depth;
|
||||||
|
args.check_self_contained_and_connected =
|
||||||
|
data->options.check_self_contained_and_connected;
|
||||||
|
|
||||||
if (!data->got_remote_heads) {
|
if (!data->got_remote_heads) {
|
||||||
connect_setup(transport, 0, 0);
|
connect_setup(transport, 0, 0);
|
||||||
@ -551,6 +553,8 @@ static int fetch_refs_via_pack(struct transport *transport,
|
|||||||
refs = NULL;
|
refs = NULL;
|
||||||
data->conn = NULL;
|
data->conn = NULL;
|
||||||
data->got_remote_heads = 0;
|
data->got_remote_heads = 0;
|
||||||
|
data->options.self_contained_and_connected =
|
||||||
|
args.self_contained_and_connected;
|
||||||
|
|
||||||
free_refs(refs_tmp);
|
free_refs(refs_tmp);
|
||||||
|
|
||||||
|
@ -8,6 +8,8 @@ struct git_transport_options {
|
|||||||
unsigned thin : 1;
|
unsigned thin : 1;
|
||||||
unsigned keep : 1;
|
unsigned keep : 1;
|
||||||
unsigned followtags : 1;
|
unsigned followtags : 1;
|
||||||
|
unsigned check_self_contained_and_connected : 1;
|
||||||
|
unsigned self_contained_and_connected : 1;
|
||||||
int depth;
|
int depth;
|
||||||
const char *uploadpack;
|
const char *uploadpack;
|
||||||
const char *receivepack;
|
const char *receivepack;
|
||||||
|
Loading…
Reference in New Issue
Block a user