HBASE-27852: Interrupt BucketCachePersister thread when BucketCache is shutdown (#5230)
Signed-off-by: Wellington Chevreuil <wchevreuil@apache.org>
(cherry picked from commit e343584b50
)
This commit is contained in:
parent
026ff18053
commit
40a8a2a651
|
@ -173,6 +173,8 @@ public class BucketCache implements BlockCache, HeapSize {
|
|||
|
||||
private final BucketCacheStats cacheStats = new BucketCacheStats();
|
||||
|
||||
/** BucketCache persister thread */
|
||||
private BucketCachePersister cachePersister;
|
||||
private final String persistencePath;
|
||||
static AtomicBoolean isCacheInconsistent = new AtomicBoolean(false);
|
||||
private final long cacheCapacity;
|
||||
|
@ -366,8 +368,7 @@ public class BucketCache implements BlockCache, HeapSize {
|
|||
}
|
||||
|
||||
void startBucketCachePersisterThread() {
|
||||
BucketCachePersister cachePersister =
|
||||
new BucketCachePersister(this, bucketcachePersistInterval);
|
||||
cachePersister = new BucketCachePersister(this, bucketcachePersistInterval);
|
||||
cachePersister.setDaemon(true);
|
||||
cachePersister.start();
|
||||
}
|
||||
|
@ -1401,6 +1402,7 @@ public class BucketCache implements BlockCache, HeapSize {
|
|||
LOG.info("Shutdown bucket cache: IO persistent=" + ioEngine.isPersistent() + "; path to write="
|
||||
+ persistencePath);
|
||||
if (ioEngine.isPersistent() && persistencePath != null) {
|
||||
cachePersister.interrupt();
|
||||
try {
|
||||
join();
|
||||
persistToFile();
|
||||
|
|
|
@ -44,8 +44,11 @@ public class BucketCachePersister extends Thread {
|
|||
cache.persistToFile();
|
||||
cache.setCacheInconsistent(false);
|
||||
}
|
||||
} catch (IOException | InterruptedException e) {
|
||||
LOG.warn("Exception in BucketCachePersister" + e.getMessage());
|
||||
} catch (IOException e) {
|
||||
LOG.warn("IOException in BucketCachePersister {} ", e.getMessage());
|
||||
} catch (InterruptedException iex) {
|
||||
LOG.warn("InterruptedException in BucketCachePersister {} ", iex.getMessage());
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -29,10 +29,12 @@ import java.nio.file.Files;
|
|||
import java.nio.file.attribute.FileTime;
|
||||
import java.time.Instant;
|
||||
import java.util.Arrays;
|
||||
import org.apache.hadoop.conf.Configuration;
|
||||
import org.apache.hadoop.fs.Path;
|
||||
import org.apache.hadoop.hbase.HBaseClassTestRule;
|
||||
import org.apache.hadoop.hbase.HBaseTestingUtility;
|
||||
import org.apache.hadoop.hbase.io.hfile.BlockCacheKey;
|
||||
import org.apache.hadoop.hbase.io.hfile.CacheConfig;
|
||||
import org.apache.hadoop.hbase.io.hfile.CacheTestUtils;
|
||||
import org.apache.hadoop.hbase.io.hfile.Cacheable;
|
||||
import org.apache.hadoop.hbase.testclassification.SmallTests;
|
||||
|
@ -143,6 +145,44 @@ public class TestVerifyBucketCacheFile {
|
|||
TEST_UTIL.cleanupTestDir();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testRetrieveFromFileAfterDelete() throws Exception {
|
||||
|
||||
HBaseTestingUtility TEST_UTIL = new HBaseTestingUtility();
|
||||
Path testDir = TEST_UTIL.getDataTestDir();
|
||||
TEST_UTIL.getTestFileSystem().mkdirs(testDir);
|
||||
Configuration conf = TEST_UTIL.getConfiguration();
|
||||
conf.setLong(CacheConfig.BUCKETCACHE_PERSIST_INTERVAL_KEY, 300);
|
||||
|
||||
BucketCache bucketCache = new BucketCache("file:" + testDir + "/bucket.cache", capacitySize,
|
||||
constructedBlockSize, constructedBlockSizes, writeThreads, writerQLen,
|
||||
testDir + "/bucket.persistence", 60 * 1000, conf);
|
||||
|
||||
long usedSize = bucketCache.getAllocator().getUsedSize();
|
||||
assertEquals(0, usedSize);
|
||||
CacheTestUtils.HFileBlockPair[] blocks =
|
||||
CacheTestUtils.generateHFileBlocks(constructedBlockSize, 1);
|
||||
// Add blocks
|
||||
for (CacheTestUtils.HFileBlockPair block : blocks) {
|
||||
cacheAndWaitUntilFlushedToBucket(bucketCache, block.getBlockName(), block.getBlock());
|
||||
}
|
||||
usedSize = bucketCache.getAllocator().getUsedSize();
|
||||
assertNotEquals(0, usedSize);
|
||||
// Shutdown BucketCache
|
||||
bucketCache.shutdown();
|
||||
// Delete the persistence file
|
||||
final java.nio.file.Path mapFile =
|
||||
FileSystems.getDefault().getPath(testDir.toString(), "bucket.persistence");
|
||||
assertTrue(Files.deleteIfExists(mapFile));
|
||||
Thread.sleep(350);
|
||||
// Create BucketCache
|
||||
bucketCache = new BucketCache("file:" + testDir + "/bucket.cache", capacitySize,
|
||||
constructedBlockSize, constructedBlockSizes, writeThreads, writerQLen,
|
||||
testDir + "/bucket.persistence", 60 * 1000, conf);
|
||||
assertEquals(0, bucketCache.getAllocator().getUsedSize());
|
||||
assertEquals(0, bucketCache.backingMap.size());
|
||||
}
|
||||
|
||||
/**
|
||||
* Test whether BucketCache is started normally after modifying the cache file. Start BucketCache
|
||||
* and add some blocks, then shutdown BucketCache and persist cache to file. Restart BucketCache
|
||||
|
|
Loading…
Reference in New Issue