View Javadoc
1   /*
2    * Copyright (C) 2008, Robin Rosenberg <robin.rosenberg@dewire.com>
3    * Copyright (C) 2008, Shawn O. Pearce <spearce@spearce.org>
4    * and other copyright owners as documented in the project's IP log.
5    *
6    * This program and the accompanying materials are made available
7    * under the terms of the Eclipse Distribution License v1.0 which
8    * accompanies this distribution, is reproduced below, and is
9    * available at http://www.eclipse.org/org/documents/edl-v10.php
10   *
11   * All rights reserved.
12   *
13   * Redistribution and use in source and binary forms, with or
14   * without modification, are permitted provided that the following
15   * conditions are met:
16   *
17   * - Redistributions of source code must retain the above copyright
18   *   notice, this list of conditions and the following disclaimer.
19   *
20   * - Redistributions in binary form must reproduce the above
21   *   copyright notice, this list of conditions and the following
22   *   disclaimer in the documentation and/or other materials provided
23   *   with the distribution.
24   *
25   * - Neither the name of the Eclipse Foundation, Inc. nor the
26   *   names of its contributors may be used to endorse or promote
27   *   products derived from this software without specific prior
28   *   written permission.
29   *
30   * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
31   * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
32   * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
33   * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
34   * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
35   * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
36   * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
37   * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
38   * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
39   * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
40   * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
41   * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
42   * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
43   */
44  
45  package org.eclipse.jgit.transport;
46  
47  import static org.eclipse.jgit.lib.RefDatabase.ALL;
48  
49  import java.io.File;
50  import java.io.FileNotFoundException;
51  import java.io.FileOutputStream;
52  import java.io.IOException;
53  import java.text.MessageFormat;
54  import java.util.ArrayList;
55  import java.util.Collection;
56  import java.util.HashMap;
57  import java.util.HashSet;
58  import java.util.Iterator;
59  import java.util.LinkedList;
60  import java.util.List;
61  import java.util.Map;
62  import java.util.Set;
63  
64  import org.eclipse.jgit.errors.CompoundException;
65  import org.eclipse.jgit.errors.CorruptObjectException;
66  import org.eclipse.jgit.errors.MissingObjectException;
67  import org.eclipse.jgit.errors.TransportException;
68  import org.eclipse.jgit.internal.JGitText;
69  import org.eclipse.jgit.internal.storage.file.ObjectDirectory;
70  import org.eclipse.jgit.internal.storage.file.PackIndex;
71  import org.eclipse.jgit.internal.storage.file.PackLock;
72  import org.eclipse.jgit.internal.storage.file.UnpackedObject;
73  import org.eclipse.jgit.lib.AnyObjectId;
74  import org.eclipse.jgit.lib.Constants;
75  import org.eclipse.jgit.lib.FileMode;
76  import org.eclipse.jgit.lib.MutableObjectId;
77  import org.eclipse.jgit.lib.ObjectChecker;
78  import org.eclipse.jgit.lib.ObjectId;
79  import org.eclipse.jgit.lib.ObjectInserter;
80  import org.eclipse.jgit.lib.ObjectLoader;
81  import org.eclipse.jgit.lib.ObjectReader;
82  import org.eclipse.jgit.lib.ProgressMonitor;
83  import org.eclipse.jgit.lib.Ref;
84  import org.eclipse.jgit.lib.Repository;
85  import org.eclipse.jgit.revwalk.DateRevQueue;
86  import org.eclipse.jgit.revwalk.RevCommit;
87  import org.eclipse.jgit.revwalk.RevFlag;
88  import org.eclipse.jgit.revwalk.RevObject;
89  import org.eclipse.jgit.revwalk.RevTag;
90  import org.eclipse.jgit.revwalk.RevTree;
91  import org.eclipse.jgit.revwalk.RevWalk;
92  import org.eclipse.jgit.treewalk.TreeWalk;
93  import org.eclipse.jgit.util.FileUtils;
94  
95  /**
96   * Generic fetch support for dumb transport protocols.
97   * <p>
98   * Since there are no Git-specific smarts on the remote side of the connection
99   * the client side must determine which objects it needs to copy in order to
100  * completely fetch the requested refs and their history. The generic walk
101  * support in this class parses each individual object (once it has been copied
102  * to the local repository) and examines the list of objects that must also be
103  * copied to create a complete history. Objects which are already available
104  * locally are retained (and not copied), saving bandwidth for incremental
105  * fetches. Pack files are copied from the remote repository only as a last
106  * resort, as the entire pack must be copied locally in order to access any
107  * single object.
108  * <p>
109  * This fetch connection does not actually perform the object data transfer.
110  * Instead it delegates the transfer to a {@link WalkRemoteObjectDatabase},
111  * which knows how to read individual files from the remote repository and
112  * supply the data as a standard Java InputStream.
113  *
114  * @see WalkRemoteObjectDatabase
115  */
116 class WalkFetchConnection extends BaseFetchConnection {
117 	/** The repository this transport fetches into, or pushes out of. */
118 	final Repository local;
119 
120 	/** If not null the validator for received objects. */
121 	final ObjectChecker objCheck;
122 
123 	/**
124 	 * List of all remote repositories we may need to get objects out of.
125 	 * <p>
126 	 * The first repository in the list is the one we were asked to fetch from;
127 	 * the remaining repositories point to the alternate locations we can fetch
128 	 * objects through.
129 	 */
130 	private final List<WalkRemoteObjectDatabase> remotes;
131 
132 	/** Most recently used item in {@link #remotes}. */
133 	private int lastRemoteIdx;
134 
135 	private final RevWalk revWalk;
136 
137 	private final TreeWalk treeWalk;
138 
139 	/** Objects whose direct dependents we know we have (or will have). */
140 	private final RevFlag COMPLETE;
141 
142 	/** Objects that have already entered {@link #workQueue}. */
143 	private final RevFlag IN_WORK_QUEUE;
144 
145 	/** Commits that have already entered {@link #localCommitQueue}. */
146 	private final RevFlag LOCALLY_SEEN;
147 
148 	/** Commits already reachable from all local refs. */
149 	private final DateRevQueue localCommitQueue;
150 
151 	/** Objects we need to copy from the remote repository. */
152 	private LinkedList<ObjectId> workQueue;
153 
154 	/** Databases we have not yet obtained the list of packs from. */
155 	private final LinkedList<WalkRemoteObjectDatabase> noPacksYet;
156 
157 	/** Databases we have not yet obtained the alternates from. */
158 	private final LinkedList<WalkRemoteObjectDatabase> noAlternatesYet;
159 
160 	/** Packs we have discovered, but have not yet fetched locally. */
161 	private final LinkedList<RemotePack> unfetchedPacks;
162 
163 	/**
164 	 * Packs whose indexes we have looked at in {@link #unfetchedPacks}.
165 	 * <p>
166 	 * We try to avoid getting duplicate copies of the same pack through
167 	 * multiple alternates by only looking at packs whose names are not yet in
168 	 * this collection.
169 	 */
170 	private final Set<String> packsConsidered;
171 
172 	private final MutableObjectId idBuffer = new MutableObjectId();
173 
174 	/**
175 	 * Errors received while trying to obtain an object.
176 	 * <p>
177 	 * If the fetch winds up failing because we cannot locate a specific object
178 	 * then we need to report all errors related to that object back to the
179 	 * caller as there may be cascading failures.
180 	 */
181 	private final HashMap<ObjectId, List<Throwable>> fetchErrors;
182 
183 	String lockMessage;
184 
185 	final List<PackLock> packLocks;
186 
187 	/** Inserter to write objects onto {@link #local}. */
188 	final ObjectInserter inserter;
189 
190 	/** Inserter to read objects from {@link #local}. */
191 	private final ObjectReader reader;
192 
193 	WalkFetchConnection(final WalkTransport t, final WalkRemoteObjectDatabase w) {
194 		Transport wt = (Transport)t;
195 		local = wt.local;
196 		objCheck = wt.getObjectChecker();
197 		inserter = local.newObjectInserter();
198 		reader = inserter.newReader();
199 
200 		remotes = new ArrayList<>();
201 		remotes.add(w);
202 
203 		unfetchedPacks = new LinkedList<>();
204 		packsConsidered = new HashSet<>();
205 
206 		noPacksYet = new LinkedList<>();
207 		noPacksYet.add(w);
208 
209 		noAlternatesYet = new LinkedList<>();
210 		noAlternatesYet.add(w);
211 
212 		fetchErrors = new HashMap<>();
213 		packLocks = new ArrayList<>(4);
214 
215 		revWalk = new RevWalk(reader);
216 		revWalk.setRetainBody(false);
217 		treeWalk = new TreeWalk(reader);
218 		COMPLETE = revWalk.newFlag("COMPLETE"); //$NON-NLS-1$
219 		IN_WORK_QUEUE = revWalk.newFlag("IN_WORK_QUEUE"); //$NON-NLS-1$
220 		LOCALLY_SEEN = revWalk.newFlag("LOCALLY_SEEN"); //$NON-NLS-1$
221 
222 		localCommitQueue = new DateRevQueue();
223 		workQueue = new LinkedList<>();
224 	}
225 
226 	@Override
227 	public boolean didFetchTestConnectivity() {
228 		return true;
229 	}
230 
231 	@Override
232 	protected void doFetch(final ProgressMonitor monitor,
233 			final Collection<Ref> want, final Set<ObjectId> have)
234 			throws TransportException {
235 		markLocalRefsComplete(have);
236 		queueWants(want);
237 
238 		while (!monitor.isCancelled() && !workQueue.isEmpty()) {
239 			final ObjectId id = workQueue.removeFirst();
240 			if (!(id instanceof RevObject) || !((RevObject) id).has(COMPLETE))
241 				downloadObject(monitor, id);
242 			process(id);
243 		}
244 
245 		try {
246 			inserter.flush();
247 		} catch (IOException e) {
248 			throw new TransportException(e.getMessage(), e);
249 		}
250 	}
251 
252 	@Override
253 	public Collection<PackLock> getPackLocks() {
254 		return packLocks;
255 	}
256 
257 	@Override
258 	public void setPackLockMessage(final String message) {
259 		lockMessage = message;
260 	}
261 
262 	@Override
263 	public void close() {
264 		inserter.close();
265 		reader.close();
266 		for (final RemotePack p : unfetchedPacks) {
267 			if (p.tmpIdx != null)
268 				p.tmpIdx.delete();
269 		}
270 		for (final WalkRemoteObjectDatabase r : remotes)
271 			r.close();
272 	}
273 
274 	private void queueWants(final Collection<Ref> want)
275 			throws TransportException {
276 		final HashSet<ObjectId> inWorkQueue = new HashSet<>();
277 		for (final Ref r : want) {
278 			final ObjectId id = r.getObjectId();
279 			if (id == null) {
280 				throw new NullPointerException(MessageFormat.format(
281 						JGitText.get().transportProvidedRefWithNoObjectId, r.getName()));
282 			}
283 			try {
284 				final RevObject obj = revWalk.parseAny(id);
285 				if (obj.has(COMPLETE))
286 					continue;
287 				if (inWorkQueue.add(id)) {
288 					obj.add(IN_WORK_QUEUE);
289 					workQueue.add(obj);
290 				}
291 			} catch (MissingObjectException e) {
292 				if (inWorkQueue.add(id))
293 					workQueue.add(id);
294 			} catch (IOException e) {
295 				throw new TransportException(MessageFormat.format(JGitText.get().cannotRead, id.name()), e);
296 			}
297 		}
298 	}
299 
300 	private void process(final ObjectId id) throws TransportException {
301 		final RevObject obj;
302 		try {
303 			if (id instanceof RevObject) {
304 				obj = (RevObject) id;
305 				if (obj.has(COMPLETE))
306 					return;
307 				revWalk.parseHeaders(obj);
308 			} else {
309 				obj = revWalk.parseAny(id);
310 				if (obj.has(COMPLETE))
311 					return;
312 			}
313 		} catch (IOException e) {
314 			throw new TransportException(MessageFormat.format(JGitText.get().cannotRead, id.name()), e);
315 		}
316 
317 		switch (obj.getType()) {
318 		case Constants.OBJ_BLOB:
319 			processBlob(obj);
320 			break;
321 		case Constants.OBJ_TREE:
322 			processTree(obj);
323 			break;
324 		case Constants.OBJ_COMMIT:
325 			processCommit(obj);
326 			break;
327 		case Constants.OBJ_TAG:
328 			processTag(obj);
329 			break;
330 		default:
331 			throw new TransportException(MessageFormat.format(JGitText.get().unknownObjectType, id.name()));
332 		}
333 
334 		// If we had any prior errors fetching this object they are
335 		// now resolved, as the object was parsed successfully.
336 		//
337 		fetchErrors.remove(id);
338 	}
339 
340 	private void processBlob(final RevObject obj) throws TransportException {
341 		try {
342 			if (reader.has(obj, Constants.OBJ_BLOB))
343 				obj.add(COMPLETE);
344 			else
345 				throw new TransportException(MessageFormat.format(JGitText
346 						.get().cannotReadBlob, obj.name()),
347 						new MissingObjectException(obj, Constants.TYPE_BLOB));
348 		} catch (IOException error) {
349 			throw new TransportException(MessageFormat.format(
350 					JGitText.get().cannotReadBlob, obj.name()), error);
351 		}
352 	}
353 
354 	private void processTree(final RevObject obj) throws TransportException {
355 		try {
356 			treeWalk.reset(obj);
357 			while (treeWalk.next()) {
358 				final FileMode mode = treeWalk.getFileMode(0);
359 				final int sType = mode.getObjectType();
360 
361 				switch (sType) {
362 				case Constants.OBJ_BLOB:
363 				case Constants.OBJ_TREE:
364 					treeWalk.getObjectId(idBuffer, 0);
365 					needs(revWalk.lookupAny(idBuffer, sType));
366 					continue;
367 
368 				default:
369 					if (FileMode.GITLINK.equals(mode))
370 						continue;
371 					treeWalk.getObjectId(idBuffer, 0);
372 					throw new CorruptObjectException(MessageFormat.format(JGitText.get().invalidModeFor
373 							, mode, idBuffer.name(), treeWalk.getPathString(), obj.getId().name()));
374 				}
375 			}
376 		} catch (IOException ioe) {
377 			throw new TransportException(MessageFormat.format(JGitText.get().cannotReadTree, obj.name()), ioe);
378 		}
379 		obj.add(COMPLETE);
380 	}
381 
382 	private void processCommit(final RevObject obj) throws TransportException {
383 		final RevCommit commit = (RevCommit) obj;
384 		markLocalCommitsComplete(commit.getCommitTime());
385 		needs(commit.getTree());
386 		for (final RevCommit p : commit.getParents())
387 			needs(p);
388 		obj.add(COMPLETE);
389 	}
390 
391 	private void processTag(final RevObject obj) {
392 		final RevTag tag = (RevTag) obj;
393 		needs(tag.getObject());
394 		obj.add(COMPLETE);
395 	}
396 
397 	private void needs(final RevObject obj) {
398 		if (obj.has(COMPLETE))
399 			return;
400 		if (!obj.has(IN_WORK_QUEUE)) {
401 			obj.add(IN_WORK_QUEUE);
402 			workQueue.add(obj);
403 		}
404 	}
405 
406 	private void downloadObject(final ProgressMonitor pm, final AnyObjectId id)
407 			throws TransportException {
408 		if (alreadyHave(id))
409 			return;
410 
411 		for (;;) {
412 			// Try a pack file we know about, but don't have yet. Odds are
413 			// that if it has this object, it has others related to it so
414 			// getting the pack is a good bet.
415 			//
416 			if (downloadPackedObject(pm, id))
417 				return;
418 
419 			// Search for a loose object over all alternates, starting
420 			// from the one we last successfully located an object through.
421 			//
422 			final String idStr = id.name();
423 			final String subdir = idStr.substring(0, 2);
424 			final String file = idStr.substring(2);
425 			final String looseName = subdir + "/" + file; //$NON-NLS-1$
426 
427 			for (int i = lastRemoteIdx; i < remotes.size(); i++) {
428 				if (downloadLooseObject(id, looseName, remotes.get(i))) {
429 					lastRemoteIdx = i;
430 					return;
431 				}
432 			}
433 			for (int i = 0; i < lastRemoteIdx; i++) {
434 				if (downloadLooseObject(id, looseName, remotes.get(i))) {
435 					lastRemoteIdx = i;
436 					return;
437 				}
438 			}
439 
440 			// Try to obtain more pack information and search those.
441 			//
442 			while (!noPacksYet.isEmpty()) {
443 				final WalkRemoteObjectDatabase wrr = noPacksYet.removeFirst();
444 				final Collection<String> packNameList;
445 				try {
446 					pm.beginTask(JGitText.get().listingPacks,
447 							ProgressMonitor.UNKNOWN);
448 					packNameList = wrr.getPackNames();
449 				} catch (IOException e) {
450 					// Try another repository.
451 					//
452 					recordError(id, e);
453 					continue;
454 				} finally {
455 					pm.endTask();
456 				}
457 
458 				if (packNameList == null || packNameList.isEmpty())
459 					continue;
460 				for (final String packName : packNameList) {
461 					if (packsConsidered.add(packName))
462 						unfetchedPacks.add(new RemotePack(wrr, packName));
463 				}
464 				if (downloadPackedObject(pm, id))
465 					return;
466 			}
467 
468 			// Try to expand the first alternate we haven't expanded yet.
469 			//
470 			Collection<WalkRemoteObjectDatabase> al = expandOneAlternate(id, pm);
471 			if (al != null && !al.isEmpty()) {
472 				for (final WalkRemoteObjectDatabase alt : al) {
473 					remotes.add(alt);
474 					noPacksYet.add(alt);
475 					noAlternatesYet.add(alt);
476 				}
477 				continue;
478 			}
479 
480 			// We could not obtain the object. There may be reasons why.
481 			//
482 			List<Throwable> failures = fetchErrors.get(id);
483 			final TransportException te;
484 
485 			te = new TransportException(MessageFormat.format(JGitText.get().cannotGet, id.name()));
486 			if (failures != null && !failures.isEmpty()) {
487 				if (failures.size() == 1)
488 					te.initCause(failures.get(0));
489 				else
490 					te.initCause(new CompoundException(failures));
491 			}
492 			throw te;
493 		}
494 	}
495 
496 	private boolean alreadyHave(final AnyObjectId id) throws TransportException {
497 		try {
498 			return reader.has(id);
499 		} catch (IOException error) {
500 			throw new TransportException(MessageFormat.format(
501 					JGitText.get().cannotReadObject, id.name()), error);
502 		}
503 	}
504 
505 	private boolean downloadPackedObject(final ProgressMonitor monitor,
506 			final AnyObjectId id) throws TransportException {
507 		// Search for the object in a remote pack whose index we have,
508 		// but whose pack we do not yet have.
509 		//
510 		final Iterator<RemotePack> packItr = unfetchedPacks.iterator();
511 		while (packItr.hasNext() && !monitor.isCancelled()) {
512 			final RemotePack pack = packItr.next();
513 			try {
514 				pack.openIndex(monitor);
515 			} catch (IOException err) {
516 				// If the index won't open its either not found or
517 				// its a format we don't recognize. In either case
518 				// we may still be able to obtain the object from
519 				// another source, so don't consider it a failure.
520 				//
521 				recordError(id, err);
522 				packItr.remove();
523 				continue;
524 			}
525 
526 			if (monitor.isCancelled()) {
527 				// If we were cancelled while the index was opening
528 				// the open may have aborted. We can't search an
529 				// unopen index.
530 				//
531 				return false;
532 			}
533 
534 			if (!pack.index.hasObject(id)) {
535 				// Not in this pack? Try another.
536 				//
537 				continue;
538 			}
539 
540 			// It should be in the associated pack. Download that
541 			// and attach it to the local repository so we can use
542 			// all of the contained objects.
543 			//
544 			try {
545 				pack.downloadPack(monitor);
546 			} catch (IOException err) {
547 				// If the pack failed to download, index correctly,
548 				// or open in the local repository we may still be
549 				// able to obtain this object from another pack or
550 				// an alternate.
551 				//
552 				recordError(id, err);
553 				continue;
554 			} finally {
555 				// If the pack was good its in the local repository
556 				// and Repository.hasObject(id) will succeed in the
557 				// future, so we do not need this data anymore. If
558 				// it failed the index and pack are unusable and we
559 				// shouldn't consult them again.
560 				//
561 				try {
562 					if (pack.tmpIdx != null)
563 						FileUtils.delete(pack.tmpIdx);
564 				} catch (IOException e) {
565 					throw new TransportException(e.getMessage(), e);
566 				}
567 				packItr.remove();
568 			}
569 
570 			if (!alreadyHave(id)) {
571 				// What the hell? This pack claimed to have
572 				// the object, but after indexing we didn't
573 				// actually find it in the pack.
574 				//
575 				recordError(id, new FileNotFoundException(MessageFormat.format(
576 						JGitText.get().objectNotFoundIn, id.name(), pack.packName)));
577 				continue;
578 			}
579 
580 			// Complete any other objects that we can.
581 			//
582 			final Iterator<ObjectId> pending = swapFetchQueue();
583 			while (pending.hasNext()) {
584 				final ObjectId p = pending.next();
585 				if (pack.index.hasObject(p)) {
586 					pending.remove();
587 					process(p);
588 				} else {
589 					workQueue.add(p);
590 				}
591 			}
592 			return true;
593 
594 		}
595 		return false;
596 	}
597 
598 	private Iterator<ObjectId> swapFetchQueue() {
599 		final Iterator<ObjectId> r = workQueue.iterator();
600 		workQueue = new LinkedList<>();
601 		return r;
602 	}
603 
604 	private boolean downloadLooseObject(final AnyObjectId id,
605 			final String looseName, final WalkRemoteObjectDatabase remote)
606 			throws TransportException {
607 		try {
608 			final byte[] compressed = remote.open(looseName).toArray();
609 			verifyAndInsertLooseObject(id, compressed);
610 			return true;
611 		} catch (FileNotFoundException e) {
612 			// Not available in a loose format from this alternate?
613 			// Try another strategy to get the object.
614 			//
615 			recordError(id, e);
616 			return false;
617 		} catch (IOException e) {
618 			throw new TransportException(MessageFormat.format(JGitText.get().cannotDownload, id.name()), e);
619 		}
620 	}
621 
622 	private void verifyAndInsertLooseObject(final AnyObjectId id,
623 			final byte[] compressed) throws IOException {
624 		final ObjectLoader uol;
625 		try {
626 			uol = UnpackedObject.parse(compressed, id);
627 		} catch (CorruptObjectException parsingError) {
628 			// Some HTTP servers send back a "200 OK" status with an HTML
629 			// page that explains the requested file could not be found.
630 			// These servers are most certainly misconfigured, but many
631 			// of them exist in the world, and many of those are hosting
632 			// Git repositories.
633 			//
634 			// Since an HTML page is unlikely to hash to one of our loose
635 			// objects we treat this condition as a FileNotFoundException
636 			// and attempt to recover by getting the object from another
637 			// source.
638 			//
639 			final FileNotFoundException e;
640 			e = new FileNotFoundException(id.name());
641 			e.initCause(parsingError);
642 			throw e;
643 		}
644 
645 		final int type = uol.getType();
646 		final byte[] raw = uol.getCachedBytes();
647 		if (objCheck != null) {
648 			try {
649 				objCheck.check(id, type, raw);
650 			} catch (CorruptObjectException e) {
651 				throw new TransportException(MessageFormat.format(
652 						JGitText.get().transportExceptionInvalid,
653 						Constants.typeString(type), id.name(), e.getMessage()));
654 			}
655 		}
656 
657 		ObjectId act = inserter.insert(type, raw);
658 		if (!AnyObjectId.equals(id, act)) {
659 			throw new TransportException(MessageFormat.format(
660 					JGitText.get().incorrectHashFor, id.name(), act.name(),
661 					Constants.typeString(type),
662 					Integer.valueOf(compressed.length)));
663 		}
664 	}
665 
666 	private Collection<WalkRemoteObjectDatabase> expandOneAlternate(
667 			final AnyObjectId id, final ProgressMonitor pm) {
668 		while (!noAlternatesYet.isEmpty()) {
669 			final WalkRemoteObjectDatabase wrr = noAlternatesYet.removeFirst();
670 			try {
671 				pm.beginTask(JGitText.get().listingAlternates, ProgressMonitor.UNKNOWN);
672 				Collection<WalkRemoteObjectDatabase> altList = wrr
673 						.getAlternates();
674 				if (altList != null && !altList.isEmpty())
675 					return altList;
676 			} catch (IOException e) {
677 				// Try another repository.
678 				//
679 				recordError(id, e);
680 			} finally {
681 				pm.endTask();
682 			}
683 		}
684 		return null;
685 	}
686 
687 	private void markLocalRefsComplete(final Set<ObjectId> have) throws TransportException {
688 		Map<String, Ref> refs;
689 		try {
690 			refs = local.getRefDatabase().getRefs(ALL);
691 		} catch (IOException e) {
692 			throw new TransportException(e.getMessage(), e);
693 		}
694 		for (final Ref r : refs.values()) {
695 			try {
696 				markLocalObjComplete(revWalk.parseAny(r.getObjectId()));
697 			} catch (IOException readError) {
698 				throw new TransportException(MessageFormat.format(JGitText.get().localRefIsMissingObjects, r.getName()), readError);
699 			}
700 		}
701 		for (final ObjectId id : have) {
702 			try {
703 				markLocalObjComplete(revWalk.parseAny(id));
704 			} catch (IOException readError) {
705 				throw new TransportException(MessageFormat.format(JGitText.get().transportExceptionMissingAssumed, id.name()), readError);
706 			}
707 		}
708 	}
709 
710 	private void markLocalObjComplete(RevObject obj) throws IOException {
711 		while (obj.getType() == Constants.OBJ_TAG) {
712 			obj.add(COMPLETE);
713 			obj = ((RevTag) obj).getObject();
714 			revWalk.parseHeaders(obj);
715 		}
716 
717 		switch (obj.getType()) {
718 		case Constants.OBJ_BLOB:
719 			obj.add(COMPLETE);
720 			break;
721 		case Constants.OBJ_COMMIT:
722 			pushLocalCommit((RevCommit) obj);
723 			break;
724 		case Constants.OBJ_TREE:
725 			markTreeComplete((RevTree) obj);
726 			break;
727 		}
728 	}
729 
730 	private void markLocalCommitsComplete(final int until)
731 			throws TransportException {
732 		try {
733 			for (;;) {
734 				final RevCommit c = localCommitQueue.peek();
735 				if (c == null || c.getCommitTime() < until)
736 					return;
737 				localCommitQueue.next();
738 
739 				markTreeComplete(c.getTree());
740 				for (final RevCommit p : c.getParents())
741 					pushLocalCommit(p);
742 			}
743 		} catch (IOException err) {
744 			throw new TransportException(JGitText.get().localObjectsIncomplete, err);
745 		}
746 	}
747 
748 	private void pushLocalCommit(final RevCommit p)
749 			throws MissingObjectException, IOException {
750 		if (p.has(LOCALLY_SEEN))
751 			return;
752 		revWalk.parseHeaders(p);
753 		p.add(LOCALLY_SEEN);
754 		p.add(COMPLETE);
755 		p.carry(COMPLETE);
756 		localCommitQueue.add(p);
757 	}
758 
759 	private void markTreeComplete(final RevTree tree) throws IOException {
760 		if (tree.has(COMPLETE))
761 			return;
762 		tree.add(COMPLETE);
763 		treeWalk.reset(tree);
764 		while (treeWalk.next()) {
765 			final FileMode mode = treeWalk.getFileMode(0);
766 			final int sType = mode.getObjectType();
767 
768 			switch (sType) {
769 			case Constants.OBJ_BLOB:
770 				treeWalk.getObjectId(idBuffer, 0);
771 				revWalk.lookupAny(idBuffer, sType).add(COMPLETE);
772 				continue;
773 
774 			case Constants.OBJ_TREE: {
775 				treeWalk.getObjectId(idBuffer, 0);
776 				final RevObject o = revWalk.lookupAny(idBuffer, sType);
777 				if (!o.has(COMPLETE)) {
778 					o.add(COMPLETE);
779 					treeWalk.enterSubtree();
780 				}
781 				continue;
782 			}
783 			default:
784 				if (FileMode.GITLINK.equals(mode))
785 					continue;
786 				treeWalk.getObjectId(idBuffer, 0);
787 				throw new CorruptObjectException(MessageFormat.format(JGitText.get().corruptObjectInvalidMode3
788 						, mode, idBuffer.name(), treeWalk.getPathString(), tree.name()));
789 			}
790 		}
791 	}
792 
793 	private void recordError(final AnyObjectId id, final Throwable what) {
794 		final ObjectId objId = id.copy();
795 		List<Throwable> errors = fetchErrors.get(objId);
796 		if (errors == null) {
797 			errors = new ArrayList<>(2);
798 			fetchErrors.put(objId, errors);
799 		}
800 		errors.add(what);
801 	}
802 
803 	private class RemotePack {
804 		final WalkRemoteObjectDatabase connection;
805 
806 		final String packName;
807 
808 		final String idxName;
809 
810 		File tmpIdx;
811 
812 		PackIndex index;
813 
814 		RemotePack(final WalkRemoteObjectDatabase c, final String pn) {
815 			connection = c;
816 			packName = pn;
817 			idxName = packName.substring(0, packName.length() - 5) + ".idx"; //$NON-NLS-1$
818 
819 			String tn = idxName;
820 			if (tn.startsWith("pack-")) //$NON-NLS-1$
821 				tn = tn.substring(5);
822 			if (tn.endsWith(".idx")) //$NON-NLS-1$
823 				tn = tn.substring(0, tn.length() - 4);
824 
825 			if (local.getObjectDatabase() instanceof ObjectDirectory) {
826 				tmpIdx = new File(((ObjectDirectory) local.getObjectDatabase())
827 								.getDirectory(),
828 						"walk-" + tn + ".walkidx"); //$NON-NLS-1$ //$NON-NLS-2$
829 			}
830 		}
831 
832 		void openIndex(final ProgressMonitor pm) throws IOException {
833 			if (index != null)
834 				return;
835 			if (tmpIdx == null)
836 				tmpIdx = File.createTempFile("jgit-walk-", ".idx"); //$NON-NLS-1$ //$NON-NLS-2$
837 			else if (tmpIdx.isFile()) {
838 				try {
839 					index = PackIndex.open(tmpIdx);
840 					return;
841 				} catch (FileNotFoundException err) {
842 					// Fall through and get the file.
843 				}
844 			}
845 
846 			final WalkRemoteObjectDatabase.FileStream s;
847 			s = connection.open("pack/" + idxName); //$NON-NLS-1$
848 			pm.beginTask("Get " + idxName.substring(0, 12) + "..idx", //$NON-NLS-1$ //$NON-NLS-2$
849 					s.length < 0 ? ProgressMonitor.UNKNOWN
850 							: (int) (s.length / 1024));
851 			try {
852 				final FileOutputStream fos = new FileOutputStream(tmpIdx);
853 				try {
854 					final byte[] buf = new byte[2048];
855 					int cnt;
856 					while (!pm.isCancelled() && (cnt = s.in.read(buf)) >= 0) {
857 						fos.write(buf, 0, cnt);
858 						pm.update(cnt / 1024);
859 					}
860 				} finally {
861 					fos.close();
862 				}
863 			} catch (IOException err) {
864 				FileUtils.delete(tmpIdx);
865 				throw err;
866 			} finally {
867 				s.in.close();
868 			}
869 			pm.endTask();
870 
871 			if (pm.isCancelled()) {
872 				FileUtils.delete(tmpIdx);
873 				return;
874 			}
875 
876 			try {
877 				index = PackIndex.open(tmpIdx);
878 			} catch (IOException e) {
879 				FileUtils.delete(tmpIdx);
880 				throw e;
881 			}
882 		}
883 
884 		void downloadPack(final ProgressMonitor monitor) throws IOException {
885 			String name = "pack/" + packName; //$NON-NLS-1$
886 			WalkRemoteObjectDatabase.FileStream s = connection.open(name);
887 			try {
888 				PackParser parser = inserter.newPackParser(s.in);
889 				parser.setAllowThin(false);
890 				parser.setObjectChecker(objCheck);
891 				parser.setLockMessage(lockMessage);
892 				PackLock lock = parser.parse(monitor);
893 				if (lock != null)
894 					packLocks.add(lock);
895 			} finally {
896 				s.in.close();
897 			}
898 		}
899 	}
900 }