1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
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
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116 class WalkFetchConnection extends BaseFetchConnection {
117
118 final Repository local;
119
120
121 final ObjectChecker objCheck;
122
123
124
125
126
127
128
129
130 private final List<WalkRemoteObjectDatabase> remotes;
131
132
133 private int lastRemoteIdx;
134
135 private final RevWalk revWalk;
136
137 private final TreeWalk treeWalk;
138
139
140 private final RevFlag COMPLETE;
141
142
143 private final RevFlag IN_WORK_QUEUE;
144
145
146 private final RevFlag LOCALLY_SEEN;
147
148
149 private final DateRevQueue localCommitQueue;
150
151
152 private LinkedList<ObjectId> workQueue;
153
154
155 private final LinkedList<WalkRemoteObjectDatabase> noPacksYet;
156
157
158 private final LinkedList<WalkRemoteObjectDatabase> noAlternatesYet;
159
160
161 private final LinkedList<RemotePack> unfetchedPacks;
162
163
164
165
166
167
168
169
170 private final Set<String> packsConsidered;
171
172 private final MutableObjectId idBuffer = new MutableObjectId();
173
174
175
176
177
178
179
180
181 private final HashMap<ObjectId, List<Throwable>> fetchErrors;
182
183 String lockMessage;
184
185 final List<PackLock> packLocks;
186
187
188 final ObjectInserter inserter;
189
190
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");
219 IN_WORK_QUEUE = revWalk.newFlag("IN_WORK_QUEUE");
220 LOCALLY_SEEN = revWalk.newFlag("LOCALLY_SEEN");
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
335
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
413
414
415
416 if (downloadPackedObject(pm, id))
417 return;
418
419
420
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;
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
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
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
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
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
508
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
517
518
519
520
521 recordError(id, err);
522 packItr.remove();
523 continue;
524 }
525
526 if (monitor.isCancelled()) {
527
528
529
530
531 return false;
532 }
533
534 if (!pack.index.hasObject(id)) {
535
536
537 continue;
538 }
539
540
541
542
543
544 try {
545 pack.downloadPack(monitor);
546 } catch (IOException err) {
547
548
549
550
551
552 recordError(id, err);
553 continue;
554 } finally {
555
556
557
558
559
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
572
573
574
575 recordError(id, new FileNotFoundException(MessageFormat.format(
576 JGitText.get().objectNotFoundIn, id.name(), pack.packName)));
577 continue;
578 }
579
580
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
613
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
629
630
631
632
633
634
635
636
637
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
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";
818
819 String tn = idxName;
820 if (tn.startsWith("pack-"))
821 tn = tn.substring(5);
822 if (tn.endsWith(".idx"))
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");
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");
837 else if (tmpIdx.isFile()) {
838 try {
839 index = PackIndex.open(tmpIdx);
840 return;
841 } catch (FileNotFoundException err) {
842
843 }
844 }
845
846 final WalkRemoteObjectDatabase.FileStream s;
847 s = connection.open("pack/" + idxName);
848 pm.beginTask("Get " + idxName.substring(0, 12) + "..idx",
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;
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 }