blob: 8ed4a6feaac2868523e6516e02865132dedbc9f5 [file] [log] [blame]
Linus Torvaldsdef88e92005-07-04 13:26:53 -07001#include "cache.h"
Linus Torvaldsfb9040c2005-07-04 15:29:17 -07002#include "refs.h"
Linus Torvaldsdef88e92005-07-04 13:26:53 -07003#include "pkt-line.h"
Junio C Hamano49bb8052005-10-19 14:27:02 -07004#include "commit.h"
5#include "tag.h"
Nicolas Pitreda093d32006-11-01 17:06:23 -05006#include "exec_cmd.h"
Junio C Hamano9e10fd12007-01-22 22:37:33 -08007#include "pack.h"
Nicolas Pitreda093d32006-11-01 17:06:23 -05008#include "sideband.h"
Daniel Barkalow2d4177c2007-09-10 23:03:00 -04009#include "fetch-pack.h"
Daniel Barkalowba227852008-02-04 13:26:23 -050010#include "remote.h"
Johannes Sixt477822c2007-10-19 21:47:57 +020011#include "run-command.h"
Linus Torvaldsdef88e92005-07-04 13:26:53 -070012
Junio C Hamanoe28714c2007-01-24 17:02:15 -080013static int transfer_unpack_limit = -1;
14static int fetch_unpack_limit = -1;
Junio C Hamanoaf7cf262007-01-24 16:47:24 -080015static int unpack_limit = 100;
Nicolas Pitref04833e2009-05-01 20:18:02 -040016static int prefer_ofs_delta = 1;
Daniel Barkalow09149c72007-10-29 22:35:08 -040017static struct fetch_pack_args args = {
18 /* .uploadpack = */ "git-upload-pack",
19};
Shawn O. Pearcefa740522007-09-19 00:49:35 -040020
Junio C Hamano33b83032005-08-12 02:08:29 -070021static const char fetch_pack_usage[] =
Stephan Beyer1b1dd232008-07-13 15:36:15 +020022"git fetch-pack [--all] [--quiet|-q] [--keep|-k] [--thin] [--include-tag] [--upload-pack=<git-upload-pack>] [--depth=<n>] [--no-progress] [-v] [<host>:]<directory> [<refs>...]";
Linus Torvaldsdef88e92005-07-04 13:26:53 -070023
Johannes Schindelin0a8944d2005-10-19 16:14:34 -070024#define COMPLETE (1U << 0)
Johannes Schindelin23d61f82005-10-28 04:46:27 +020025#define COMMON (1U << 1)
26#define COMMON_REF (1U << 2)
27#define SEEN (1U << 3)
28#define POPPED (1U << 4)
29
Daniel Barkalow420e9af2008-03-17 22:15:02 -040030static int marked;
31
Junio C Hamanof061e5f2006-05-24 21:48:34 -070032/*
33 * After sending this many "have"s if we do not get any new ACK , we
34 * give up traversing our history.
35 */
36#define MAX_IN_VAIN 256
37
David Rientjes96f1e582006-08-15 10:23:48 -070038static struct commit_list *rev_list;
Nicolas Pitre3891f392007-11-07 17:20:22 -050039static int non_common_revs, multi_ack, use_sideband;
Johannes Schindelin23d61f82005-10-28 04:46:27 +020040
41static void rev_list_push(struct commit *commit, int mark)
42{
43 if (!(commit->object.flags & mark)) {
44 commit->object.flags |= mark;
45
46 if (!(commit->object.parsed))
Martin Koeglerf3ec5492008-03-03 07:31:23 +010047 if (parse_commit(commit))
48 return;
Johannes Schindelin23d61f82005-10-28 04:46:27 +020049
50 insert_by_date(commit, &rev_list);
51
52 if (!(commit->object.flags & COMMON))
53 non_common_revs++;
54 }
55}
56
Junio C Hamano8da19772006-09-20 22:02:01 -070057static int rev_list_insert_ref(const char *path, const unsigned char *sha1, int flag, void *cb_data)
Johannes Schindelin23d61f82005-10-28 04:46:27 +020058{
Junio C Hamano9534f402005-11-02 15:19:13 -080059 struct object *o = deref_tag(parse_object(sha1), path, 0);
Johannes Schindelin23d61f82005-10-28 04:46:27 +020060
Linus Torvalds19746322006-07-11 20:45:31 -070061 if (o && o->type == OBJ_COMMIT)
Johannes Schindelin23d61f82005-10-28 04:46:27 +020062 rev_list_push((struct commit *)o, SEEN);
63
64 return 0;
65}
66
Daniel Barkalow420e9af2008-03-17 22:15:02 -040067static int clear_marks(const char *path, const unsigned char *sha1, int flag, void *cb_data)
68{
69 struct object *o = deref_tag(parse_object(sha1), path, 0);
70
71 if (o && o->type == OBJ_COMMIT)
72 clear_commit_marks((struct commit *)o,
73 COMMON | COMMON_REF | SEEN | POPPED);
74 return 0;
75}
76
Johannes Schindelin23d61f82005-10-28 04:46:27 +020077/*
78 This function marks a rev and its ancestors as common.
79 In some cases, it is desirable to mark only the ancestors (for example
80 when only the server does not yet know that they are common).
81*/
82
83static void mark_common(struct commit *commit,
84 int ancestors_only, int dont_parse)
85{
86 if (commit != NULL && !(commit->object.flags & COMMON)) {
87 struct object *o = (struct object *)commit;
88
89 if (!ancestors_only)
90 o->flags |= COMMON;
91
92 if (!(o->flags & SEEN))
93 rev_list_push(commit, SEEN);
94 else {
95 struct commit_list *parents;
96
97 if (!ancestors_only && !(o->flags & POPPED))
98 non_common_revs--;
99 if (!o->parsed && !dont_parse)
Martin Koeglerf3ec5492008-03-03 07:31:23 +0100100 if (parse_commit(commit))
101 return;
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200102
103 for (parents = commit->parents;
104 parents;
105 parents = parents->next)
106 mark_common(parents->item, 0, dont_parse);
107 }
108 }
109}
110
111/*
112 Get the next rev to send, ignoring the common.
113*/
114
Felipe Contreras4b25d092009-05-01 12:06:36 +0300115static const unsigned char *get_rev(void)
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200116{
117 struct commit *commit = NULL;
118
119 while (commit == NULL) {
120 unsigned int mark;
Linus Torvalds72269ad2008-04-28 16:27:49 -0700121 struct commit_list *parents;
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200122
123 if (rev_list == NULL || non_common_revs == 0)
124 return NULL;
125
126 commit = rev_list->item;
Junio C Hamano2d8bed92008-04-30 11:42:05 -0700127 if (!commit->object.parsed)
Linus Torvalds72269ad2008-04-28 16:27:49 -0700128 parse_commit(commit);
129 parents = commit->parents;
Martin Koeglerf3ec5492008-03-03 07:31:23 +0100130
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200131 commit->object.flags |= POPPED;
132 if (!(commit->object.flags & COMMON))
133 non_common_revs--;
Junio C Hamanoa6080a02007-06-07 00:04:01 -0700134
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200135 if (commit->object.flags & COMMON) {
136 /* do not send "have", and ignore ancestors */
137 commit = NULL;
138 mark = COMMON | SEEN;
139 } else if (commit->object.flags & COMMON_REF)
140 /* send "have", and ignore ancestors */
141 mark = COMMON | SEEN;
142 else
143 /* send "have", also for its ancestors */
144 mark = SEEN;
145
146 while (parents) {
147 if (!(parents->item->object.flags & SEEN))
148 rev_list_push(parents->item, mark);
149 if (mark & COMMON)
150 mark_common(parents->item, 1, 0);
151 parents = parents->next;
152 }
153
154 rev_list = rev_list->next;
155 }
156
157 return commit->object.sha1;
158}
Johannes Schindelin0a8944d2005-10-19 16:14:34 -0700159
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700160enum ack_type {
161 NAK = 0,
162 ACK,
163 ACK_continue,
164 ACK_common,
165 ACK_ready
166};
167
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700168static void consume_shallow_list(int fd)
169{
170 if (args.stateless_rpc && args.depth > 0) {
171 /* If we sent a depth we will get back "duplicate"
172 * shallow and unshallow commands every time there
173 * is a block of have lines exchanged.
174 */
175 char line[1000];
176 while (packet_read_line(fd, line, sizeof(line))) {
177 if (!prefixcmp(line, "shallow "))
178 continue;
179 if (!prefixcmp(line, "unshallow "))
180 continue;
181 die("git fetch-pack: expected shallow list");
182 }
183 }
184}
185
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700186static enum ack_type get_ack(int fd, unsigned char *result_sha1)
Shawn O. Pearce28754ab2009-10-30 17:47:24 -0700187{
188 static char line[1000];
189 int len = packet_read_line(fd, line, sizeof(line));
190
191 if (!len)
192 die("git fetch-pack: expected ACK/NAK, got EOF");
193 if (line[len-1] == '\n')
194 line[--len] = 0;
195 if (!strcmp(line, "NAK"))
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700196 return NAK;
Shawn O. Pearce28754ab2009-10-30 17:47:24 -0700197 if (!prefixcmp(line, "ACK ")) {
198 if (!get_sha1_hex(line+4, result_sha1)) {
199 if (strstr(line+45, "continue"))
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700200 return ACK_continue;
201 if (strstr(line+45, "common"))
202 return ACK_common;
203 if (strstr(line+45, "ready"))
204 return ACK_ready;
205 return ACK;
Shawn O. Pearce28754ab2009-10-30 17:47:24 -0700206 }
207 }
208 die("git fetch_pack: expected ACK/NAK, got '%s'", line);
209}
210
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700211static void send_request(int fd, struct strbuf *buf)
212{
213 if (args.stateless_rpc) {
214 send_sideband(fd, -1, buf->buf, buf->len, LARGE_PACKET_MAX);
215 packet_flush(fd);
216 } else
217 safe_write(fd, buf->buf, buf->len);
218}
219
Junio C Hamano33b83032005-08-12 02:08:29 -0700220static int find_common(int fd[2], unsigned char *result_sha1,
221 struct ref *refs)
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700222{
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700223 int fetching;
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200224 int count = 0, flushes = 0, retval;
225 const unsigned char *sha1;
Junio C Hamanof061e5f2006-05-24 21:48:34 -0700226 unsigned in_vain = 0;
227 int got_continue = 0;
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700228 struct strbuf req_buf = STRBUF_INIT;
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700229 size_t state_len = 0;
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700230
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700231 if (args.stateless_rpc && multi_ack == 1)
232 die("--stateless-rpc requires multi_ack_detailed");
Daniel Barkalow420e9af2008-03-17 22:15:02 -0400233 if (marked)
234 for_each_ref(clear_marks, NULL);
235 marked = 1;
236
Junio C Hamanocb5d7092006-09-20 21:47:42 -0700237 for_each_ref(rev_list_insert_ref, NULL);
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200238
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700239 fetching = 0;
240 for ( ; refs ; refs = refs->next) {
Junio C Hamano33b83032005-08-12 02:08:29 -0700241 unsigned char *remote = refs->old_sha1;
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700242 const char *remote_hex;
Junio C Hamano4dab94d2005-10-19 18:28:17 -0700243 struct object *o;
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700244
Johannes Schindelin0a8944d2005-10-19 16:14:34 -0700245 /*
Junio C Hamano4dab94d2005-10-19 18:28:17 -0700246 * If that object is complete (i.e. it is an ancestor of a
247 * local ref), we tell them we have it but do not have to
248 * tell them about its ancestors, which they already know
249 * about.
Junio C Hamanof1f0a2b2005-10-19 21:55:49 -0700250 *
251 * We use lookup_object here because we are only
252 * interested in the case we *know* the object is
253 * reachable and we have already scanned it.
Junio C Hamano4dab94d2005-10-19 18:28:17 -0700254 */
Junio C Hamanof1f0a2b2005-10-19 21:55:49 -0700255 if (((o = lookup_object(remote)) != NULL) &&
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200256 (o->flags & COMPLETE)) {
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700257 continue;
Johannes Schindelin0a8944d2005-10-19 16:14:34 -0700258 }
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200259
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700260 remote_hex = sha1_to_hex(remote);
261 if (!fetching) {
262 struct strbuf c = STRBUF_INIT;
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700263 if (multi_ack == 2) strbuf_addstr(&c, " multi_ack_detailed");
264 if (multi_ack == 1) strbuf_addstr(&c, " multi_ack");
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700265 if (use_sideband == 2) strbuf_addstr(&c, " side-band-64k");
266 if (use_sideband == 1) strbuf_addstr(&c, " side-band");
267 if (args.use_thin_pack) strbuf_addstr(&c, " thin-pack");
268 if (args.no_progress) strbuf_addstr(&c, " no-progress");
269 if (args.include_tag) strbuf_addstr(&c, " include-tag");
270 if (prefer_ofs_delta) strbuf_addstr(&c, " ofs-delta");
271 packet_buf_write(&req_buf, "want %s%s\n", remote_hex, c.buf);
272 strbuf_release(&c);
273 } else
274 packet_buf_write(&req_buf, "want %s\n", remote_hex);
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700275 fetching++;
Junio C Hamano33b83032005-08-12 02:08:29 -0700276 }
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700277
278 if (!fetching) {
279 strbuf_release(&req_buf);
280 packet_flush(fd[1]);
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700281 return 1;
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700282 }
283
284 if (is_repository_shallow())
285 write_shallow_commits(&req_buf, 1);
286 if (args.depth > 0)
287 packet_buf_write(&req_buf, "deepen %d", args.depth);
288 packet_buf_flush(&req_buf);
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700289 state_len = req_buf.len;
Johannes Schindelin0a8944d2005-10-19 16:14:34 -0700290
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400291 if (args.depth > 0) {
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100292 char line[1024];
293 unsigned char sha1[20];
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100294
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700295 send_request(fd[1], &req_buf);
Benjamin Kramereb3a9dd2009-03-07 21:02:10 +0100296 while (packet_read_line(fd[0], line, sizeof(line))) {
Junio C Hamano599065a2007-02-20 01:54:00 -0800297 if (!prefixcmp(line, "shallow ")) {
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100298 if (get_sha1_hex(line + 8, sha1))
299 die("invalid shallow line: %s", line);
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100300 register_shallow(sha1);
Junio C Hamanocf01bd52006-11-13 22:04:56 -0800301 continue;
302 }
Junio C Hamano599065a2007-02-20 01:54:00 -0800303 if (!prefixcmp(line, "unshallow ")) {
Johannes Schindelinf53514b2006-10-30 20:09:53 +0100304 if (get_sha1_hex(line + 10, sha1))
305 die("invalid unshallow line: %s", line);
306 if (!lookup_object(sha1))
307 die("object not found: %s", line);
308 /* make sure that it is parsed as shallow */
Martin Koeglerf3ec5492008-03-03 07:31:23 +0100309 if (!parse_object(sha1))
310 die("error in object: %s", line);
Johannes Schindelinf53514b2006-10-30 20:09:53 +0100311 if (unregister_shallow(sha1))
312 die("no shallow found: %s", line);
Junio C Hamanocf01bd52006-11-13 22:04:56 -0800313 continue;
314 }
315 die("expected shallow/unshallow, got %s", line);
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100316 }
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700317 } else if (!args.stateless_rpc)
318 send_request(fd[1], &req_buf);
319
320 if (!args.stateless_rpc) {
321 /* If we aren't using the stateless-rpc interface
322 * we don't need to retain the headers.
323 */
324 strbuf_setlen(&req_buf, 0);
325 state_len = 0;
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100326 }
327
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200328 flushes = 0;
Linus Torvalds75bfc6c2005-07-04 16:35:13 -0700329 retval = -1;
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200330 while ((sha1 = get_rev())) {
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700331 packet_buf_write(&req_buf, "have %s\n", sha1_to_hex(sha1));
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400332 if (args.verbose)
Junio C Hamano33b83032005-08-12 02:08:29 -0700333 fprintf(stderr, "have %s\n", sha1_to_hex(sha1));
Junio C Hamanof061e5f2006-05-24 21:48:34 -0700334 in_vain++;
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700335 if (!(31 & ++count)) {
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200336 int ack;
337
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700338 packet_buf_flush(&req_buf);
339 send_request(fd[1], &req_buf);
340 strbuf_setlen(&req_buf, state_len);
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700341 flushes++;
342
343 /*
344 * We keep one window "ahead" of the other side, and
345 * will wait for an ACK only on the next one
346 */
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700347 if (!args.stateless_rpc && count == 32)
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700348 continue;
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200349
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700350 consume_shallow_list(fd[0]);
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200351 do {
352 ack = get_ack(fd[0], result_sha1);
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400353 if (args.verbose && ack)
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200354 fprintf(stderr, "got ack %d %s\n", ack,
355 sha1_to_hex(result_sha1));
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700356 switch (ack) {
357 case ACK:
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200358 flushes = 0;
359 multi_ack = 0;
360 retval = 0;
361 goto done;
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700362 case ACK_common:
363 case ACK_ready:
364 case ACK_continue: {
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200365 struct commit *commit =
366 lookup_commit(result_sha1);
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700367 if (args.stateless_rpc
368 && ack == ACK_common
369 && !(commit->object.flags & COMMON)) {
370 /* We need to replay the have for this object
371 * on the next RPC request so the peer knows
372 * it is in common with us.
373 */
374 const char *hex = sha1_to_hex(result_sha1);
375 packet_buf_write(&req_buf, "have %s\n", hex);
376 state_len = req_buf.len;
377 }
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200378 mark_common(commit, 0, 1);
379 retval = 0;
Junio C Hamanof061e5f2006-05-24 21:48:34 -0700380 in_vain = 0;
381 got_continue = 1;
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700382 break;
383 }
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200384 }
385 } while (ack);
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700386 flushes--;
Junio C Hamanof061e5f2006-05-24 21:48:34 -0700387 if (got_continue && MAX_IN_VAIN < in_vain) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400388 if (args.verbose)
Junio C Hamanof061e5f2006-05-24 21:48:34 -0700389 fprintf(stderr, "giving up\n");
390 break; /* give up */
391 }
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700392 }
393 }
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200394done:
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700395 packet_buf_write(&req_buf, "done\n");
396 send_request(fd[1], &req_buf);
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400397 if (args.verbose)
Junio C Hamano33b83032005-08-12 02:08:29 -0700398 fprintf(stderr, "done\n");
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200399 if (retval != 0) {
400 multi_ack = 0;
Johannes Schindelin23d61f82005-10-28 04:46:27 +0200401 flushes++;
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200402 }
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700403 strbuf_release(&req_buf);
404
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700405 consume_shallow_list(fd[0]);
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200406 while (flushes || multi_ack) {
407 int ack = get_ack(fd[0], result_sha1);
408 if (ack) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400409 if (args.verbose)
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200410 fprintf(stderr, "got ack (%d) %s\n", ack,
411 sha1_to_hex(result_sha1));
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700412 if (ack == ACK)
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200413 return 0;
414 multi_ack = 1;
415 continue;
Junio C Hamano33b83032005-08-12 02:08:29 -0700416 }
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200417 flushes--;
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700418 }
Johannes Schindelin8cb560f2008-07-02 18:06:56 +0100419 /* it is no error to fetch into a completely empty repo */
420 return count ? retval : 0;
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700421}
422
David Rientjes96f1e582006-08-15 10:23:48 -0700423static struct commit_list *complete;
Junio C Hamano49bb8052005-10-19 14:27:02 -0700424
Junio C Hamano8da19772006-09-20 22:02:01 -0700425static int mark_complete(const char *path, const unsigned char *sha1, int flag, void *cb_data)
Junio C Hamano49bb8052005-10-19 14:27:02 -0700426{
427 struct object *o = parse_object(sha1);
428
Linus Torvalds19746322006-07-11 20:45:31 -0700429 while (o && o->type == OBJ_TAG) {
Junio C Hamanof1f0a2b2005-10-19 21:55:49 -0700430 struct tag *t = (struct tag *) o;
431 if (!t->tagged)
432 break; /* broken repository */
Junio C Hamano49bb8052005-10-19 14:27:02 -0700433 o->flags |= COMPLETE;
Junio C Hamanof1f0a2b2005-10-19 21:55:49 -0700434 o = parse_object(t->tagged->sha1);
Junio C Hamano49bb8052005-10-19 14:27:02 -0700435 }
Linus Torvalds19746322006-07-11 20:45:31 -0700436 if (o && o->type == OBJ_COMMIT) {
Junio C Hamano49bb8052005-10-19 14:27:02 -0700437 struct commit *commit = (struct commit *)o;
438 commit->object.flags |= COMPLETE;
439 insert_by_date(commit, &complete);
440 }
441 return 0;
442}
443
444static void mark_recent_complete_commits(unsigned long cutoff)
445{
446 while (complete && cutoff <= complete->item->date) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400447 if (args.verbose)
Junio C Hamano49bb8052005-10-19 14:27:02 -0700448 fprintf(stderr, "Marking %s as complete\n",
449 sha1_to_hex(complete->item->object.sha1));
450 pop_most_recent_commit(&complete, COMPLETE);
451 }
452}
453
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200454static void filter_refs(struct ref **refs, int nr_match, char **match)
455{
Junio C Hamano95460102006-05-11 15:28:44 -0700456 struct ref **return_refs;
457 struct ref *newlist = NULL;
458 struct ref **newtail = &newlist;
459 struct ref *ref, *next;
460 struct ref *fastarray[32];
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200461
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400462 if (nr_match && !args.fetch_all) {
Junio C Hamano95460102006-05-11 15:28:44 -0700463 if (ARRAY_SIZE(fastarray) < nr_match)
464 return_refs = xcalloc(nr_match, sizeof(struct ref *));
465 else {
466 return_refs = fastarray;
467 memset(return_refs, 0, sizeof(struct ref *) * nr_match);
468 }
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200469 }
Junio C Hamano95460102006-05-11 15:28:44 -0700470 else
471 return_refs = NULL;
472
473 for (ref = *refs; ref; ref = next) {
474 next = ref->next;
475 if (!memcmp(ref->name, "refs/", 5) &&
476 check_ref_format(ref->name + 5))
477 ; /* trash */
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400478 else if (args.fetch_all &&
479 (!args.depth || prefixcmp(ref->name, "refs/tags/") )) {
Junio C Hamano95460102006-05-11 15:28:44 -0700480 *newtail = ref;
481 ref->next = NULL;
482 newtail = &ref->next;
483 continue;
484 }
485 else {
486 int order = path_match(ref->name, nr_match, match);
487 if (order) {
488 return_refs[order-1] = ref;
489 continue; /* we will link it later */
490 }
491 }
492 free(ref);
493 }
494
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400495 if (!args.fetch_all) {
Junio C Hamano95460102006-05-11 15:28:44 -0700496 int i;
497 for (i = 0; i < nr_match; i++) {
498 ref = return_refs[i];
499 if (ref) {
500 *newtail = ref;
501 ref->next = NULL;
502 newtail = &ref->next;
503 }
504 }
505 if (return_refs != fastarray)
506 free(return_refs);
507 }
508 *refs = newlist;
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200509}
510
511static int everything_local(struct ref **refs, int nr_match, char **match)
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700512{
Junio C Hamano49bb8052005-10-19 14:27:02 -0700513 struct ref *ref;
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700514 int retval;
Junio C Hamano49bb8052005-10-19 14:27:02 -0700515 unsigned long cutoff = 0;
516
Junio C Hamano49bb8052005-10-19 14:27:02 -0700517 save_commit_buffer = 0;
518
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200519 for (ref = *refs; ref; ref = ref->next) {
Junio C Hamano49bb8052005-10-19 14:27:02 -0700520 struct object *o;
521
522 o = parse_object(ref->old_sha1);
523 if (!o)
524 continue;
525
526 /* We already have it -- which may mean that we were
527 * in sync with the other side at some time after
528 * that (it is OK if we guess wrong here).
529 */
Linus Torvalds19746322006-07-11 20:45:31 -0700530 if (o->type == OBJ_COMMIT) {
Junio C Hamano49bb8052005-10-19 14:27:02 -0700531 struct commit *commit = (struct commit *)o;
532 if (!cutoff || cutoff < commit->date)
533 cutoff = commit->date;
534 }
535 }
536
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400537 if (!args.depth) {
Johannes Schindelinf53514b2006-10-30 20:09:53 +0100538 for_each_ref(mark_complete, NULL);
539 if (cutoff)
540 mark_recent_complete_commits(cutoff);
541 }
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700542
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200543 /*
544 * Mark all complete remote refs as common refs.
545 * Don't mark them common yet; the server has to be told so first.
546 */
547 for (ref = *refs; ref; ref = ref->next) {
Junio C Hamano9534f402005-11-02 15:19:13 -0800548 struct object *o = deref_tag(lookup_object(ref->old_sha1),
549 NULL, 0);
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200550
Linus Torvalds19746322006-07-11 20:45:31 -0700551 if (!o || o->type != OBJ_COMMIT || !(o->flags & COMPLETE))
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200552 continue;
553
554 if (!(o->flags & SEEN)) {
555 rev_list_push((struct commit *)o, COMMON_REF | SEEN);
556
557 mark_common((struct commit *)o, 1, 1);
558 }
559 }
560
561 filter_refs(refs, nr_match, match);
562
563 for (retval = 1, ref = *refs; ref ; ref = ref->next) {
564 const unsigned char *remote = ref->old_sha1;
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700565 unsigned char local[20];
Junio C Hamano49bb8052005-10-19 14:27:02 -0700566 struct object *o;
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700567
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200568 o = lookup_object(remote);
Junio C Hamano49bb8052005-10-19 14:27:02 -0700569 if (!o || !(o->flags & COMPLETE)) {
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700570 retval = 0;
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400571 if (!args.verbose)
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700572 continue;
573 fprintf(stderr,
574 "want %s (%s)\n", sha1_to_hex(remote),
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200575 ref->name);
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700576 continue;
577 }
578
Shawn Pearcee7024962006-08-23 02:49:00 -0400579 hashcpy(ref->new_sha1, local);
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400580 if (!args.verbose)
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700581 continue;
582 fprintf(stderr,
583 "already have %s (%s)\n", sha1_to_hex(remote),
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200584 ref->name);
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700585 }
586 return retval;
587}
588
Johannes Sixt088fab52007-10-19 21:48:01 +0200589static int sideband_demux(int fd, void *data)
Nicolas Pitreda093d32006-11-01 17:06:23 -0500590{
Johannes Sixt088fab52007-10-19 21:48:01 +0200591 int *xd = data;
Nicolas Pitreda093d32006-11-01 17:06:23 -0500592
Johannes Sixt3ef67cf2009-06-08 10:51:22 +0200593 int ret = recv_sideband("fetch-pack", xd[0], fd);
594 close(fd);
595 return ret;
Johannes Sixt088fab52007-10-19 21:48:01 +0200596}
597
Shawn O. Pearce1788c392007-09-14 03:31:23 -0400598static int get_pack(int xd[2], char **pack_lockfile)
Nicolas Pitreda093d32006-11-01 17:06:23 -0500599{
Johannes Sixt088fab52007-10-19 21:48:01 +0200600 struct async demux;
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800601 const char *argv[20];
602 char keep_arg[256];
603 char hdr_arg[256];
604 const char **av;
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400605 int do_keep = args.keep_pack;
Johannes Sixt477822c2007-10-19 21:47:57 +0200606 struct child_process cmd;
Nicolas Pitreda093d32006-11-01 17:06:23 -0500607
Johannes Sixt1f759ee2007-11-17 23:09:28 +0100608 memset(&demux, 0, sizeof(demux));
609 if (use_sideband) {
610 /* xd[] is talking with upload-pack; subprocess reads from
611 * xd[0], spits out band#2 to stderr, and feeds us band#1
612 * through demux->out.
613 */
614 demux.proc = sideband_demux;
615 demux.data = xd;
616 if (start_async(&demux))
617 die("fetch-pack: unable to fork off sideband"
618 " demultiplexer");
619 }
620 else
621 demux.out = xd[0];
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800622
Johannes Sixt477822c2007-10-19 21:47:57 +0200623 memset(&cmd, 0, sizeof(cmd));
624 cmd.argv = argv;
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800625 av = argv;
626 *hdr_arg = 0;
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400627 if (!args.keep_pack && unpack_limit) {
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800628 struct pack_header header;
629
Johannes Sixt1f759ee2007-11-17 23:09:28 +0100630 if (read_pack_header(demux.out, &header))
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800631 die("protocol error: bad pack header");
Ramsay Jones6e1c2342008-07-03 16:52:09 +0100632 snprintf(hdr_arg, sizeof(hdr_arg),
633 "--pack_header=%"PRIu32",%"PRIu32,
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800634 ntohl(header.hdr_version), ntohl(header.hdr_entries));
Junio C Hamanoaf7cf262007-01-24 16:47:24 -0800635 if (ntohl(header.hdr_entries) < unpack_limit)
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800636 do_keep = 0;
637 else
638 do_keep = 1;
639 }
640
641 if (do_keep) {
Johannes Sixt477822c2007-10-19 21:47:57 +0200642 if (pack_lockfile)
643 cmd.out = -1;
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800644 *av++ = "index-pack";
645 *av++ = "--stdin";
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400646 if (!args.quiet && !args.no_progress)
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800647 *av++ = "-v";
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400648 if (args.use_thin_pack)
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800649 *av++ = "--fix-thin";
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400650 if (args.lock_pack || unpack_limit) {
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800651 int s = sprintf(keep_arg,
David Soria Parra85e72832008-08-31 14:09:39 +0200652 "--keep=fetch-pack %"PRIuMAX " on ", (uintmax_t) getpid());
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800653 if (gethostname(keep_arg + s, sizeof(keep_arg) - s))
654 strcpy(keep_arg + s, "localhost");
655 *av++ = keep_arg;
656 }
657 }
658 else {
659 *av++ = "unpack-objects";
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400660 if (args.quiet)
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800661 *av++ = "-q";
662 }
663 if (*hdr_arg)
664 *av++ = hdr_arg;
665 *av++ = NULL;
666
Johannes Sixt1f759ee2007-11-17 23:09:28 +0100667 cmd.in = demux.out;
Johannes Sixt477822c2007-10-19 21:47:57 +0200668 cmd.git_cmd = 1;
669 if (start_command(&cmd))
Nicolas Pitreda093d32006-11-01 17:06:23 -0500670 die("fetch-pack: unable to fork off %s", argv[0]);
Johannes Sixte72ae282008-02-16 18:36:38 +0100671 if (do_keep && pack_lockfile) {
Johannes Sixt477822c2007-10-19 21:47:57 +0200672 *pack_lockfile = index_pack_lockfile(cmd.out);
Johannes Sixte72ae282008-02-16 18:36:38 +0100673 close(cmd.out);
674 }
Johannes Sixt477822c2007-10-19 21:47:57 +0200675
676 if (finish_command(&cmd))
677 die("%s failed", argv[0]);
Johannes Sixt088fab52007-10-19 21:48:01 +0200678 if (use_sideband && finish_async(&demux))
679 die("error in sideband demultiplexer");
Johannes Sixt477822c2007-10-19 21:47:57 +0200680 return 0;
Nicolas Pitreda093d32006-11-01 17:06:23 -0500681}
682
Shawn O. Pearce1788c392007-09-14 03:31:23 -0400683static struct ref *do_fetch_pack(int fd[2],
Daniel Barkalowba227852008-02-04 13:26:23 -0500684 const struct ref *orig_ref,
Shawn O. Pearce1788c392007-09-14 03:31:23 -0400685 int nr_match,
686 char **match,
687 char **pack_lockfile)
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700688{
Daniel Barkalowba227852008-02-04 13:26:23 -0500689 struct ref *ref = copy_ref_list(orig_ref);
Linus Torvaldsd1c133f2005-07-16 13:55:50 -0700690 unsigned char sha1[20];
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700691
Johannes Schindelined09aef2006-10-30 20:09:06 +0100692 if (is_repository_shallow() && !server_supports("shallow"))
693 die("Server does not support shallow clients");
Shawn O. Pearce78affc42009-10-30 17:47:25 -0700694 if (server_supports("multi_ack_detailed")) {
695 if (args.verbose)
696 fprintf(stderr, "Server supports multi_ack_detailed\n");
697 multi_ack = 2;
698 }
699 else if (server_supports("multi_ack")) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400700 if (args.verbose)
Johannes Schindelinc4c86f02005-10-28 04:50:26 +0200701 fprintf(stderr, "Server supports multi_ack\n");
702 multi_ack = 1;
703 }
Junio C Hamanod47f3db2006-09-10 16:27:08 -0700704 if (server_supports("side-band-64k")) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400705 if (args.verbose)
Junio C Hamanod47f3db2006-09-10 16:27:08 -0700706 fprintf(stderr, "Server supports side-band-64k\n");
707 use_sideband = 2;
708 }
709 else if (server_supports("side-band")) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400710 if (args.verbose)
Junio C Hamano583b7ea2006-06-21 00:30:21 -0700711 fprintf(stderr, "Server supports side-band\n");
712 use_sideband = 1;
713 }
Nicolas Pitref04833e2009-05-01 20:18:02 -0400714 if (server_supports("ofs-delta")) {
715 if (args.verbose)
716 fprintf(stderr, "Server supports ofs-delta\n");
717 } else
718 prefer_ofs_delta = 0;
Johannes Schindelin1baaae52005-10-28 04:47:07 +0200719 if (everything_local(&ref, nr_match, match)) {
Linus Torvalds2759cbc2005-10-18 11:35:17 -0700720 packet_flush(fd[1]);
721 goto all_done;
722 }
Junio C Hamano33b83032005-08-12 02:08:29 -0700723 if (find_common(fd, sha1, ref) < 0)
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400724 if (!args.keep_pack)
Junio C Hamanodfeff662006-03-20 00:21:10 -0800725 /* When cloning, it is not unusual to have
726 * no common commit.
727 */
Miklos Vajna78509d22009-03-24 02:09:12 +0100728 warning("no common commits");
Junio C Hamanoad897212005-12-14 22:17:38 -0800729
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700730 if (args.stateless_rpc)
731 packet_flush(fd[1]);
Shawn O. Pearce1788c392007-09-14 03:31:23 -0400732 if (get_pack(fd, pack_lockfile))
Junio C Hamano7e44c932008-08-31 09:39:19 -0700733 die("git fetch-pack: fetch failed.");
Junio C Hamanoad897212005-12-14 22:17:38 -0800734
735 all_done:
Daniel Barkalow2d4177c2007-09-10 23:03:00 -0400736 return ref;
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700737}
738
Junio C Hamano310b86d2006-11-25 01:33:06 -0800739static int remove_duplicates(int nr_heads, char **heads)
740{
741 int src, dst;
742
743 for (src = dst = 0; src < nr_heads; src++) {
744 /* If heads[src] is different from any of
745 * heads[0..dst], push it in.
746 */
747 int i;
748 for (i = 0; i < dst; i++) {
749 if (!strcmp(heads[i], heads[src]))
750 break;
751 }
752 if (i < dst)
753 continue;
754 if (src != dst)
755 heads[dst] = heads[src];
756 dst++;
757 }
Junio C Hamano310b86d2006-11-25 01:33:06 -0800758 return dst;
759}
760
Johannes Schindelinef90d6d2008-05-14 18:46:53 +0100761static int fetch_pack_config(const char *var, const char *value, void *cb)
Junio C Hamanoaf7cf262007-01-24 16:47:24 -0800762{
763 if (strcmp(var, "fetch.unpacklimit") == 0) {
Junio C Hamanoe28714c2007-01-24 17:02:15 -0800764 fetch_unpack_limit = git_config_int(var, value);
765 return 0;
766 }
767
768 if (strcmp(var, "transfer.unpacklimit") == 0) {
769 transfer_unpack_limit = git_config_int(var, value);
Junio C Hamanoaf7cf262007-01-24 16:47:24 -0800770 return 0;
771 }
772
Nicolas Pitref04833e2009-05-01 20:18:02 -0400773 if (strcmp(var, "repack.usedeltabaseoffset") == 0) {
774 prefer_ofs_delta = git_config_bool(var, value);
775 return 0;
776 }
777
Johannes Schindelinef90d6d2008-05-14 18:46:53 +0100778 return git_default_config(var, value, cb);
Junio C Hamanoaf7cf262007-01-24 16:47:24 -0800779}
780
Junio C Hamano54b9e022007-01-02 11:22:08 -0800781static struct lock_file lock;
782
Shawn O. Pearce50ab5fd2007-09-19 00:49:39 -0400783static void fetch_pack_setup(void)
784{
785 static int did_setup;
786 if (did_setup)
787 return;
Johannes Schindelinef90d6d2008-05-14 18:46:53 +0100788 git_config(fetch_pack_config, NULL);
Shawn O. Pearce50ab5fd2007-09-19 00:49:39 -0400789 if (0 <= transfer_unpack_limit)
790 unpack_limit = transfer_unpack_limit;
791 else if (0 <= fetch_unpack_limit)
792 unpack_limit = fetch_unpack_limit;
793 did_setup = 1;
794}
795
Daniel Barkalow2d4177c2007-09-10 23:03:00 -0400796int cmd_fetch_pack(int argc, const char **argv, const char *prefix)
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700797{
798 int i, ret, nr_heads;
Daniel Barkalowba227852008-02-04 13:26:23 -0500799 struct ref *ref = NULL;
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700800 char *dest = NULL, **heads;
Daniel Barkalowba227852008-02-04 13:26:23 -0500801 int fd[2];
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700802 char *pack_lockfile = NULL;
803 char **pack_lockfile_ptr = NULL;
Daniel Barkalowba227852008-02-04 13:26:23 -0500804 struct child_process *conn;
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700805
806 nr_heads = 0;
807 heads = NULL;
808 for (i = 1; i < argc; i++) {
Daniel Barkalow2d4177c2007-09-10 23:03:00 -0400809 const char *arg = argv[i];
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700810
811 if (*arg == '-') {
Junio C Hamano599065a2007-02-20 01:54:00 -0800812 if (!prefixcmp(arg, "--upload-pack=")) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400813 args.uploadpack = arg + 14;
Uwe Kleine-König27dca072007-01-23 09:20:17 +0100814 continue;
815 }
Junio C Hamano599065a2007-02-20 01:54:00 -0800816 if (!prefixcmp(arg, "--exec=")) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400817 args.uploadpack = arg + 7;
Junio C Hamano8b3d9dc2005-07-14 00:08:37 -0700818 continue;
819 }
Junio C Hamano2247efb2005-12-18 01:55:29 -0800820 if (!strcmp("--quiet", arg) || !strcmp("-q", arg)) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400821 args.quiet = 1;
Junio C Hamano33b83032005-08-12 02:08:29 -0700822 continue;
823 }
Junio C Hamano2247efb2005-12-18 01:55:29 -0800824 if (!strcmp("--keep", arg) || !strcmp("-k", arg)) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400825 args.lock_pack = args.keep_pack;
826 args.keep_pack = 1;
Junio C Hamano9e10fd12007-01-22 22:37:33 -0800827 continue;
828 }
Junio C Hamanob19696c2006-02-20 00:38:39 -0800829 if (!strcmp("--thin", arg)) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400830 args.use_thin_pack = 1;
Junio C Hamanob19696c2006-02-20 00:38:39 -0800831 continue;
832 }
Shawn O. Pearce348e3902008-03-03 22:27:33 -0500833 if (!strcmp("--include-tag", arg)) {
834 args.include_tag = 1;
835 continue;
836 }
Junio C Hamanodfeff662006-03-20 00:21:10 -0800837 if (!strcmp("--all", arg)) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400838 args.fetch_all = 1;
Junio C Hamanodfeff662006-03-20 00:21:10 -0800839 continue;
840 }
Junio C Hamano33b83032005-08-12 02:08:29 -0700841 if (!strcmp("-v", arg)) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400842 args.verbose = 1;
Junio C Hamano33b83032005-08-12 02:08:29 -0700843 continue;
844 }
Junio C Hamano599065a2007-02-20 01:54:00 -0800845 if (!prefixcmp(arg, "--depth=")) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400846 args.depth = strtol(arg + 8, NULL, 0);
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100847 continue;
848 }
Johannes Schindelin83a5ad62007-02-20 03:01:44 +0100849 if (!strcmp("--no-progress", arg)) {
Shawn O. Pearcefa740522007-09-19 00:49:35 -0400850 args.no_progress = 1;
Johannes Schindelin83a5ad62007-02-20 03:01:44 +0100851 continue;
852 }
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700853 if (!strcmp("--stateless-rpc", arg)) {
854 args.stateless_rpc = 1;
855 continue;
856 }
857 if (!strcmp("--lock-pack", arg)) {
858 args.lock_pack = 1;
859 pack_lockfile_ptr = &pack_lockfile;
860 continue;
861 }
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700862 usage(fetch_pack_usage);
863 }
Daniel Barkalow2d4177c2007-09-10 23:03:00 -0400864 dest = (char *)arg;
865 heads = (char **)(argv + i + 1);
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700866 nr_heads = argc - i - 1;
867 break;
868 }
869 if (!dest)
870 usage(fetch_pack_usage);
Daniel Barkalow2d4177c2007-09-10 23:03:00 -0400871
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700872 if (args.stateless_rpc) {
873 conn = NULL;
874 fd[0] = 0;
875 fd[1] = 1;
Daniel Barkalowba227852008-02-04 13:26:23 -0500876 } else {
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700877 conn = git_connect(fd, (char *)dest, args.uploadpack,
878 args.verbose ? CONNECT_VERBOSE : 0);
Daniel Barkalowba227852008-02-04 13:26:23 -0500879 }
Shawn O. Pearce249b2002009-10-30 17:47:42 -0700880
881 get_remote_heads(fd[0], &ref, 0, NULL, 0, NULL);
882
883 ref = fetch_pack(&args, fd, conn, ref, dest,
884 nr_heads, heads, pack_lockfile_ptr);
885 if (pack_lockfile) {
886 printf("lock %s\n", pack_lockfile);
887 fflush(stdout);
888 }
889 close(fd[0]);
890 close(fd[1]);
891 if (finish_connect(conn))
892 ref = NULL;
Daniel Barkalowba227852008-02-04 13:26:23 -0500893 ret = !ref;
Junio C Hamano9e5d2b42005-11-06 00:09:59 -0800894
895 if (!ret && nr_heads) {
896 /* If the heads to pull were given, we should have
897 * consumed all of them by matching the remote.
Heikki Orsila05207a22008-09-09 13:28:30 +0300898 * Otherwise, 'git fetch remote no-such-ref' would
Junio C Hamano9e5d2b42005-11-06 00:09:59 -0800899 * silently succeed without issuing an error.
900 */
901 for (i = 0; i < nr_heads; i++)
902 if (heads[i] && heads[i][0]) {
903 error("no such remote ref %s", heads[i]);
904 ret = 1;
905 }
906 }
Daniel Barkalowba227852008-02-04 13:26:23 -0500907 while (ref) {
908 printf("%s %s\n",
909 sha1_to_hex(ref->old_sha1), ref->name);
910 ref = ref->next;
911 }
Junio C Hamano9e5d2b42005-11-06 00:09:59 -0800912
Daniel Barkalowba227852008-02-04 13:26:23 -0500913 return ret;
914}
915
916struct ref *fetch_pack(struct fetch_pack_args *my_args,
917 int fd[], struct child_process *conn,
918 const struct ref *ref,
919 const char *dest,
920 int nr_heads,
921 char **heads,
922 char **pack_lockfile)
923{
924 struct stat st;
925 struct ref *ref_cpy;
926
927 fetch_pack_setup();
Thomas Rastd551bba2008-12-06 21:50:09 +0100928 if (&args != my_args)
929 memcpy(&args, my_args, sizeof(args));
Daniel Barkalowba227852008-02-04 13:26:23 -0500930 if (args.depth > 0) {
931 if (stat(git_path("shallow"), &st))
932 st.st_mtime = 0;
933 }
934
935 if (heads && nr_heads)
936 nr_heads = remove_duplicates(nr_heads, heads);
937 if (!ref) {
938 packet_flush(fd[1]);
939 die("no matching remote head");
940 }
941 ref_cpy = do_fetch_pack(fd, ref, nr_heads, heads, pack_lockfile);
942
943 if (args.depth > 0) {
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100944 struct cache_time mtime;
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700945 struct strbuf sb = STRBUF_INIT;
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100946 char *shallow = git_path("shallow");
947 int fd;
948
949 mtime.sec = st.st_mtime;
Kjetil Barvikc06ff492009-03-04 18:47:40 +0100950 mtime.nsec = ST_MTIME_NSEC(st);
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100951 if (stat(shallow, &st)) {
952 if (mtime.sec)
953 die("shallow file was removed during fetch");
954 } else if (st.st_mtime != mtime.sec
955#ifdef USE_NSEC
Kjetil Barvik5bcf1092009-03-15 12:38:55 +0100956 || ST_MTIME_NSEC(st) != mtime.nsec
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100957#endif
958 )
959 die("shallow file was changed during fetch");
960
Junio C Hamanoacd3b9e2008-10-17 15:44:39 -0700961 fd = hold_lock_file_for_update(&lock, shallow,
962 LOCK_DIE_ON_ERROR);
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700963 if (!write_shallow_commits(&sb, 0)
964 || write_in_full(fd, sb.buf, sb.len) != sb.len) {
Alex Riesen691f1a22009-04-29 23:22:56 +0200965 unlink_or_warn(shallow);
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100966 rollback_lock_file(&lock);
967 } else {
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100968 commit_lock_file(&lock);
969 }
Shawn O. Pearceedace6f2009-10-30 17:47:23 -0700970 strbuf_release(&sb);
Johannes Schindelin016e6cc2006-10-30 20:09:29 +0100971 }
972
Johan Herland48ec3e52008-06-15 16:04:20 +0200973 reprepare_packed_git();
Daniel Barkalowba227852008-02-04 13:26:23 -0500974 return ref_cpy;
Linus Torvaldsdef88e92005-07-04 13:26:53 -0700975}