1 #define USE_THE_REPOSITORY_VARIABLE
3 #include "git-compat-util.h"
6 #include "object-store-ll.h"
7 #include "run-command.h"
10 #include "transport.h"
12 #include "promisor-remote.h"
15 * If we feed all the commits we want to verify to this command
17 * $ git rev-list --objects --stdin --not --all
19 * and if it does not error out, that means everything reachable from
20 * these commits locally exists and is connected to our existing refs.
21 * Note that this does _not_ validate the individual objects.
23 * Returns 0 if everything is connected, non-zero otherwise.
25 int check_connected(oid_iterate_fn fn
, void *cb_data
,
26 struct check_connected_options
*opt
)
28 struct child_process rev_list
= CHILD_PROCESS_INIT
;
30 struct check_connected_options defaults
= CHECK_CONNECTED_INIT
;
31 const struct object_id
*oid
;
33 struct packed_git
*new_pack
= NULL
;
34 struct transport
*transport
;
39 transport
= opt
->transport
;
48 if (transport
&& transport
->smart_options
&&
49 transport
->smart_options
->self_contained_and_connected
&&
50 transport
->pack_lockfiles
.nr
== 1 &&
51 strip_suffix(transport
->pack_lockfiles
.items
[0].string
,
52 ".keep", &base_len
)) {
53 struct strbuf idx_file
= STRBUF_INIT
;
54 strbuf_add(&idx_file
, transport
->pack_lockfiles
.items
[0].string
,
56 strbuf_addstr(&idx_file
, ".idx");
57 new_pack
= add_packed_git(idx_file
.buf
, idx_file
.len
, 1);
58 strbuf_release(&idx_file
);
61 if (repo_has_promisor_remote(the_repository
)) {
63 * For partial clones, we don't want to have to do a regular
64 * connectivity check because we have to enumerate and exclude
65 * all promisor objects (slow), and then the connectivity check
66 * itself becomes a no-op because in a partial clone every
67 * object is a promisor object. Instead, just make sure we
68 * received, in a promisor packfile, the objects pointed to by
71 * Before checking for promisor packs, be sure we have the
72 * latest pack-files loaded into memory.
74 reprepare_packed_git(the_repository
);
78 for (p
= get_all_packs(the_repository
); p
; p
= p
->next
) {
79 if (!p
->pack_promisor
)
81 if (find_pack_entry_one(oid
->hash
, p
))
82 goto promisor_pack_found
;
85 * Fallback to rev-list with oid and the rest of the
86 * object IDs provided by fn.
88 goto no_promisor_pack_found
;
91 } while ((oid
= fn(cb_data
)) != NULL
);
96 no_promisor_pack_found
:
97 if (opt
->shallow_file
) {
98 strvec_push(&rev_list
.args
, "--shallow-file");
99 strvec_push(&rev_list
.args
, opt
->shallow_file
);
101 strvec_push(&rev_list
.args
,"rev-list");
102 strvec_push(&rev_list
.args
, "--objects");
103 strvec_push(&rev_list
.args
, "--stdin");
104 if (repo_has_promisor_remote(the_repository
))
105 strvec_push(&rev_list
.args
, "--exclude-promisor-objects");
106 if (!opt
->is_deepening_fetch
) {
107 strvec_push(&rev_list
.args
, "--not");
108 if (opt
->exclude_hidden_refs_section
)
109 strvec_pushf(&rev_list
.args
, "--exclude-hidden=%s",
110 opt
->exclude_hidden_refs_section
);
111 strvec_push(&rev_list
.args
, "--all");
113 strvec_push(&rev_list
.args
, "--quiet");
114 strvec_push(&rev_list
.args
, "--alternate-refs");
116 strvec_pushf(&rev_list
.args
, "--progress=%s",
117 _("Checking connectivity"));
119 rev_list
.git_cmd
= 1;
121 strvec_pushv(&rev_list
.env
, opt
->env
);
123 rev_list
.no_stdout
= 1;
125 rev_list
.err
= opt
->err_fd
;
127 rev_list
.no_stderr
= opt
->quiet
;
129 if (start_command(&rev_list
)) {
131 return error(_("Could not run 'git rev-list'"));
134 sigchain_push(SIGPIPE
, SIG_IGN
);
136 rev_list_in
= xfdopen(rev_list
.in
, "w");
140 * If index-pack already checked that:
141 * - there are no dangling pointers in the new pack
142 * - the pack is self contained
143 * Then if the updated ref is in the new pack, then we
144 * are sure the ref is good and not sending it to
145 * rev-list for verification.
147 if (new_pack
&& find_pack_entry_one(oid
->hash
, new_pack
))
150 if (fprintf(rev_list_in
, "%s\n", oid_to_hex(oid
)) < 0)
152 } while ((oid
= fn(cb_data
)) != NULL
);
154 if (ferror(rev_list_in
) || fflush(rev_list_in
)) {
155 if (errno
!= EPIPE
&& errno
!= EINVAL
)
156 error_errno(_("failed write to rev-list"));
160 if (fclose(rev_list_in
))
161 err
= error_errno(_("failed to close rev-list's stdin"));
163 sigchain_pop(SIGPIPE
);
165 return finish_command(&rev_list
) || err
;