summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorChristian Halstrick <christian.halstrick@sap.com>2013-11-11 10:57:46 +0100
committerChristian Halstrick <christian.halstrick@sap.com>2013-11-13 08:59:43 +0100
commit8da939221c2e244fe0ba979fa8419a90bf618315 (patch)
tree6e311d6d0f60adf456c0bf340419850820bf3b81
parentf1c086f692cb1e53a075af16ae662f150443682e (diff)
downloadjgit-8da939221c2e244fe0ba979fa8419a90bf618315.tar.gz
jgit-8da939221c2e244fe0ba979fa8419a90bf618315.zip
Don't delete .idx file if .pack file can't be deleted
If during an garbage collection old packfiles are deleted it could happen that on certain platforms the index file can be deleted but the packfile can't be deleted (because someone locked the file). This led to repositories with packfiles without corresponding index files. Those zombie-packfiles potentially consume a lot of space on disk and it is never tried to delete them again. Try to avoid this situation by deleting packfiles first and don't try to delete the other files if we can't delete the packfile. This gives us the chance to delete the packfile during next GC. This commit only improves the situation - there is still the chance for orphan files during packfile deletion. We don't have an atomic delete of multiple files . Change-Id: I0a19ae630186f07d0cc7fe9df246fa1cedeca8f6
-rw-r--r--org.eclipse.jgit.test/tst/org/eclipse/jgit/internal/storage/file/GCTest.java36
-rw-r--r--org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/GC.java57
2 files changed, 75 insertions, 18 deletions
diff --git a/org.eclipse.jgit.test/tst/org/eclipse/jgit/internal/storage/file/GCTest.java b/org.eclipse.jgit.test/tst/org/eclipse/jgit/internal/storage/file/GCTest.java
index 9d2a03b097..35455f48a9 100644
--- a/org.eclipse.jgit.test/tst/org/eclipse/jgit/internal/storage/file/GCTest.java
+++ b/org.eclipse.jgit.test/tst/org/eclipse/jgit/internal/storage/file/GCTest.java
@@ -49,6 +49,7 @@ import static org.junit.Assert.assertSame;
import static org.junit.Assert.assertTrue;
import java.io.File;
+import java.io.FileInputStream;
import java.io.IOException;
import java.util.Collection;
import java.util.Collections;
@@ -66,6 +67,7 @@ import org.eclipse.jgit.api.Git;
import org.eclipse.jgit.internal.JGitText;
import org.eclipse.jgit.internal.storage.file.GC.RepoStatistics;
import org.eclipse.jgit.internal.storage.file.PackIndex.MutableEntry;
+import org.eclipse.jgit.internal.storage.pack.PackExt;
import org.eclipse.jgit.junit.LocalDiskRepositoryTestCase;
import org.eclipse.jgit.junit.RepositoryTestCase;
import org.eclipse.jgit.junit.TestRepository;
@@ -621,6 +623,40 @@ public class GCTest extends LocalDiskRepositoryTestCase {
}
@Test
+ public void testPruneOldPacksWithOpenHandleOnPack() throws Exception {
+ gc.setExpireAgeMillis(0);
+
+ BranchBuilder bb = tr.branch("refs/heads/master");
+ bb.commit().add("A", "A").add("B", "B").create();
+ fsTick();
+ gc.gc();
+
+ Collection<PackFile> packs = repo.getObjectDatabase().getPacks();
+ assertEquals(1, packs.size());
+ PackFile pack = packs.iterator().next();
+ File packFile = pack.getPackFile();
+ File indexFile = new File(packFile.getParentFile(), "pack-"
+ + pack.getPackName()
+ + "."
+ + PackExt.INDEX.getExtension());
+ FileInputStream fis = new FileInputStream(packFile);
+ try {
+ bb.commit().add("A", "A2").add("B", "B2").create();
+ fsTick();
+ gc.gc();
+ if (packFile.exists()) {
+ assertTrue(
+ "The pack was present but the index file was missing.",
+ indexFile.exists());
+ }
+
+ } finally {
+ fis.close();
+ }
+
+ }
+
+ @Test
public void testPackCommitsAndLooseOneWithPruneNowNoReflog()
throws Exception {
BranchBuilder bb = tr.branch("refs/heads/master");
diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/GC.java b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/GC.java
index 3e26bc3e62..9eaeaa8c14 100644
--- a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/GC.java
+++ b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/GC.java
@@ -175,21 +175,9 @@ public class GC {
*
* @param oldPacks
* @param newPacks
- * @param ignoreErrors
- * <code>true</code> if we should ignore the fact that a certain
- * pack files or index files couldn't be deleted.
- * <code>false</code> if an exception should be thrown in such
- * cases
- * @throws IOException
- * if a pack file couldn't be deleted and
- * <code>ignoreErrors</code> is set to <code>false</code>
*/
private void deleteOldPacks(Collection<PackFile> oldPacks,
- Collection<PackFile> newPacks, boolean ignoreErrors)
- throws IOException {
- int deleteOptions = FileUtils.RETRY | FileUtils.SKIP_MISSING;
- if (ignoreErrors)
- deleteOptions |= FileUtils.IGNORE_ERRORS;
+ Collection<PackFile> newPacks) {
oldPackLoop: for (PackFile oldPack : oldPacks) {
String oldName = oldPack.getPackName();
// check whether an old pack file is also among the list of new
@@ -200,10 +188,7 @@ public class GC {
if (!oldPack.shouldBeKept()) {
oldPack.close();
- for (PackExt ext : PackExt.values()) {
- File f = nameFor(oldName, "." + ext.getExtension()); //$NON-NLS-1$
- FileUtils.delete(f, deleteOptions);
- }
+ prunePack(oldName);
}
}
// close the complete object database. Thats my only chance to force
@@ -212,6 +197,42 @@ public class GC {
}
/**
+ * Delete files associated with a single pack file. First try to delete the
+ * ".pack" file because on some platforms the ".pack" file may be locked and
+ * can't be deleted. In such a case it is better to detect this early and
+ * give up on deleting files for this packfile. Otherwise we may delete the
+ * ".index" file and when failing to delete the ".pack" file we are left
+ * with a ".pack" file without a ".index" file.
+ *
+ * @param packName
+ */
+ private void prunePack(String packName) {
+ PackExt[] extensions = PackExt.values();
+ try {
+ // Delete the .pack file first and if this fails give up on deleting
+ // the other files
+ int deleteOptions = FileUtils.RETRY | FileUtils.SKIP_MISSING;
+ for (PackExt ext : extensions)
+ if (PackExt.PACK.equals(ext)) {
+ File f = nameFor(packName, "." + ext.getExtension()); //$NON-NLS-1$
+ FileUtils.delete(f, deleteOptions);
+ break;
+ }
+ // The .pack file has been deleted. Delete as many as the other
+ // files as you can.
+ deleteOptions |= FileUtils.IGNORE_ERRORS;
+ for (PackExt ext : extensions) {
+ if (!PackExt.PACK.equals(ext)) {
+ File f = nameFor(packName, "." + ext.getExtension()); //$NON-NLS-1$
+ FileUtils.delete(f, deleteOptions);
+ }
+ }
+ } catch (IOException e) {
+ // Deletion of the .pack file failed. Silently return.
+ }
+ }
+
+ /**
* Like "git prune-packed" this method tries to prune all loose objects
* which can be found in packs. If certain objects can't be pruned (e.g.
* because the filesystem delete operation fails) this is silently ignored.
@@ -533,7 +554,7 @@ public class GC {
if (rest != null)
ret.add(rest);
}
- deleteOldPacks(toBeDeleted, ret, true);
+ deleteOldPacks(toBeDeleted, ret);
prunePacked();
lastPackedRefs = refsBefore;