diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/snapshot/SnapshotFileCache.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/snapshot/SnapshotFileCache.java index 030cdabdc4b5..7bc931b41301 100644 --- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/snapshot/SnapshotFileCache.java +++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/snapshot/SnapshotFileCache.java @@ -22,6 +22,7 @@ import java.util.List; import java.util.Timer; import java.util.TimerTask; +import java.util.concurrent.TimeUnit; import java.util.concurrent.locks.Lock; import org.apache.commons.lang3.ArrayUtils; import org.apache.hadoop.conf.Configuration; @@ -97,6 +98,8 @@ Collection filesUnderSnapshot(final FileSystem fs, final Path snapshotDi private ImmutableMap snapshots = ImmutableMap.of(); private final Timer refreshTimer; + private static final int LOCK_TIMEOUT_MS = 30000; + /** * Create a snapshot file cache for all snapshots under the specified [root]/.snapshot on the * filesystem. @@ -192,39 +195,47 @@ public Iterable getUnreferencedFiles(Iterable files, if (snapshotManager != null) { lock = snapshotManager.getTakingSnapshotLock().writeLock(); } - if (lock == null || lock.tryLock()) { - try { - if (snapshotManager != null && snapshotManager.isTakingAnySnapshot()) { - LOG.warn("Not checking unreferenced files since snapshot is running, it will " - + "skip to clean the HFiles this time"); - return unReferencedFiles; - } - ImmutableSet currentCache = cache; - for (FileStatus file : files) { - String fileName = file.getPath().getName(); - if (!refreshed && !currentCache.contains(fileName)) { - synchronized (this) { - refreshCache(); - currentCache = cache; - refreshed = true; - } - } - if (currentCache.contains(fileName)) { - continue; + try { + if (lock == null || lock.tryLock(LOCK_TIMEOUT_MS, TimeUnit.MILLISECONDS)) { + try { + if (snapshotManager != null && snapshotManager.isTakingAnySnapshot()) { + LOG.warn("Not checking unreferenced files since snapshot is running, it will " + + "skip to clean the HFiles this time"); + return unReferencedFiles; } - if (snapshotsInProgress == null) { - snapshotsInProgress = getSnapshotsInProgress(); + ImmutableSet currentCache = cache; + for (FileStatus file : files) { + String fileName = file.getPath().getName(); + if (!refreshed && !currentCache.contains(fileName)) { + synchronized (this) { + refreshCache(); + currentCache = cache; + refreshed = true; + } + } + if (currentCache.contains(fileName)) { + continue; + } + if (snapshotsInProgress == null) { + snapshotsInProgress = getSnapshotsInProgress(); + } + if (snapshotsInProgress.contains(fileName)) { + continue; + } + unReferencedFiles.add(file); } - if (snapshotsInProgress.contains(fileName)) { - continue; + } finally { + if (lock != null) { + lock.unlock(); } - unReferencedFiles.add(file); - } - } finally { - if (lock != null) { - lock.unlock(); } + } else { + LOG.warn("Failed to acquire write lock on taking snapshot after waiting {}ms", + LOCK_TIMEOUT_MS); } + } catch (InterruptedException e) { + LOG.warn("Interrupted while acquiring write lock on taking snapshot"); + Thread.currentThread().interrupt(); // restore the interrupt flag } return unReferencedFiles; }